iOS 14 pasted from anothor device 的 Tips

在 iOS 14 开发者预览版发布后,第一时间下载尝鲜了,最大的变化就是 app 访问剪贴板在顶部有一个提示:

image.png

pasted from anothor device / pasted from xxx

怎么触发的

出于好奇这个检测是怎么触发的,通过一步步 debug 最发现在(一开始还分析错了,感谢页面仔指正):
-[PBServerConnection requestItemFromPasteboardWithName:UUID:authenticationMessage:itemIndex:typeIdentifier:completionBlock:] 这个方法,伪代码如下:

NSXPCConnection *xpcConnection = [self serverConnection];
void (^loggingErrorHandler)(NSError *) = _loggingErrorHandler;
 __NSXPCInterfaceProxy_PBClientToServerProtocol *proxy = [xpcConnection remoteObjectProxyWithErrorHandler:loggingErrorHandler];
 [proxy requestItemFromPasteboardWithName:UUID:authenticationMessage:itemIndex:typeIdentifier:completionBlock:
    ... callback
]

一旦执行 proxy 的 request 方法,就使用 XPC 发送消息,将请求参数和回调一起发送到系统的 pasteboard 进程来请求数据,当检测如果你粘贴的数据不是来源你的 app 就会弹出提示,由于这个校验和提醒的逻辑并不在 app 进程内,在应用层是没有办法绕过检测拿到数据的,apple 还是很严谨的。

如何查看是哪个方法导致了提示

Xcode 打一个符号断点-[PBServerConnection requestItemFromPasteboardWithName:UUID:authenticationMessage:itemIndex:typeIdentifier:completionBlock:] 当然了如果你嫌太长,下面这个也可以:
-[PBItemRepresentation _loadCompletionBlock:]

如何减少弹提示

  1. 尽量减少 Direct access 的方法调用,这类方法在 UIPasteboard.h 有列出来,比如 [[UIPasteboard generalPasteboard] items]
    [[UIPasteboard generalPasteboard] itestringss],可以先用 Queries 的方法做一些判,比如 [[UIPasteboard generalPasteboard] hasStrings] ,另外如果有频繁访问剪贴板的需求,最好监听 UIPasteboardChangedNotification 通知,在剪贴板发生改变才去读取。

  2. iOS 14 提供了新的 API 用来检测特定类型的剪贴板数据:

- (void)detectValuesForPatterns:(NSSet *)patterns
              completionHandler:(void(^)(NSDictionary * _Nullable,
                                         NSError * _Nullable))completionHandler

不过目前的类型只有 2 个,而且不能自定义 Pattern:
UIPasteboardDetectionPatternProbableWebURL 用于检测剪贴板的 URL 数据
UIPasteboardDetectionPatternProbableWebSearch 用于检测剪贴板可能为搜索关键词的数据

比如你是一个浏览器应用,在用户激活地址输入框的时候,你就可以用 UIPasteboardDetectionPatternProbableWebURL 来访问剪贴板读取为 URL 的数据,如果此时没有 URL 也不会弹框提醒,这样可以避免每次直接读剪贴板而导致频繁弹框。

既然有这个特性,那么 H5 利用剪贴板来传递数据,也可以改成 URL 的形式,把要传递的内容放在 URL 的参数里,这样 app 可以检测到有 URL 类型才真正访问剪贴板读取 URL 并解析出内容。

3.通过 pasteboardWithName:create: 实例化的剪贴板数据访问是不会触发弹框提醒的,所以应用间的数据传递暂时不受影响(很多 SDK 都是这么干的)

你可能感兴趣的:(iOS 14 pasted from anothor device 的 Tips)