1. 程式人生 > >iOS 模仿支付寶支付到賬推送,播報錢數

iOS 模仿支付寶支付到賬推送,播報錢數

最近申請了支付寶的二維碼收錢碼,其中支付寶有這麼一個功能,就是,別人掃描你的二維碼給你轉賬之後,收到錢會有一條語音推送,”支付寶到賬 1000萬“之類的推送訊息,不管你的支付寶app有沒有被殺死。

只要你的遠端推送開著,並且支付寶的"二維碼收錢到賬語音提醒",都開啟著,就可以收到。

開啟方式:支付寶點選右上角設定-通用-新訊息通知,開啟到賬提醒即可。

image.pngimage.png

並且別人給你轉多少錢就會播報到賬多少錢。

探索實現一下。

當前有兩種方案實現了上面描述的場景。

必備條件
上面的描述場景只有在iOS10以上版本才可以,因為必須要基於Notification Servivice Extension

image.pngimage.png

實現方式
1、ServiceExtension中收到推送之後,用AVSpeechSynthesisVoice相關類,直接把推送過來需要播報相關的文字轉化成語音播報
2、ServiceExtension中收到推送之後,將要播報的數字,找到對應的單個音訊,排序,用拼接音訊的方式<通過推送過來的文字去查詢相關的音訊,然後拼接成一個音訊>,然後使用AudioServicesCreateSystemSoundID播放

在介紹相關方式之前,先介紹一個測試工具
SmartPush

使用方式也很簡單,應該一看就懂

image.png

正式進入主題
方式一

@implementation NotificationService
 
- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent * _Nonnull))contentHandler {
    self.contentHandler = contentHandler;
    self.bestAttemptContent = [request.content mutableCopy];
 
    // Modify the notification content here...
    self.bestAttemptContent.title = [NSString stringWithFormat:@"%@ [modified]", self.bestAttemptContent.title];
 
    self.contentHandler(self.bestAttemptContent);
 
 
    [self playVoiceWithAVSpeechSynthesisVoiceWithContent:self.bestAttemptContent.body];
 
}
 
- (void)playVoiceWithAVSpeechSynthesisVoiceWithContent:(NSString *)content
{
    if (content.length == 0) {
        return;
    }
    // 建立嗓音,指定嗓音不存在則返回nil
    AVSpeechSynthesisVoice *voice = [AVSpeechSynthesisVoice voiceWithLanguage:@"zh-CN"];
 
    // 建立語音合成器
    AVSpeechSynthesizer *synthesizer = [[AVSpeechSynthesizer alloc] init];
 
    // 例項化發聲的物件
    AVSpeechUtterance *utterance = [AVSpeechUtterance speechUtteranceWithString:content];
    utterance.voice = voice;
    utterance.rate = 0.5; // 語速
 
    // 朗讀的內容
    [synthesizer speakUtterance:utterance];
}

推送的內容是:

{
  "aps":{
    "alert":{
      "title":"iOS 10 title",
      "subtitle":"iOS 10 subtitle",
      "body":"世上只有媽媽好,有媽的孩子像塊寶。投進媽媽的懷抱,幸福哪裡找。沒媽的孩子像根草"
    },
    "my-attachment":"http://img01.taopic.com/160317/240440-16031FU23937.jpg",
    "mutable-content":1,
    "category":"myNotificationCategory1",
    "badge":3
 
  }
}

坑點
說明:當前實現的是將push內容中的body播放出來
1、如果你收到推送了但是添加了系統的鈴聲,也就是你在push的json中添加了"sound":"default"那麼就可能會影響推送聲音的播放
2、收到推送了,但是沒有播報語音,檢查一下這裡

image.png
3、播放的聲音時間長度,經過測試最多是5秒鐘,這裡應該是蘋果做了限制,拿上面的推送內容舉例子"body":"世上只有媽媽好,有媽的孩子像塊寶。投進媽媽的懷抱,幸福哪裡找。沒媽的孩子像根草",最多也就是播放到世上只有媽媽好,有媽的孩子像塊寶。投進媽媽的懷抱

方式二
經過對比,支付寶播放的聲音明顯比系統方法文字轉語音播放的好聽,一聽就是小姑娘錄得。要麼就是自己集成了一套文字轉語音的東西。
首先嚐試使用科大訊飛來實現,結果失敗了。
然後嘗試的是語音合成的方式來播放
比如提前先錄好 以下可能播報的內容

支付寶到賬、 0、 1、 2、 3、 4、 5、 6、 7、 8、 9、 十、 百、 千、 萬、 十萬、 百萬、 千萬、 億、 元 等等

這樣的幾種錄音,然後用相關的名字命名好<相關的規則自己命名就好>。
比如push過來的是內容是 10010,那麼轉化成的錄音檔名稱的陣列就是
@[@"支付寶到賬",@"1",@"萬",@"0",@"1",@"十",@"元"]
然後找到這幾個檔案,然後按照順序拼接成一個語音檔案進行播放

程式碼演示:

@implementation NotificationService
 
static int lianxunPlay = 1;
 
- (void)didReceiveNotificationRequest:(UNNotificationRequest *)request withContentHandler:(void (^)(UNNotificationContent * _Nonnull))contentHandler {
    self.contentHandler = contentHandler;
    self.bestAttemptContent = [request.content mutableCopy];
 
    // Modify the notification content here...
    self.bestAttemptContent.title = [NSString stringWithFormat:@"%@ [modified]", self.bestAttemptContent.title];
 
 
    self.contentHandler(self.bestAttemptContent);
 
    // 方式4,語音合成,使用AudioServicesPlayAlertSoundWithCompletion播放,成功,但是時間最多5秒
    [self hechengVoice];
 
}


- (void)hechengVoice
{
    /************************合成音訊並播放*****************************/
    NSMutableArray *audioAssetArray = [[NSMutableArray alloc] init];
    NSMutableArray *durationArray = [[NSMutableArray alloc] init];
    [durationArray addObject:@(0)];
 
    AVMutableComposition *composition = [AVMutableComposition composition];
 
    NSArray *fileNameArray = @[@"daozhang",@"1",@"2",@"3",@"4",@"5",@"6"];
 
 
    CMTime allTime = kCMTimeZero;
 
    for (NSInteger i = 0; i < fileNameArray.count; i++) {
        NSString *auidoPath = [[NSBundle mainBundle] pathForResource:[NSString stringWithFormat:@"%@",fileNameArray[i]] ofType:@"m4a"];
        AVURLAsset *audioAsset = [AVURLAsset assetWithURL:[NSURL fileURLWithPath:auidoPath]];
        [audioAssetArray addObject:audioAsset];
 
        // 音訊軌道
        AVMutableCompositionTrack *audioTrack = [composition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:0];
        // 音訊素材軌道
        AVAssetTrack *audioAssetTrack = [[audioAsset tracksWithMediaType:AVMediaTypeAudio] firstObject];
 
 
        // 音訊合併 - 插入音軌檔案
        [audioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, audioAsset.duration) ofTrack:audioAssetTrack atTime:allTime error:nil];
 
        // 更新當前的位置
        allTime = CMTimeAdd(allTime, audioAsset.duration);
 
    }
 
    // 合併後的檔案匯出 - `presetName`要和之後的`session.outputFileType`相對應。
    AVAssetExportSession *session = [[AVAssetExportSession alloc] initWithAsset:composition presetName:AVAssetExportPresetAppleM4A];
    NSString *outPutFilePath = [[self.filePath stringByDeletingLastPathComponent] stringByAppendingPathComponent:@"xindong.m4a"];
 
    if ([[NSFileManager defaultManager] fileExistsAtPath:outPutFilePath]) {
        [[NSFileManager defaultManager] removeItemAtPath:outPutFilePath error:nil];
    }
 
    // 檢視當前session支援的fileType型別
    NSLog(@"---%@",[session supportedFileTypes]);
    session.outputURL = [NSURL fileURLWithPath:outPutFilePath];
    session.outputFileType = AVFileTypeAppleM4A; //與上述的`present`相對應
    session.shouldOptimizeForNetworkUse = YES;   //優化網路
 
    [session exportAsynchronouslyWithCompletionHandler:^{
        if (session.status == AVAssetExportSessionStatusCompleted) {
            NSLog(@"合併成功----%@", outPutFilePath);
 
            NSURL *url = [NSURL fileURLWithPath:outPutFilePath];
 
            static SystemSoundID soundID = 0;
 
            AudioServicesCreateSystemSoundID((__bridge CFURLRef _Nonnull)(url), &soundID);
 
            AudioServicesPlayAlertSoundWithCompletion(soundID, ^{
                NSLog(@"播放完成");
            });
 
 
 
        } else {
            // 其他情況, 具體請看這裡`AVAssetExportSessionStatus`.
        }
    }];
 
    /************************合成音訊並播放*****************************/
}

說明:
上面並沒有實現 數字轉對應音訊檔名稱陣列的過程,直接實現的是合成音訊的方法。

坑點:
1、播放的時長仍然受到限制,大概5秒鐘。不過如果說播放一個錢數,足夠了。
2、合成之後的音訊檔案用AVAudioPlayer播放是沒有聲音的

分析:
以上的功能只是針對iOS10以上的系統版本可以,那麼iOS10以下的怎麼辦?可以這麼辦,不用播報到賬多少錢,可以通過定製遠端推送的語音,來播報”支付寶,您有一筆到賬,請及時檢視“之類的,支付寶好像也是這個套路。

每次push之前,先去後臺檢視當前需要推送的裝置的系統版本是啥<這個不難實現>,然後定製推送不同的內容。iOS10以上的就推送錢數,並且不推送sound。
iOS10以下的就推送 "sound"="定製的聲音檔名稱"。

綜上所述,支付寶實現的方式應該是方式二,本地合成音訊檔案播放的。

還有一個注意點就是,因為打開了下面的這個開關

image.pngimage.png
再上線的時候可能需要作出一下說明,不然有很大的可能被蘋果打回來,最近蘋果對這種許可權的開啟之類的比較嚴格,如果你沒有類似的功能,你還開啟了這樣的許可權,可能被幹回來。如果有相關的好的像蘋果說明的方法,還請給大家普及一下。

最後獻上相關的Demo地址,如果你有更好的建議歡迎留言,如有不正,歡迎來噴。

可以直接用我的Demo進行除錯,除錯的時候注意修改下bundleId,然後用自己的開發者賬號配置一下相關的push證書就可以了

image.png

image.png

支付寶

iOS

Apple

單元測試