WebViewJavascriptBridge

一个用于在 WKWebViewsUIWebViewsDeprecated) 和 WebViews 中的 Obj-C 和 JavaScript 之间发送消息的 iOS/OSX 桥梁。
该框架实现了 Objective-C 和 JavaScript 方法的相互调用,还可以传参和响应回调。(iOS 中,WKWebView 的原生方法不支持回调!)

安装

通过 CocoaPods 安装,将以下代码添加到 podfile 文件中并运行 pod install 来安装。

pod 'WebViewJavascriptBridge', '~> 6.0'

用法

  1. 导入头文件并声明一个实例变量:
#import <WKWebViewJavascriptBridge.h>

...

@property (nonatomic, strong) WKWebViewJavascriptBridge *bridge;
  1. 使用 WKWebViewUIWebView(iOS)或 WebView(OSX)实例化 WKWebViewJavascriptBridge 对象。
self.bridge = [WKWebViewJavascriptBridge bridgeForWebView:self.webView];
  1. 注册一个 Objc 处理程序和一个要调用的 JS 处理程序。
// JS 调用 OC
// 注册一个让 JS 可以调用 OC 的方法
[self.bridge registerHandler:@"ObjC Echo" handler:^(id data, WVJBResponseCallback responseCallback) {
    NSLog(@"ObjC Echo called with: %@", data);
    responseCallback(data);
}];
// OC 调用 JS
[self.bridge callHandler:@"JS Echo" data:nil responseCallback:^(id responseData) {
    NSLog(@"ObjC received response: %@", responseData);
}];
  1. 复制和粘贴 setupWebViewJavascriptBridge 函数到你的 JS 代码中:
function setupWebViewJavascriptBridge(callback) {
    if (window.WebViewJavascriptBridge) { return callback(WebViewJavascriptBridge); }
    if (window.WVJBCallbacks) { return window.WVJBCallbacks.push(callback); }
    window.WVJBCallbacks = [callback];
    var WVJBIframe = document.createElement('iframe');
    WVJBIframe.style.display = 'none';
    WVJBIframe.src = 'https://__bridge_loaded__';
    document.documentElement.appendChild(WVJBIframe);
    setTimeout(function() { document.documentElement.removeChild(WVJBIframe) }, 0)
}
  1. 最后,调用 setupWebViewJavascriptBridge 方法,并使用 bridge 来注册 JS 处理程序和要调用的 Objc 处理程序:
setupWebViewJavascriptBridge(function(bridge) {
    
    /* Initialize your app here */
        // OC 调用 JS
        // 在 JS 中可以注册让 OC 调用的方法
    bridge.registerHandler('JS Echo', function(data, responseCallback) {
        console.log("JS Echo called with:", data)
        responseCallback(data)
    })
        // JS 调用 OC 的方法,方法名就是 OC 中提前注册的方法
    bridge.callHandler('ObjC Echo', {'key':'value'}, function responseCallback(responseData) {
        console.log("JS received response:", responseData)
    })
})

API 文档

Objc API

[WKWebViewJavascriptBridge bridgeForWebView:(WKWebView*)webView];

该方法传递一个给定的 WKWebView 参数,让 Objective-C 和 JavaScript 之间建立桥梁。


[bridge registerHandler:(NSString*)handlerName handler:(WVJBHandler)handler];

该方法会在 Objc 中注册一个名为 handlerName 的处理程序。JavaScript 就可以通过 WebViewJavascriptBridge.callHandler("handlerName") 来调用这个处理程序。

示例:

[self.bridge registerHandler:@"getScreenHeight" handler:^(id data, WVJBResponseCallback responseCallback) {
    responseCallback([NSNumber numberWithInt:[UIScreen mainScreen].bounds.size.height]);
}];
[self.bridge registerHandler:@"log" handler:^(id data, WVJBResponseCallback responseCallback) {
    NSLog(@"Log: %@", data);
}];

- (void)callHandler:(NSString*)handlerName;
- (void)callHandler:(NSString*)handlerName data:(id)data;
- (void)callHandler:(NSString*)handlerName data:(id)data responseCallback:(WVJBResponseCallback)responseCallback;

以上方法调用名为 handlerName 的 JavaScript 方法。如果指定了 responseCallback block 块,那么 JavaScript 还可以接收回调。

示例:

[self.bridge callHandler:@"showAlert" data:@"Hi from ObjC to JS!"];
[self.bridge callHandler:@"getCurrentPageUrl" data:nil responseCallback:^(id responseData) {
    NSLog(@"Current UIWebView page URL is: %@", responseData);
}];

[bridge setWebViewDelegate:(id)webViewDelegate];

可选,如果你需要响应 web view's lifecycle events,你也可以设置 WKNavigationDelegate/UIWebViewDelegate 代理。


[bridge disableJavscriptAlertBoxSafetyTimeout];

不安全。通过禁用 setTimeout 安全检查来加速桥接消息的传递。仅当您不调用任何 javascript 弹出框功能(警告,确认和提示)时,才能禁用此安全检查。如果您从桥接的 javascript 代码中调用这些功能中的任何一个,则该应用将挂起。

JavaScript API

bridge.registerHandler("handlerName", function(responseData) { ... })

在 JavaScript 上注册一个名为 handlerName 的处理程序,Objc 就可以通过:

- (void)callHandler:(NSString*)handlerName;
- (void)callHandler:(NSString*)handlerName data:(id)data;
- (void)callHandler:(NSString*)handlerName data:(id)data responseCallback:(WVJBResponseCallback)responseCallback;

这三个实例方法来调用 JS 函数。

示例:

bridge.registerHandler("showAlert", function(data) { alert(data) })
bridge.registerHandler("getCurrentPageUrl", function(data, responseCallback) {
    responseCallback(document.location.toString())
})

bridge.callHandler("handlerName", data)
bridge.callHandler("handlerName", data, function responseCallback(responseData) { ... })

这两个是 JavaScript 调用 Objc 方法的方法,如果 Objc 中注册时设置了 responseCallback,那么 JavaScript 就可以接受回调(即调用 Objc 方法时,Objc 会有返回值返回给 JavaScript)。

示例:

bridge.callHandler("Log", "Foo")
bridge.callHandler("getScreenHeight", null, function(response) {
    alert('Screen height:' + response)
})

bridge.disableJavscriptAlertBoxSafetyTimeout()

在 JavaScript 中调用 bridge.disableJavscriptAlertBoxSafetyTimeout() 方法和在 Objc 中调用 [bridge disableJavscriptAlertBoxSafetyTimeout]; 是同样的效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容