標籤:
iOS的AVFoundation架構提供了基本的音視頻播放工具,我們基本上可以靠其中提供的類完成絕大部分的音視頻播放任務。但是在音頻播放的輸出音量的處理上,蘋果的策略比較保守。儘管AVPlayer和AVPAudiolayerzhe這些類提供了音量大小功能,但這些音量控制屬於App層級的控制。好處就是音量大小獨立於系統音量,調節大小時不會影響系統音量。但有時候我們可能希望修改系統音量,以免在調節聲音的時候,如果系統音量過小,App調節音量效果不明顯。一般來說要調節系統音量會有以下方法:
請注意:修改系統音量無法在模擬器上看到效果,必須使用真機調試才能看到效果!
使用
MPVolumeView
這個方法是蘋果官方推薦的方法。MPVolumeView是Media Player Framework中的一個UI組件,直接包含了對系統音量和Airplay裝置的音頻鏡像路由的控制功能。其中包含一個MPVolumeSlider的subview用來控制音量。這個MPVolumeSlider是一個私人類,我們無法手動建立此類,但這個類是UISlider的子類。MPVolumeView的使用很簡單,只需要將其加入到一個父視圖中,給予父視圖合適的大小,再建立MPVolumeView樣本,將其加入到父視圖中即可,蘋果官方的文檔1中有範例程式碼可以參考。
這個方法的缺點如下:
如果還想給UI加入手勢操作來控制音量,這種直接使用MPVolumeView是做不到的,那麼有沒有什麼方法可以繞過這限制呢?辦法還是有的。
編程實現系統音量大小2
上一小節我們提到了MPVolumeView這個組件中,有一個subview來控制音量,即MPVolumeSlider。其實我們可以通過遍曆MPVolumeView執行個體的subviews來得到MPVolumeSlider的執行個體,從而通過這個UI組件來作業系統音量。
通過
MPVolumeSlider的執行個體來作業系統音量
我們首先通過建立一個MPVolumeView,然後遍曆找出MPVolumeSlider的執行個體。這個執行個體提供setValue:animated:方法來設定系統音量。我們也可以通過volumeSlider.value這個屬性來擷取當前的系統音量。具體的代碼如下:
MPVolumeView *volumeView = [[MPVolumeView alloc] init];UISlider* volumeViewSlider = nil;for (UIView *view in [_instance.volumeView subviews]){ if ([view.class.description isEqualToString:@"MPVolumeSlider"]){ volumeViewSlider = (UISlider*)view; break; }}// retrieve system volumefloat systemVolume = volumeViewSlider.value;// change system volume, the value is between 0.0f and 1.0f[volumeViewSlider setValue:1.0f animated:NO];// send UI control event to make the change effect right now.[volumeViewSlider sendActionsForControlEvents:UIControlEventTouchUpInside];
上面的代碼示範如何擷取和修改系統音量,注意音量取值為0到1之間的浮點數。
有問題!我不喜歡系統彈出音量提示
上面通過編程的方法可以很完美的調節系統音量,但是每次修改都會彈出系統提示框告知:
有時候這種提示我們未必會需要,那麼怎麼取消掉這個提示呢?實際上MPVolumeView沒有提供任何介面來調節是否需要顯示系統音量提示。但是我們發現一點:當MPVolumeView處在當前視圖的層級之中時,系統就不會顯示音量提示。那麼事情好辦了,我們只要確保兩點:
MPVolumeView視圖處在螢幕上看不見的地方,比如某個不透明視圖的下方,或者本視圖的非可見地區,一個常見的做法就是把該視圖的frame設定為地區以外的地方,比如volumeView.frame = CGRectMake(-1000, -100, 100, 100);
確保MPVolumeView視圖的hidden屬性值為NO。因為當hidden為YES時,同樣會彈出提示。
還有問題,我修改了系統音量但是不是通過我的UI
另一個可能的情況就是使用者自己通過硬體的音量大小按鈕(位於裝置側邊)來調節音量,這種情況會使得你的商務邏輯出現問題,因為你只為自己的App UI寫了回調,那麼怎麼為硬體按鈕的事件添加回調呢?我們可以使用Notification Center來完成。
這裡只需要監聽AVSystemController_SystemVolumeDidChangeNotification事件即可。具體代碼如下:
NSError *error;
// Active audio session before you listen to the volume change event.
// It must be called first.
// The old style code equivalent to the line below is:
//// AudioSessionInitialize(NULL, NULL, NULL, NULL);
// AudioSessionSetActive(YES);
//// Now the code above is deprecated in iOS 7.0, you should use the new
// code here.
[[AVAudioSession sharedInstance] setActive:YES error:&error];
// add event handler, for this example, it is `volumeChange:` method
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(volumeChanged:) name:@"AVSystemController_SystemVolumeDidChangeNotification" object:nil];
- (void)volumeChanged:(NSNotification *)notification
{
// service logic here.
}
- (void)dealloc
{
[[NSNotificationCenter defaultCenter] removeObserver:self name:@"AVSystemController_SystemVolumeDidChangeNotification" object:nil];
}
這樣,每次使用者使用硬體按鈕調節音量的時候也會執行你寫好的邏輯。
以上除了第一個方案以外,所有的解決方案都屬於非官方的hack性質的方法,但是都沒有調用私人API,所以沒有被Apple審核拒掉的風險。
iOS調節系統音量