OC 与 JS 交互之 WKWebView

上一篇讲了借助 JavaScriptCore 进行 OC 与 JS 的交互,链接请戳⬇️
OC 与 JS 交互之 JavaScriptCore
今天总结一下通过 WKWebView 来进行 OC 与 JS 交互~

WKWebView

WKWebView 是苹果爸爸在 iOS8 推出的希望替代 UIWebView 的WebView,推出了很多新的功能,新的属性,并且通过 WKScriptMessageHandler、WKNavigationDelegate、WKUIDelegate 这三个协议,让请求、执行 web 页面的过程变得更加可控。

与之交互用到的三大代理:
WKNavigationDelegate:与页面导航加载相关
WKUIDelegate:与 JS 交互时的 UI 展示相关,比较 JS 的alert、confirm、prompt
WKScriptMessageHandler:与 JS 交互相关,通常是 iOS 端注入名称,JS 端通过 window.webkit.messageHandlers.{NAME}.postMessage()来发消息到 iOS 端

关于 WKWebView 的更多细节有时间再总结下,今天先说说有关 OC 与 JS 交互的过程。

OC 调用 JS
WKWebViewConfiguration *configuration = [WKWebViewConfiguration new];
configuration.userContentController = [WKUserContentController new];

NSString *sourceStr = @"alert('在载入webview时通过oc注入的js方法');";
WKUserScript *script = [[WKUserScript alloc] initWithSource:sourceStr injectionTime:WKUserScriptInjectionTimeAtDocumentStart forMainFrameOnly:YES];
[configuration.userContentController addUserScript:script];

_webView = [[WKWebView alloc] initWithFrame:self.view.bounds configuration:configuration];
_webView.UIDelegate = self;

生成的 configuration ,包含了之后新建 webView 的一系列属性;再生成一个脚本片段,包含 JS 的 alert 语句,将其加给 configuration 的 userContentController,在开始加载 web 页面时,注入 JS 方法,执行这个 JS 语句。
或者用 webView 的 evaluateJavaScript 直接调用 JS,与 JavaScriptCore 中的 evaluateScript 类似:

NSString *script = @"alert('webview 直接调用 JS 方法');";    
[self.webView evaluateJavaScript:script completionHandler:^(id _Nullable result, NSError * _Nullable error) {
    NSLog(@"result: %@, error: %@", result, error);
}];

由于苹果爸爸要求用 native 的方式处理 web 中的弹框,这里就需要
controller 实现 UIDelegate 中的方法,拿到 JS 中的参数,生成弹框,提醒用户:

- (void)webView:(WKWebView *)webView runJavaScriptAlertPanelWithMessage:(NSString *)message initiatedByFrame:(WKFrameInfo *)frame completionHandler:(void (^)(void))completionHandler
{
    UIAlertController *alertController = [UIAlertController alertControllerWithTitle:@"Tip" message:message preferredStyle:UIAlertControllerStyleAlert];
    [alertController addAction:[UIAlertAction actionWithTitle:@"Ok" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
        completionHandler();
    }]];
    [self presentViewController:alertController animated:YES completion:nil];
}

效果如图:

在载入WebView时,执行 JS 方法

UIDelegate 有不同的协议方法分别对应 native 确定、选择、输入文本框的等弹框,效果还不错~

JS 调用 OC

先新建一个 html 文件如下:

<!DOCTYPE html>
<html>
    <head>
        <title>test javascript</title>
        <style type="text/css">
            * {font-size: 40px;}
        </style>
    </head>    
    <body>        
        <div style="margin-top: 100px">
            <input type="button" value="js call oc" onclick="jsCallOC()">
        </div> 
        <script type="text/javascript">
            function jsCallOC(){
                var message = {
                    'method' : 'jsCallOC',
                    'param1' : 'zn',
                };
            window.webkit.messageHandlers.AppModel.
            postMessage({body: message});
            }
        </script>
    </body>
</html>

点击按钮,调用 JS 方法,在 JS 中
window.webkit.messageHandlers.{NAME}.postMessage() 将 JS 中的对象转换为 OC 中的处理消息对象,并且将 message 传递给他。

//将 self 作为 AppModel ,处理 JS 传递过来的数据
[configuration.userContentController addScriptMessageHandler:self name:@"AppModel"];

拿到 message 后,self 需要继承 WKWebView 的另一个协议 WKScriptMessageHandler 的方法,解析消息,从而实现 JS 到 OC 的调用:

// MARK: - WKScriptMessageHandler
// 注入 JS 名称,在 JS 端通过 window.webkit.messageHandlers.AppModel.postMessage() 方法来发送消息到 native
- (void)userContentController:(WKUserContentController *)userContentController didReceiveScriptMessage:(WKScriptMessage *)message
{
    NSLog(@"messageBody:  %@", message.body);
    NSLog(@"messageName: %@", message.name);
}
JS 调用 OC 方法

参考博客:
WKWebView
WKWebView特性及使用

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容