iOS 替换系统音量提示视图

实现过程

实现自定义音量提示视图,需要做到以下几步:
1.激活AudioSession配置。
2.隐藏系统的音量提示视图。
3.监听音量按钮的触发事件。

一、激活AudioSession配置

[[AVAudioSession sharedInstance] setActive:YES error:nil];

需要注意的点是:当APP切换到前台时,还需要进行激活AudioSession配置,否则会出现收不到KVO的通知场景。

- (void)private_addWillEnterForegroundNotification {
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(didReceiveWillEnterForegroundNotification:)
                                                 name:UIApplicationWillEnterForegroundNotification
                                               object:nil];
}

- (void)didReceiveWillEnterForegroundNotification:(NSNotification *)notification {
    [[AVAudioSession sharedInstance] setActive:YES error:nil];
}

二、隐藏系统的音量提示视图

创建MPVolumeView并将其添加到当前可见的视图层中,同时将其 frame 设置到不可见区域。

- (void)onHiddenSystemVolumeView {
        MPVolumeView *mpVolumeView = [[MPVolumeView alloc] init];
        if (@available(iOS 12.0, *)) { //解决卡顿仅处理12以上设备
            id controller = [mpVolumeView valueForKey:@"lightweightRoutingController"];
           //解决MPVolumeView导致的卡顿
            if (controller && [controller isKindOfClass:NSClassFromString(@"MPAVLightweightRoutingController")] && [controller respondsToSelector:@selector(setDelegate:)])
            {
                [controller performSelector:@selector(setDelegate:) withObject:nil];
            }
        }
        [mpVolumeView setFrame:CGRectMake(-240, -100, 200, 40)];
}

三、监听音量按钮的触发事件

监听系统音量按钮的触发事件,有两种方式可以做到,各有优劣,下面来做一个简单对比。
音量按钮每触发一次,变化量都是 6.25%,连续按16次,即可调节至最大或最小。

3.1.KVO方式

通过KVO监听[AVAudioSession sharedInstance]的outputVolume属性,然后来显示自定义的 UI 控件。这种方式有一个不好的地方就是:在音量调节至最大/最小时,这个时候再调大/调小音量,由于outputVolume的值不变,所以不会触发KVO,也就无法展示自定义音量视图。代码大概长下面这样:

- (void)addObserver {
    [[AVAudioSession sharedInstance] addObserver:self
                                      forKeyPath:NSStringFromSelector(@selector(outputVolume))
                                         options:NSKeyValueObservingOptionNew
                                         context:nil];
}

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary<NSKeyValueChangeKey,id> *)change
                       context:(void *)context {
    if ([change isKindOfClass:[NSDictionary class]]) {
        NSNumber *volumeNum = change[@"new"];
        if (volumeNum) {
            [self volumeDidChange:[volumeNum floatValue]];
        }
    }
}

- (void)volumeDidChange:(CGFloat)volume {
    // 显示自定义音量提示
}

- (void)dealloc {
    [[AVAudioSession sharedInstance] removeObserver:self 
                                         forKeyPath:NSStringFromSelector(@selector(outputVolume))];
}
3.2.通知

这种方式通过监听系统私有(未公开的)通知,名字是AVSystemController_SystemVolumeDidChangeNotification,这个监听不会受到最大/最小音量时,调大/调小音量的影响,只要音量键按下,始终都会触发。但是这个通知由于是私有的,可能存在被拒风险,而且将来系统版本该通知名字发生改变,由于是硬编码而不像其它系统通知使用的是常量,会导致监听不到的问题。代码大概长这样:

static NSNotificationName const kSystemVolumeDidChangeNotification = @"AVSystemController_SystemVolumeDidChangeNotification";

- (void)addObserver {
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(volumeDidChange:)
                                                 name:kSystemVolumeDidChangeNotification
                                               object:nil];
}

- (void)volumeDidChange:(NSNotification *)notification {
    NSString *category = notification.userInfo[@"AVSystemController_AudioCategoryNotificationParameter"];
    NSString *changeReason = notification.userInfo[@"AVSystemController_AudioVolumeChangeReasonNotificationParameter"];
    if (![category isEqualToString:@"Audio/Video"] || ![changeReason isEqualToString:@"ExplicitVolumeChange"]) {
        return;
    }

    CGFloat volume = [[notification userInfo][@"AVSystemController_AudioVolumeNotificationParameter"] floatValue];
    // 显示自定义音量提示
}

- (void)dealloc {
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

以上两种方式各自优劣势都已经列出来了,Instagram使用的是通知的方式,哔哩哔哩、腾讯视频都是用KVO的方式。如果在最大或最小时,调节音量可以接受不展示音量视图的话,个人推荐使用KVO的形式。

4、创建自己的音量提示视图

为了调用统一且音量视图层级永远在最上方(即不被Alert等挡住),自定义的音量视图使用UIWindow来实现,然后自定义视图和系统的视图加到这个视图层级上。由于自己创建的UIWindow的hidden属性默认是YES,所以需要手动将其设成NO。

直接使用UIWindow这种方式不太行,会导致其它地方取keyWindow的时候会取错,导致其他问题。最终自定义一个Window继承自UIWindow,然后复写becomeKeyWindow方法,在这个方法里让自身不成为keyWindow同时将[UIApplication sharedApplication].delegate.window设置为keyWindow,大致代码长这样:

@interface VolumeWindow : UIWindow

@end

@implementation VolumeWindow

- (void)becomeKeyWindow {
    [self resignKeyWindow];
    [[UIApplication sharedApplication].delegate.window makeKeyWindow];
}

@end

5、自定义后的问题

问题一:
拍摄页拍摄之前系统音量提示是可以被替换的,但是拍摄一段之后,莫名其妙音量按钮按下后自定义提示不见了,出现了系统的铃声提示。一脸懵逼,后面发现是由于设置了UIAVCaptureSession的usesApplicationAudioSession为NO,会导致在拍摄之后会变成铃声,这个和是否替换系统音量提示无关。这个属性是由于项目中很久之前需要兼容iOS6,然后一直遗留着这个属性设置没有删除。由于iOS7之后,AVCaptureSession和应用使用的是同一个AudioSession,支持同时播放和录制且不会受到影响和打断,所以不需要再去设置这个属性。

问题二:
做了以上操作,在 iPhoneX 下,当拉起控制中心,并上下滑调整音量后,再回到应用,会发现自定义音量视图会出现在状态栏下面,猜测虽然在应用内自定义音量视图window层级高于状态栏window层级,但是由于状态栏是全局的,在重新进入到应用时会出现状态栏层级高于音量视图。所以就索性仅在应用为active的情况下才处理KVO。

最后一个需要注意的点是在语音电话(或者其它使用系统音量的场景下)时,去自己应用内调节音量是无效,因为这个时候音量其实代表的是系统在占用,系统优先级高于应用,所以在这些场景下,即使在应用内调节音量,也无法触发出自己的音量视图。

最后推荐一个VolumeBar开源库:https://github.com/gizmosachin/VolumeBar

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容