iOS小技能:iOS14 读取用户剪切板数据弹出提示的兼容方案

语言: CN / TW / HK

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第12天,点击查看活动详情

前言

背景:在iOS14中,如果APP读取剪切版的内容时,手机会弹出提示,提示哪个APP在获取剪切板内容。

读取UIPasteboardstringstringsURLURLsimageimagescolorcolors的时候会触发系统提示。 使用hasStringshasURLshasImageshasColors等方法的时候不会触发系统提示。

  • 读取剪贴板的代码 在这里插入图片描述

像这种第三方SDK,及时联系对应的SDK开发者。比如极光的v3.3.6就及时进行了适配

在这里插入图片描述

在这里插入图片描述

划重点

  • 先判断剪切板内容的各式,如果符合规则才读取剪切板
  • 查找哪些SDK使用了剪切板,及时升级SDK。

比如发现了JCore SDK在iOS 14引用剪贴板,该行为导致APP被用户怀疑隐私泄露,请予以重视

I、弹出提示的兼容方案:尽可能少的去调用会触发系统提示的方法

弹出提示的原因:使用 UIPasteboard 访问用户数据

方案一:先判断剪切板内容的各式,如果符合规则采取读取。

例如app读取口令时判断是否符合数字和链接的规则

iOS14 剪切板适配案例分析:读取口令的实现方法

http://kunnan.blog.csdn.net/article/details/108894751

方案二:使用changeCount来记录剪切板的数据是否发生变化

1.1 先判断剪切板内容的各式,如果符合规则才读取剪切板(例如淘宝的淘口令)

如果应用仅仅访问只访问URL格式的剪切板内容,或者特定规则的内容,比如淘口令,就可以使用API先判断,确实是符合规则的时候再去读取

判断是否为URL格式: UIPasteboardDetectionPatternAPI

``` typedef NSString * UIPasteboardDetectionPattern NS_TYPED_ENUM API_AVAILABLE(ios(14.0));

/// NSString value, suitable for implementing "Paste and Go" UIKIT_EXTERN UIPasteboardDetectionPattern const UIPasteboardDetectionPatternProbableWebURL API_AVAILABLE(ios(14.0));

/// NSString value, suitable for implementing "Paste and Search" UIKIT_EXTERN UIPasteboardDetectionPattern const UIPasteboardDetectionPatternProbableWebSearch API_AVAILABLE(ios(14.0));

/// NSNumber value UIKIT_EXTERN UIPasteboardDetectionPattern const UIPasteboardDetectionPatternNumber API_AVAILABLE(ios(14.0)); ```

下面 API 可以获得具体的 URL 信息,但是会触发剪切板提示。并且实测当用户剪切板中包含多个 URL 时只会返回第一个。

``` // Detection

/// Detects patterns in the first pasteboard item. /// /// @param patterns Detect only these patterns. /// @param completionHandler Receives which patterns were detected, or an error. - (void)detectPatternsForPatterns:(NSSet )patterns                 completionHandler:(void(^)(NSSet * _Nullable,                                            NSError * _Nullable))completionHandler NS_REFINED_FOR_SWIFT API_AVAILABLE(ios(14.0));              /// Detects patterns in the specified pasteboard items. /// /// @param patterns Detect only these patterns. /// @param itemSet Specifies which pasteboard items by their position. Nil means all items. /// @param completionHandler Receives which patterns were detected per item specified, ///                          or an error. - (void)detectPatternsForPatterns:(NSSet )patterns                         inItemSet:(NSIndexSet * _Nullable)itemSet                 completionHandler:(void(^)(NSArray *> * _Nullable,                                            NSError * _Nullable))completionHandler NS_REFINED_FOR_SWIFT API_AVAILABLE(ios(14.0));                   ```

  • 例子

``` NSSet *patterns = [[NSSet alloc] initWithObjects:UIPasteboardDetectionPatternProbableWebURL, nil];

[[UIPasteboard generalPasteboard] detectPatternsForPatterns:patterns completionHandler:^(NSSet * _Nullable result, NSError * _Nullable error) {     if (result && result.count) {             // 当前剪切板中存在 URL                 } }]; ```

1.2 使用changeCount来记录剪切板的数据是否发生变化


记录一下真正读取剪切板时的changeCount,如果下次读取的时候没有发生变化则不读取。 这样一来效果就好多了,应用运行生命周期内,基本上只会弹出一次提示

II 、 查找哪些SDK使用了剪切板

app出现了从后台唤起会弹出 读取剪切板的 提示,安全部门要求要赶紧查,全局搜索了code,发现并没有读取剪切板的代码,比窦娥还冤,只能是第三方SDK的了,

2.1 Symbolic breakpoint 进行查找


  • 用Xcode的Symbolic breakpoint,调试[UIPasteboard generalPasteboard]

在这里插入图片描述 在这里插入图片描述

使用例子:

  • 从后台唤起app会有这个提示
  • 单步向下走几步,然后就会出现调用方

在这里插入图片描述

在这里插入图片描述

立马发现了JCore iOS SDK在iOS 14引用剪贴板该行为会导致APP被用户怀疑隐私泄露,请予以重视

升级新版进行适配 iOS 14。

pod 'JPush' , '3.3.6'

2.2 使用命令行查找哪些SDK使用了剪切板

  • grep -r "UIPasteboard" . 在这里插入图片描述

➜  retail git:(develop) grep -r "UIPasteboard" . ./retail/other/tool/CopyTextLabel/QCTCopyTextLabel.m:    UIPasteboard * paste = [UIPasteboard generalPasteboard]; ./retail/class/business/Merchant_self_recording开户申请/oldNewAddStores/NewAddStores/QCTCopyLabel.m:    UIPasteboard * paste = [UIPasteboard generalPasteboard]; ./retail/class/business/Merchant_self_recording开户申请/oldNewAddStores/NewAddStores/QCTStoresDetailViewController.m:    [UIPasteboard generalPasteboard].string = titleLab.text ? titleLab.text : @""; Binary file ./Pods/JCore/libjcore-ios-2.3.2.a matches ./Pods/QMUIKit/QMUIKit/QMUIComponents/QMUILabel.m:        UIPasteboard *pasteboard = [UIPasteboard generalPasteboard]; Binary file ./Pods/JPush/libjpush-ios-3.3.4.a matches

  • find . -type f | grep -e ".framework" | xargs grep -s UIPasteboard

```

➜  Housekeeper git:(develop) find . -type f | grep -e ".framework" | xargs grep -s UIPasteboard

Binary file ./Housekeeper/other/extension/iOS/thirdparties/thirdparties_ios_1.0.5/UTDID.framework/Versions/A/UTDID matches Binary file ./Housekeeper/other/extension/iOS/common/common_ios_2.1.1/normal/UMCommon.framework/Versions/A/UMCommon matches ```

  • find . -type f | grep -e ".a" | xargs grep -s AUPasteboard

III 、例子:升级JPush SDK

pod 'JPush' , '3.3.6'

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

IV、 see also

在这里插入图片描述

在这里插入图片描述

4.1 升级CocoaPods(适配Xcode12)

  • 升级采用gem即可,命令如下

sudo gem install cocoapods

发现版本1.9.1 与Xcode12 不是很匹配,于是就升级新版本看下。

➜  Housekeeper git:(develop) pod --version 1.9.1

如果没升级,主要问题是,如果采用pod install --verbose --no-repo-update 之后,Xcode的编译设置不会 自动修改other Linker Flags

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

升级到1.9.3 再次升级SDK的时候,就适配Xcode12,可以自动修改other Linker Flags参数了

在这里插入图片描述

在这里插入图片描述

因此升级Xcode的时候,最好也同时升级下CocoaPods