H5如何与原生App(ios,安卓,RN)通信?

使用iframe方式,以唤起Native;以唤起分享组件为例

// h5 js code 将它封装一下

function createIframe(url){

var url = ‘jsbridge://getShare?title=分享标题&desc=分享描述&link=http%3A%2F%2Fwww.douyu.com&cbName=jsCallClientBack’;

var iframe = document.createElement(‘iframe’);

iframe.style.width = ‘1px’;

iframe.style.height = ‘1px’;

iframe.style.display = ‘none’;

iframe.src = https://segmentfault.com/a/url;

document.body.appendChild(iframe);

setTimeout(function() {

iframe.remove();

}, 100);

}

然后客户端通过拦截这个请求,并且解析出相应的方法和参数:这里以ios为例:

// IOS swift code

func webView(webView: UIWebView, shouldStartLoadWithRequest request: NSURLRequest, navigationType: UIWebViewNavigationType) -> Bool {

let url = request.URL

let scheme = url?.scheme

let method = url?.host

let query = url?.query

if url != nil && scheme == “jsbridge” {

switch method! {

case “getShare”:

self.getShare()

default:

print(“default”)

}

return false

} else {

return true

}

}

看不懂就略过,非重点。。。。。

这里我们在请求参数中加上了cbName=jsCallClientBack,这个jsCallClientBack为JS调用客户端所定义的回调函数,在业务层jsBridge封装中,我们传入一个匿名函数作为回调,底层将这个函数绑定在window的jsbridge对象下并为其定义一个独一无二的key,这个key就是jsCallClientBack,客户端在处理完逻辑后,会通过上面已经介绍过的方法来回调window下的方法。

ps: 在将回调绑定在window下时,特别注意要使用bind保持函数内this的原有指向不变

IOS客户端调用H5方法


Native调用Javascript语言,是通过UIWebView组件的stringByEvaluatingJavaScriptFromString方法来实现的,该方法返回js脚本的执行结果。

// IOS swift code

webview.stringByEvaluatingJavaScriptFromString(“window.methodName()”)

从上面代码可以看出它其实就是执行了一个字符串化的js代码,调用了window下的一个对象,如果我们要让native来调用我们js写的方法,那这个方法就要在window下能访问到。但从全局考虑,我们只要暴露一个对象如JSBridge给native调用就好了。

调用客户端原生方法的回调函数也将绑在window下供客户端成功反调用,实际上一次调用客户端方法最后产生的结果是双向互相调用。

H5调用Android客户端方法


在安卓webView中有三种调用native的方式:

通过schema方式,客户端使用shouldOverrideUrlLoading方法对url请求协议进行解析。这种js的调用方式与ios的一样,使用iframe来调用native方法。通过在webview页面里直接注入原生js代码方式,使用addJavascriptInterface方法来实现。

// android JAVA code

class JSInterface {

@JavascriptInterface

public String getShare() {

//…

return “share”;

}

}

webView.addJavascriptInterface(new JSInterface(), “AndroidNativeApi”);

上面的代码就是在页面的window对象里注入了AndroidNativeApi对象。在js里可以直接调用原生方法。

使用prompt,console.log,alert方式,这三个方法对js里是属性原生的,在android webview这一层是可以重写这三个方法的。一般我们使用prompt,因为这个在js里使用的不多,用来和native通讯副作用比较少。

// android JAVA code

class WebChromeClient extends WebChromeClient {

@Override

public boolean onJsPrompt(WebView view, String url, String message, String defaultValue, JsPromptResult result) {

// 重写window下的prompt,通过result返回结果

}

@Override

public boolean onConsoleMessage(ConsoleMessage consoleMessage) {

}

@Override

public boolean onJsAlert(WebView view, String url, String message, JsResult result) {

}

}

一般而言安卓客户端选用1、2方案中的一种进行通信,从前端层面来讲,推荐客户端都使用schema协议的方式,便于前端jsBridge底层代码的维护与迭代。

Android客户端调用H5方法


在安卓APP中,客户端通过webviewloadUrl进行调用:

// android JAVA code

webView.loadUrl(“javascript:window.jsBridge.getShare()”);

H5端将方法绑定在window下的对象即可,无需与IOS作区分

H5调用RN客户端


我们知道RN的webView组件实际上就是对原生容器的二次封装,因此我们不需要直接通过schema协议来通信,只需要使用浏览器postMessage、onMessage来传递消息即可,类似于iframe,而真正的通信过程RN已经帮我们做了。

// h5 js code

window.postMessage(data);

// rn js code

<WebView

ref=“webView”

source={require(‘…/html/index.html’)}

injectedJavaScript={‘window.androidConfig = {}’}    // 通过这个props可以在webView初始化时注入属性方法

onMessage={e => {

let { data } = e.nativeEvent;

//…

}}

/>

RN客户端调用H5


postMessage是双向的,所以也可以在RN里发消息,H5里接消息来触发对应的回调

this.refs.webView.postMessage({

cbName: ‘xxx’,

param: {}

});

前端jsBridge的封装


在了解了js与客户端底层的通信原理后,我们可以将IOS、安卓统一封装成jsBridge提供给业务层开发调用。

class JsBridge {

static lastCallTime

constructor() {

if (UA.isReactNative()) {

document.addEventListener(‘message’, function(e) {

window.jsClientCallBacke.data.cbName;

});

}

}

// 通用callNtive方法

callClient(functionName, data, callback) {

// 避免连续调用

if (JsBridge.lastCallTime && (Date.now() - JsBridge.lastCallTime) < 100) {

setTimeout(() => {

this.callClient(functionName, data, callback);

}, 100);

return;

}

JsBridge.lastCallTime = Date.now();

data = data || {};

if (callback) {

const cbName = randomName();

Object.assign(data, { cbName });

window.jsClientCallBack[cbName] = callBack.bind(this);

}

if (UA.isIOS()) {

data.forEach((key, value) => {

try {

data[key] = JSON.stringify(value);

} catch(e) { }

});

var url = ‘jsbridge://’ + functionName + ‘?’ parseJSON(data);

var iframe = document.createElement(‘iframe’);

iframe.style.width = ‘1px’;

iframe.style.height = ‘1px’;

iframe.style.display = ‘none’;

iframe.src = url;

document.body.appendChild(iframe);

setTimeout(() => {

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
ld(iframe);

setTimeout(() => {

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的一套进阶学习的视频及面试专题资料包,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家

[外链图片转存中…(img-yWxX4dlu-1715561503904)]

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 8
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要在 iOS App 中嵌入 H5 页面并实现与原生 App通信,需要进行以下详细配置步骤: 1. 创建一个 `WKWebView` 实例并将其添加到视图层级中。 ```swift import WebKit class ViewController: UIViewController { var webView: WKWebView! override func viewDidLoad() { super.viewDidLoad() let configuration = WKWebViewConfiguration() let userContentController = WKUserContentController() // 添加 messageHandler,用于接收来自 H5 页面的消息 userContentController.add(self, name: "myHandler") configuration.userContentController = userContentController webView = WKWebView(frame: view.bounds, configuration: configuration) view.addSubview(webView) // 加载 H5 页面 if let htmlPath = Bundle.main.path(forResource: "index", ofType: "html") { let htmlUrl = URL(fileURLWithPath: htmlPath) webView.loadFileURL(htmlUrl, allowingReadAccessTo: htmlUrl) } } } ``` 2. 创建一个遵循 `WKScriptMessageHandler` 协议的类,用于处理从 H5 页面发送过来的消息。 ```swift extension ViewController: WKScriptMessageHandler { // 接收并处理从 H5 页面发送过来的消息 func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { if message.name == "myHandler" { if let body = message.body as? String { print("Received message from H5: \(body)") // 在这里处理来自 H5 页面的消息 } } } } ``` 3. 在 H5 页面中,使用 JavaScript 代码发送消息给原生 App。 ```javascript // 发送消息给原生 App window.webkit.messageHandlers.myHandler.postMessage("Hello from H5"); ``` 4. 在 H5 页面中,可以使用以下 JavaScript 代码接收来自原生 App 的消息。 ```javascript // 接收原生 App 发送的消息 document.addEventListener("messageReceived", function(event) { var message = event.detail; console.log("Received message from iOS App: " + message); }); ``` 在原生 App 中,你可以调用 `webView.evaluateJavaScript(_:completionHandler:)` 方法执行 JavaScript 代码来向 H5 页面发送消息。例如: ```swift let message = "Hello from iOS App" let jsCode = "document.dispatchEvent(new CustomEvent('messageReceived', { detail: '\(message)' }));" webView.evaluateJavaScript(jsCode, completionHandler: nil) ``` 通过以上配置,你就可以在 iOS App 中嵌入 H5 页面,并实现双向的通信。在 H5 页面中使用 `window.webkit.messageHandlers.<handlerName>.postMessage(<message>)` 发送消息给原生 App,而在原生 App 中通过实现 `WKScriptMessageHandler` 协议来接收并处理来自 H5 页面的消息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值