美文网首页
WebViewJavascriptBridge实现原理

WebViewJavascriptBridge实现原理

作者: 亲爱的八路 | 来源:发表于2020-02-26 22:48 被阅读0次

    关于WebViewJavascriptBridge的功能不多做介绍,有兴趣的小伙伴可以搜一下

    使用示例

    iOS端使用如下

    let bridge = WKWebViewJavascriptBridge(for: webView)
    
    //原生调用js
    bridge.callHandler("jsFunction", data: [:], responseCallback: { (obj) in
    })
    
    //native端注册方法,以供js调用
    bridge.registerHandler("nativeFunction") { (data, reponseCallback) in
    }
    
    

    实现原理:

    js端有一个处理对象,window.WebViewJavascriptBridge对象,原生端也有一个类似的对象WebViewJavascriptBridgeBase,他们两个在两端,分别使用拦截url和注入js的方式,进行通信,他们的主要工作内容是

    • 存储自己端registed handler(实质是个block数组)

    • 存储回调(实质也是个block数组)

    • 对需要通信数据进行编码和解码

    //js端存储相关代码
    var messageHandlers = {}; //存储registerHandler方法注册过的block
    var responseCallbacks = {}; //存储callHandler中的responseCallback
    
    //原生端存储相关
    @interface WebViewJavascriptBridgeBase : NSObject
    @property (strong, nonatomic) NSMutableDictionary* responseCallbacks; //存储callHandler中的responseCallback
    @property (strong, nonatomic) NSMutableDictionary* messageHandlers; //存储registerHandler方法注册过的block
    @end
    

    初始化过程

    原生端的初始化:原生端调用WebViewJavascriptBridge库,生成需要的WebViewJavascriptBridge或者WKWebViewJavascriptBridge实例

    js端的初始化:js端加载地址为“https://__bridge_loaded__”的新页面,新页面的加载事件触发webview的代理方法

    //WKWebView
    - (void)webView:(WKWebView *)webView decidePolicyForNavigationAction:(WKNavigationAction *)navigationAction decisionHandler:(void (^)(WKNavigationActionPolicy))decisionHandler;
    
    //UIWebView
    - (BOOL)webView:(UIWebView *)webView shouldStartLoadWithRequest:(NSURLRequest *)request navigationType:(UIWebViewNavigationType)navigationType;
    

    在代理方法中识别加载地址为“https://__bridge_loaded__”,调用WebViewJavascriptBridge库中预先写好的一段js代码 — WebViewJavascriptBridge_JS,使用evaluateJavaScript执行这段js代码,在js端生成window.WebViewJavascriptBridge对象及一些需要的js方法。

    这种交互方式,比较通用的叫法是 url sheme

    原生调用js

    原生调用js的callHandler方法,实际上是调用WebViewJavascriptBridgeBase中方法对参数进行处理。

    WebViewJavascriptBridgeBase会为此次调用的回调生成一个id,并在responseCallbacks中用ID为key存储回调。然后把js方法名、参数、回调id 组织成字典,再json字符串化。

    {
      "data": ["ID": 67788],
      "callbackId": "objc_cb_1",
      "handlerName": "jsFunction"
    }
    

    然后用“WebViewJavascriptBridge._handleMessageFromObjC”把这个message字符串包起来。_handleMessageFromObjC的是初始化时通过WebViewJavascriptBridge_JS注入js中的一个js方法。接下来通过evaluateJavaScript执行handleMessageFromObjC。

    NSString* javascriptCommand = [NSString stringWithFormat:@"WebViewJavascriptBridge._handleMessageFromObjC('%@');", messageJSON];
    

    js端的bridge方法handleMessageFromObjC被调用后,先将json字符串解析成字典对象,然后根据字典中的handlerName值,在messageHandlers中寻找注册过的对应name的function,将message.data和js回调传给该function,进行调用。

    if (message.callbackId) {
        var callbackResponseId = message.callbackId;
        responseCallback = function(responseData) {
            _doSend({ handlerName:message.handlerName, responseId:callbackResponseId, responseData:responseData });
        };
    }
                    
    var handler = messageHandlers[message.handlerName];
    if (!handler) {
        console.log("WebViewJavascriptBridge: WARNING: no handler for message from ObjC:", message);
    } else {
        handler(message.data, responseCallback);
    }
    

    注意一点,上面的responseCallback是js自己的回调,原生的回调还留在WebViewJavascriptBridge的responseCallbacks字典中存储。js回调的功能是给原生发送回调消息。

    js回调调用_doSend方法中把字典类型的message添加进sendMessageQueue数组

    [handlerName: xxxx, responseId: xxx, responseData: xxx] //iOS 字典
    {handlerName: xxxx, responseId: xxx, responseData: xxx} //js 字典
    

    然后加载一个地址为“https://__wvjb_queue_message__”的新页面,原生的代理方法被调用

    WebViewJavascriptBridge中的代理方法识别到特殊的加载地址,使用evaluateJavaScript调用“WebViewJavascriptBridge._fetchQueue();”,js端的_fetchQueue把sendMessageQueue数组中的message都格式化成json字符串。原生拿到messages的json字符串后先把字符串转换成数组,然后遍历数组处理每个message(通常只有一个)。原生bridge发现message中有responseId,则用responseId调用对应的responseCallback。如果没有responseId,说明这不是 原生调js - js回调原生,而是js直接调用原生的方法,那就从messageHandlers中寻找对应handlerName的block

    在原生端初始化WebViewJavascriptBridge的时候,WebViewJavascriptBridge就把webview的代理指向了自己,所以这里能拦截到。如果webview的代理被指向了其他地方,那WebViewJavascriptBridge就不工作了。毕竟js端初始化就指望着这个代理方法拦截到请求,然后去给js注入window.WebViewJavascriptBridge对象。

    js调用原生

    前面原生调用js中说到了原生发消息给js,js怎么处理,原生的回调怎么处理。js调用原生是类似的逻辑。只是双方发消息的方式不一样,一个是通过evaluateJavaScript,一个是通过url sheme。这里不再多述。

    相关文章

      网友评论

          本文标题:WebViewJavascriptBridge实现原理

          本文链接:https://www.haomeiwen.com/subject/ynbgchtx.html