Swift. Notification Service Extension实现语音播报

总的来说,实现收款到账语音合成播报,真的是走了很长的路,踩过很多的坑。避免更多的人重复踩坑,我便记录走过哪些路,掉过哪些坑...

一. 先来说第一种,不推荐再用。因为苹果粑粑不会给你审核通过的

1.推送就不说了,我们使用的是极光推送
2.语音合成使用的是 AVSpeechSynthesizer
  • 为了app退到后台进程不被杀掉,target -> capabilities中开启了background modes中的audio选项。
    WechatIMG2.jpeg
  • 在语音合成前需要添加下面代码。
    在applicationWillResignActive函数里
//  先注册响应后台控制
UIApplication.shared.beginReceivingRemoteControlEvents()

// 语音合成前需要添加

// 设置并激活音频会话类别
do {
    try AVAudioSession.sharedInstance().setCategory(AVAudioSessionCategoryPlayback, with: .duckOthers)
} catch {
    print(error.localizedDescription)
}
do {
    try AVAudioSession.sharedInstance().setActive(true)
} catch {
    print(error.localizedDescription)
}

// 语音合成在推送回调里调用的

// 调用API
SpeechManager.shared.speechWeather(with: alert as! String)
// 实现主要代码
// 自定义语音播报方法
// 此处只例播报一个String的情况
    func speechWeather(with weather: String) {
        if let _ = speechUtterance {
            synthesizer.stopSpeaking(at: .immediate)
        }
        
        //播放声音
        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?.delegate = self
        audioPlayer?.prepareToPlay()
        audioPlayer?.play()
        
        speechUtterance = AVSpeechUtterance(string: weather)
        //语音类型
        speechUtterance?.rate = 0.5
        synthesizer.speak(speechUtterance!)
    }
3. 遇到的问题。

1.看似完成了app前台和后台的语音播报,但是app退到后台手动杀掉进程就gg了。

2.还有审核时,由于开启了background modes中的audio选项。拒绝的原因是audio应用的场景不对,
审核被拒.png

二. 最重要的是Notification Service Extension(拓展推送服务)。

1. 在项目中创建notificationservice, 选择 file -> new -> target -> Notification Service Extension
选择NotificationService.jpeg
2. 点击next。
  • product name: 拓展项目服务的名称。
  • bundle identifier: 是原app项目的bundle identifier+拓展项目服务的名称。(原app的bundle identifier为com.test,拓展项目名称为.myService,则拓展项目的bundle identifier为com.test.myService)
    拓展工程.jpeg

创建完成在scheme中会多一个myService

WechatIMG7.jpeg

3.相关配置。
  • 1 首先后端要给你个推送测试内容,内容模板如下:
    要记住在aps里一定要有"mutable -content"这个字段,alert 这个用字符串就可以,不用字典。当然字典也行,后面可以获取里面字符串也行
    后端配置.png
  • 2 在service工程里info.plist中添加App Transport Security Settings并且添加 Allow Arbitrary Loads为YES。


    配置.jpeg
4.主要代码如下。
//
//  NotificationService.swift
//  MyService
//
//  Created by JunQiang on 2018/5/24.
//  Copyright © 2018年 多飞. All rights reserved.
//

import UserNotifications
import AVFoundation

class NotificationService: UNNotificationServiceExtension {

    var contentHandler: ((UNNotificationContent) -> Void)?
    var bestAttemptContent: UNMutableNotificationContent?
    var audioPlayer: AVAudioPlayer?

    override func didReceive(_ request: UNNotificationRequest, withContentHandler contentHandler: @escaping (UNNotificationContent) -> Void) {
        self.contentHandler = contentHandler
        bestAttemptContent = (request.content.mutableCopy() as? UNMutableNotificationContent)
        
        if let bestAttemptContent = bestAttemptContent {
            // Modify the notification content here...
//            bestAttemptContent.title = "收款到账"
            do {
                try AVAudioSession.sharedInstance().setCategory(AVAudioSessionCategoryPlayback, with: .duckOthers)
            } catch {
                print(error.localizedDescription)
            }
            do {
                try AVAudioSession.sharedInstance().setActive(true)
            } catch {
                print(error.localizedDescription)
            }
            //播放收金币声音
            bestAttemptContent.sound = UNNotificationSound.init(named: "sound.wav")

            //语音合成
            let aVSpeechSynthesizer = AVSpeechSynthesizer()
            let aps = bestAttemptContent.userInfo["aps"]
            let str = (aps as! NSDictionary)["alert"];

            let aVSpeechUtterance = AVSpeechUtterance(string: str as! String)
            aVSpeechUtterance.rate = AVSpeechUtteranceDefaultSpeechRate
            aVSpeechUtterance.voice = AVSpeechSynthesisVoice.init(language: "zh-CN")
            aVSpeechSynthesizer.speak(aVSpeechUtterance)
            
            contentHandler(bestAttemptContent)
        }
    }
    
    override func serviceExtensionTimeWillExpire() {
        // Called just before the extension will be terminated by the system.
        // Use this as an opportunity to deliver your "best attempt" at modified content, otherwise the original push payload will be used.

        if let contentHandler = contentHandler, let bestAttemptContent =  bestAttemptContent {
            contentHandler(bestAttemptContent)
        }
    }
}
4. 支持iOS10以上的手机,修改service中的target -> Deployment Target 选择10.0。
image.png

注意:播放收金币的音频时。
一开始考虑的是AVAudioPlayer播放本地音频。后来发现只有第一次收到推送是有播放sound.wav,其他时候是有语音合成未有播放sound.wav。

//播放声音
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?.delegate = self
audioPlayer?.prepareToPlay()
audioPlayer?.play()

经过一番搜索,才发现bestAttemptContent本身是可以播放音频的。改完发现可以了

//播放收金币声音
bestAttemptContent.sound = UNNotificationSound.init(named: "sound.wav")

到此结束,总结不好多多指教....谢谢

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

推荐阅读更多精彩内容

  • 用到的组件 1、通过CocoaPods安装 2、第三方类库安装 3、第三方服务 友盟社会化分享组件 友盟用户反馈 ...
    SunnyLeong阅读 14,609评论 1 180
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,881评论 25 707
  • 今天在家看了部心理学方面的旧片《催眠大师》,想从以下两个方面来探讨着这部电影: 1、 为什么心理医生徐瑞宁不能治疗...
    水波儿阅读 3,273评论 0 4
  • 导 语 我很有家庭责任感,具体表现就是第一时间租房子和黄怡同居。当然,我答应了黄怡的父母:“给我两年,一定能够赚到...
    飘雨桐V阅读 277评论 1 1
  • 阿多尼斯说: "童年就是一座村庄,你永远也走不出它的边际"。打小就一直呆在老家,小小的村庄是我世界的主体,但越长大...
    阿丁h阅读 466评论 2 7