美文网首页临时收藏iOS技术iOS技术资料
iOS,推送+后台语音播报,推送+程序杀死仍语音播报,看这一篇就

iOS,推送+后台语音播报,推送+程序杀死仍语音播报,看这一篇就

作者: CoderRocker_Axl | 来源:发表于2017-09-04 14:24 被阅读1138次

第一篇文章,如果有写的不正确的、或者让人看不懂的地方,还希望大家能指正出来,共同进步,好了下面开始正题。

哇,好久不写文章真的不知道从何说起了。

一个多月前公司app改版,产品希望能加入语音播报的功能。在我们用户收到推送的时候,动态的播报出语音的内容,类似支付宝的“成功收款100元”这样的信息。由于时间紧迫当时也是在网上搜了很多的方法,参考了很多大神的文章。
http://www.jianshu.com/p/c06133d576e4

这一篇是对我帮助很大的一篇文章,当时就是按照文中的方法对推送进行了处理,可以在app前台、后台挂起的情况下进行语音播报。瑕疵就是在app被杀死的情况下无法播报。有时在后台,当用户打开qq音乐后,也不能播报(这可能是我写的问题,不能在音频被独占后再次播放)。而且审核有可能被拒绝。
我们改版后第一次提交审核并没有被拒绝,当时还是很开心的,因为如果因为后台播放音频被拒,还得录视频,还得再次审核,这个时间成本真的是耽误不起。
之后的第二版就悲剧了,果断因为这个被拒绝,我发了邮件解释了我们的用法,但是还是被拒绝了。同时我们产品也对语音播报的不稳定提出了很大的意见:支付宝程序杀死都能播报,为什么我们不能?还有那么多情况播不了??你还能不能行了,是不是想被祭天!?

当时我内心是崩溃的

但同时也激起了我的好奇心,为什么程序杀死的情况下也能对推送进行处理?这是我以前没接触过的。不过我相信,别人能做出来的,我一定也能做出来!于是马上投入了研究,最后终于找出了比较完美的办法来实现这样的功能。下面让我们进入正题吧!

需求分析

一、语音合成

语音合成是这个功能比较简单的部分,苹果已经提供了一个功能强大的语音类AVSpeechSynthesis。如果你对于声音没有什么特殊要求,AVSpeechSynthesis 是一个很好的工具。在此就不在对AVSpeechSynthesis 做赘述了。下面是一个写的比较完整的文章,大家可以参考一下。
http://www.jianshu.com/p/a41cb018f0b5

二、远程推送

这是本次功能的难点,在前台收到推送并根据内容用AVSpeechSynthesis去合成语音是一个很简单的功能,但是如何在程序进入后台、程序杀死的情况下执行代码,这是我们要研究的一个问题。在网上搜了很多的资料后,终于找到了解决这个问题的主角 ------ Notification Service Extension,通知服务扩展。

三、Notification Service Extension 通知服务扩展

这是iOS 10.0推出的新功能。
https://developer.apple.com/documentation/usernotifications/unnotificationserviceextension
上面是官方文档的连接,按我的理解总结下来一句话,就是他能让你在收到远程推送的时候先对推送内容进行预处理,处理完之后,再让你的app进行处理。而且这部分代码是和你本身app隔离开的(这个在接下来的代码环节会有个直观的认识),这样就给了我们单独处理相应推送的地方。下面就让我们具体的看一下怎么操作。

简单实现

一、为项目添加Notification Service Extension

首先打开你的项目 File ----> New ------> Target

屏幕快照 2017-09-04 下午1.22.39.png

选择Notification Service Extension

屏幕快照 2017-09-04 下午1.23.09.png

进行命名。这里Bundle identifier之类的内容,xcode会自动为你配置好,所以不用担心。其实从这里可以看出来,这个target并不属于你的app里面的一部分,而是另一个全新的程序,但是他是和你的app绑定的,这样,当推送来的时候iphone就知道究竟是谁的推送过来,需不需要进行额外处理。

屏幕快照 2017-09-04 下午1.24.13.png

Finish之后,你就可以在你的工程里看到你app的Notification Service Extension了。需要注意的是,因为是两个完全独立的target所以,你原有项目里的自己写的类,或原有项目里的资源文件,在Notification Service Extension里是完全访问不到的(打包之后也是两个完全独立的bundle)。所以如果你想要使用项目里的资源或者文件,你需要拖到Notification Service Extension目录里面,才可以使用。

屏幕快照 2017-09-04 下午1.51.04.png

二、对推送内容进行预处理

接下来就是业务代码了,在生成的NotificationService.m文件里对推送来的payload进行处理,在这里,你可以进行一些操作,例如修改推送的内容。要注意的是,并不是所有的推送都会走这个额外的方法。必须是会弹出alert、并且payload里面要设置"mutable-content"字段的值为1,才会进入这个方法,这都是需要跟你们后台沟通的。
{
"aps": {
"alert": "This is some fancy message.",
"badge": 1,
"sound": "default",
"mutable-content": "1",
}
}

屏幕快照 2017-09-04 下午1.26.32.png

在我们拿到需要播报的字段后,用AVSpeechSynthesis进行播放就可以啦。而且在处理之后,你的app是仍然可以收到该推送,而且得到的数据是你在NotificationService.m里修改过的数据(如果你修改了的话)。所以注意不要处理两遍哦。

屏幕快照 2017-09-04 下午2.12.22.png

调试的时候需要注意一点,如果你想走NotificationService.m里面的断点,需要将你Scheme调成对应的target

屏幕快照 2017-09-04 下午1.51.19.png

好啦,整个功能的简单实现到这里就结束啦,注意这个方法只有在10.0以后才可以用哦。
Notification Service Extension还有很多很强大的功能,大家可以继续发掘。
有什么建议或者疑问都可以留言,大家共同进步哇~~

相关文章

网友评论

  • 爱吃肉的兔子ZH:ios12 收到推送后不能没有声音,ios11以下可以,请问大神是什么情况啊?
    CoderRocker_Axl:@FAQ哥 12.0.1?我们这边播报没出问题
    耀敬业:大兄弟, 同问, 你的问题解决了吗
    02bba7953f3c:我也碰到iOS12收到推送后没有语音播报的问题 之前版本都是正常的 断点断了看到语音播报代码执行了 但是声音没有 请问你解决了吗
  • 凨弥:学到了 我用了puthKit 这个是App杀死也可以后台唤醒App,可以走自己的代码,需要通知的时候,可以发送本地通知,不需要的时候,直接播报。也可以做更新或数据同步操作,随意发挥,还没上线,不是道审核是否容易。
    凨弥:@CoderRocker_Axl 还没上线呢 这个出来好久了 应该不是很严吧 说明给清楚。 @ai张小张
    7c91ae2f8077:@凨弥请问用pushKit通过审核了吗
    CoderRocker_Axl::+1: 审核过了么兄弟
  • 相沫_:语音合成没问题!但是播放收金币的mp3,为什么只有第一次可以播放.mp3,其他时候都不播放?大神你有碰到过吗??//播放声音
    let path = Bundle.main.path(forResource: "sound", ofType: "wav")
    let pathURL = NSURL(fileURLWithPath: path!)
    do {
    audioPlayer = try AVAudioPlayer(contentsOf: pathURL as URL)
    } catch {
    audioPlayer = nil
    }
    audioPlayer?.prepareToPlay()
    audioPlayer?.play()
    相沫_:@CoderRocker_Axl 已经解决了!谢谢了。使用 [UNNotificationSound soundNamed:@"6414.mp3"];解决的
    CoderRocker_Axl:不太清楚,试试等播放mp3结束再播语音?
  • laoyao666:我这边研究了一下Background Modes中Audio,AirPlay,and Picture in Picture这一项不需要打钩也可以实现后台语音播报,后台语音播报好像与网络有关(App downloads content from the network),也就是说Background Modes中只需要勾选后两项就可以了。 这样的话提交版本的时候是不是无需向苹果提交后台语音播报的视频了?
    Arvin_雾里看花:这个问题我今天也研究了一下,是不需要勾选第一项的,我用的voip push方式。
    laoyao666:@CoderRocker_Axl恩恩
    CoderRocker_Axl:如果按照我文中方式写的话不需要勾选,因为本质上就不是在后台播报语音,而是用一个新的程序去播报,不存在后台不后台的问题。你如果勾选了Background Modes审核时候一般都会问你哪儿用到了后台模式,怎么用的。
  • 酒疯酒疯:大神你好,我也在做语音播报,但是出现了一些问题。求助一下!我的qq:824792205
  • sennnnn:大神你好,按照您的流程好像还是不可以进行杀死之后语音播报功能,我的QQ494313298,不知道我哪里没设置好能否帮忙看下呢.
    酒疯酒疯:大神你好,我也在做语音播报,但是出现了一些问题。求助一下!我的qq:824792205
    CoderRocker_Axl:加你了
    sennnnn:您好现在出现的是闪退问题...后台杀死点击通知栏推送任务就闪退.
  • 62149484ff2b:楼主,这种方式,锁屏收语音消息,通知栏会有通知,想实现通知栏不展示通知,voip好像可以
    CoderRocker_Axl:可以,研究研究。但voip审核起来比较麻烦。
  • Operation:楼主,这种方式,锁屏收款,通知栏会有通知,怎么才能不让他发那条通知呢
    Operation:@CoderRocker_Axl 我试过了,还是有消息退出来,但是内容是空的:joy:
    CoderRocker_Axl:不过突然想到如果在这个方法里面吧title和subtitle都置为空会是什么效果,一会儿试试
    CoderRocker_Axl:你是不是我们产品经理派来的奸细!他就一直想去掉。。经过反复测试得出的结论是不行。因为推送的alert里面必须要有内容才会走这个extension,因为苹果本意是给你提供方法去修改通知栏内容的。个人觉得没必要去掉,都已经有声音提醒了,多个通知栏也没什么大不了的吧。
  • 铃兰_900f:楼主,这么写可以通过苹果审核吗?
    铃兰_900f:@CoderRocker_Axl 谢谢,审核通过了
    CoderRocker_Axl:可以,没问题。
  • 073cf0e533a5:刚开始还有声音,后来不知道怎么没了,断点也不走.很郁闷 QQ:569303378
    CoderRocker_Axl:@年华冷了 不需要
    073cf0e533a5:@CoderRocker_Axl 使用这总方式审核时需要什么声明吗
    CoderRocker_Axl:对,有时候调试断点就是不走,我推断这种情况一般是NotificationService.m里的程序报错了。有可能是你语音合成的过长导致的。
  • 京北磊哥:先谢谢楼主的 分享。 能分享一个demo吗?
    CoderRocker_Axl:倒数第二张图就是全部代码了,推送要有相应的开发者账号,所以demo的意义其实不大。。
  • CoderRocker_Axl:忘了写了,因为是独立的程序,所以不论你app是在前台还是后台,还是被杀死了,这部分代码都会被执行。
    CoderRocker_Axl:@薛定谔的黑猫警长 在播放完成的回调里面调用contenthandler
    薛定谔的黑猫警长:我的这个播放声音时间很短,当上面通知栏消失,声音也就没有了,怎么处理呢大神 ?

本文标题:iOS,推送+后台语音播报,推送+程序杀死仍语音播报,看这一篇就

本文链接:https://www.haomeiwen.com/subject/blzsnttx.html