前言
最近两天公司有需求研究CallKit来电标签和电话拉黑,在网上找了一些相关博文。但是发现对于没接触过这方面的人来说,还是会遇到一些问题,所以打算自己再写一篇详尽的文章。
懒得看的请直接下载最下面demo。
细节
在没有了解CallKit这两个功能之前,因为下面的内容,导致产生一些失(cao)败(dan)的尝试。包括但不限于以下:
- 如果该号码已存在手机里,标记识别将会无效
- 如果屏蔽某号码,看不到该号码来电
- 打开主App和扩展App的数据共享,以实现两个应用共享数据
- 添加到call表单中的电话号码必须带上国家区号
- 添加号码到表单之前,需要对数据进行去重和升序排列
- 最大数量在100万到200万之间,原因可能是超时无效
- 目前好像只能全量更新数据,不能增量更新,效率比较低
过程
首先,建立一个工程或者在原有工程上,加入扩展应用 Call Directory Extension:
- 选择new->target->Call Directory Extension
-
在这个文件中,苹果默认已经写好了注入事件的代码,并默认添加了两个手机号供参考格式。
-
正是因为这俩默认号码的格式,容易产生误导,如果按照这个格式来写的话,号码是无法添加到 CXCallDirectoryExtensionContext 里面的,也就是会添加失败,导致产生了一种“这不可能”的心态。
- 在做这里的时候,我想当然的认为,它只是一个本地沙盒或者数据库类型的介质,也就是任何形式的字符数据都可以存储进去(CXCallDirectoryPhoneNumber是int64_t),实际上,它在存储前进行了数据格式判断,不带 #国家区号# 的数据是无法存储成功的,直接失败并导致权限丢失。
权限打开方式:设置->电话->来电阻止与身份识别
- 我尝试了一下是否可以用openUrl方式,直接让应用跳转到来电身份权限页面。遗憾的是失败了,可能是没有开放这个接口,或者我没找到。第一行跳转到电话设置页面有效。
Phone — prefs:root=Phone
Phone — prefs:root=Phone&CALL_BLOCKING&IDENTIFICATION
接着,进行Main App和Extension App之间数据通讯。
- 测试功能时,直接在扩展应用 Call Directory Extension 中写入了自己的号码尝试成功。不过,扩展APP和主APP虽然在同一个工程里,但是是两个应用,传递数据的话需要用到数据共享:App Group.
- App共享数据的方法有两种,至于写入数据的方式,可以采用数组、字典或者数据库。
1 使用共享url地址,写入文件
NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.RexLocalGroup"];
2 使用本地沙盒,写入沙盒中
NSUserDefaults *myDefaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.RexLocalGroup"];
- 简书上有人采用 fgets(char *buf, int bufsize, FILE *stream) 函数读行的形式来解析数据,我试了一下,可以实现存入100万条数据,但200万的时候会异常,应该是苹果做了一些限制。不过这种方式,只适合从服务器拉取数据文件写入,如果想在移动端去添加数据到文件里,就会非逻辑化。所以如果想实现移动端主动写入的话,用数据库会容易操作一些。
NSURL *containerURL = [[NSFileManager defaultManager] containerURLForSecurityApplicationGroupIdentifier:@"group.RexLocalGroup"];containerURL = [containerURL URLByAppendingPathComponent:@"Library/"];NSString * dbPath = [NSString stringWithFormat:@"%@%@",containerURL.absoluteString, @"LocalNumberDB.sqlite"];_dbQueue = [FMDatabaseQueue databaseQueueWithPath:dbPath];
- 主应用中把数据库建立到这个数据共享URL位置,然后,在扩展应用中提取数据并填入CXCallDirectoryExtensionContext表单中。这样就实现了数据传递。比较遗憾的是目前只能是全量更新号码库,无法实现增量更新,所以更新一条数据也需要全部重新存一遍,速度比较慢,可能这也是种隐私保护。
NSArray * contacts = [[FMDataBaseManager shareInstance] getAllContacts:kNumberTable];for (int i= 0; i < contacts.count; i ++) {@autoreleasepool {ContactModel * contact = contacts[i];if (contact.phoneNumber && contact.identification) {CXCallDirectoryPhoneNumber phoneNumber = [contact.phoneNumber longLongValue];NSString * label = contact.identification;[context addIdentificationEntryWithNextSequentialPhoneNumber:phoneNumber label:label];}contact = nil;}}
- 简单几步,即可感觉到这个强(mei)大(luan)的(yong)功能的作用。
有疑问可以加Q:12087014。
参考项目下载
做了一个简单完整的demo放在了GitHub上,有兴趣可以下载。
[link]https://github.com/Rex-xingjl/CallKitDemo