• HarmonyOS应用侧与前端页面数据通道建立


    一、应用侧调用前端页面函数

    应用侧可以通过runJavaScript()方法调用前端页面的JavaScript相关函数。在下面的示例中,点击应用侧的“runJavaScript”按钮时,来触发前端页面的htmlTest()方法。

    • 前端页面代码。
    1. html>
    2. <html>
    3. <body>
    4. <script>
    5. function htmlTest() {
    6. console.info('JavaScript Hello World! ');
    7. }
    8. script>
    9. body>
    10. html>

    应用侧代码。

    1. // xxx.ets
    2. import web_webview from '@ohos.web.webview';
    3. @Entry
    4. @Component
    5. struct WebComponent {
    6. webviewController: web_webview.WebviewController = new web_webview.WebviewController();
    7. build() {
    8. Column() {
    9. Web({ src: $rawfile('index.html'), controller: this.webviewController})
    10. Button('runJavaScript')
    11. .onClick(() => {
    12. this.webviewController.runJavaScript('htmlTest()');
    13. })
    14. }
    15. }
    16. }

    二、前端页面调用应用侧函数

    开发者使用Web组件将应用侧代码注册到前端页面中,注册完成之后,前端页面中使用注册的对象名称就可以调用应用侧的函数,实现在前端页面中调用应用侧方法。

    注册应用侧代码有两种方式,一种在Web组件初始化使用调用,使用javaScriptProxy()接口。另外一种在Web组件初始化完成后调用,使用registerJavaScriptProxy()接口。

    在下面的示例中,将test()方法注册在前端页面中, 该函数可以在前端页面触发运行。

    1. // xxx.ets
    2. import web_webview from '@ohos.web.webview';
    3. @Entry
    4. @Component
    5. struct WebComponent {
    6. webviewController: web_webview.WebviewController = new web_webview.WebviewController();
    7. // 声明需要注册的对象
    8. testObj = {
    9. test: () => {
    10. return 'ArkTS Hello World!';
    11. }
    12. }
    13. build() {
    14. Column() {
    15. // web组件加载本地index.html页面
    16. Web({ src: $rawfile('index.html'), controller: this.webviewController})
    17. // 将对象注入到web端
    18. .javaScriptProxy({
    19. object: this.testObj,
    20. name: "testObjName",
    21. methodList: ["test"],
    22. controller: this.webviewController
    23. })
    24. }
    25. }
    26. }
    • 应用侧使用registerJavaScriptProxy()接口注册。
    1. // xxx.ets
    2. import web_webview from '@ohos.web.webview';
    3. @Entry
    4. @Component
    5. struct Index {
    6. webviewController: web_webview.WebviewController = new web_webview.WebviewController();
    7. testObj = {
    8. test: (data) => {
    9. return "ArkUI Web Component";
    10. },
    11. toString: () => {
    12. console.info('Web Component toString');
    13. }
    14. }
    15. build() {
    16. Column() {
    17. Button('refresh')
    18. .onClick(() => {
    19. try {
    20. this.webviewController.refresh();
    21. } catch (error) {
    22. console.error(`Errorcode: ${error.code}, Message: ${error.message}`);
    23. }
    24. })
    25. Button('Register JavaScript To Window')
    26. .onClick(() => {
    27. try {
    28. this.webviewController.registerJavaScriptProxy(this.testObj, "objName", ["test", "toString"]);
    29. } catch (error) {
    30. console.error(`Errorcode: ${error.code}, Message: ${error.message}`);
    31. }
    32. })
    33. Web({ src: $rawfile('index.html'), controller: this.webviewController })
    34. }
    35. }
    36. }

    说明

    使用registerJavaScriptProxy()接口注册方法时,注册后需调用refresh()接口生效。

    • index.html前端页面触发应用侧代码。
    1. <!-- index.html -->
    2. <!DOCTYPE html>
    3. <html>
    4. <body>
    5. <button type="button" onclick="callArkTS()">Click Me!</button>
    6. <p id="demo"></p>
    7. <script>
    8. function callArkTS() {
    9. let str = objName.test();
    10. document.getElementById("demo").innerHTML = str;
    11. console.info('ArkTS Hello World! :' + str);
    12. }
    13. </script>
    14. </body>
    15. </html>

    三、建立应用侧与前端页面数据通道

    前端页面和应用侧之间可以用createWebMessagePorts()接口创建消息端口来实现两端的通信。

    在下面的示例中,应用侧页面中通过createWebMessagePorts方法创建消息端口,再把其中一个端口通过postMessage()接口发送到前端页面,便可以在前端页面和应用侧之间互相发送消息。

    • 应用侧代码。
    1. // xxx.ets
    2. import web_webview from '@ohos.web.webview';
    3. @Entry
    4. @Component
    5. struct WebComponent {
    6. controller: web_webview.WebviewController = new web_webview.WebviewController();
    7. ports: web_webview.WebMessagePort[];
    8.   @State sendFromEts: string = 'Send this message from ets to HTML';
    9.   @State receivedFromHtml: string = 'Display received message send from HTML';
    10. build() {
    11. Column() {
    12. // 展示接收到的来自HTML的内容
    13. Text(this.receivedFromHtml)
    14. // 输入框的内容发送到html
    15. TextInput({placeholder: 'Send this message from ets to HTML'})
    16. .onChange((value: string) => {
    17. this.sendFromEts = value;
    18. })
    19. Button('postMessage')
    20. .onClick(() => {
    21. try {
    22. // 1、创建两个消息端口。
    23. this.ports = this.controller.createWebMessagePorts();
    24. // 2、在应用侧的消息端口(如端口1)上注册回调事件。
    25. this.ports[1].onMessageEvent((result: web_webview.WebMessage) => {
    26. let msg = 'Got msg from HTML:';
    27. if (typeof(result) === 'string') {
    28. console.info(`received string message from html5, string is: ${result}`);
    29. msg = msg + result;
    30. } else if (typeof(result) === 'object') {
    31. if (result instanceof ArrayBuffer) {
    32. console.info(`received arraybuffer from html5, length is: ${result.byteLength}`);
    33. msg = msg + 'lenght is ' + result.byteLength;
    34. } else {
    35. console.info('not support');
    36. }
    37. } else {
    38. console.info('not support');
    39. }
    40. this.receivedFromHtml = msg;
    41. })
    42. // 3、将另一个消息端口(如端口0)发送到HTML侧,由HTML侧保存并使用。
    43. this.controller.postMessage('__init_port__', [this.ports[0]], '*');
    44. } catch (error) {
    45. console.error(`ErrorCode: ${error.code}, Message: ${error.message}`);
    46. }
    47. })
    48. // 4、使用应用侧的端口给另一个已经发送到html的端口发送消息。
    49. Button('SendDataToHTML')
    50. .onClick(() => {
    51. try {
    52. if (this.ports && this.ports[1]) {
    53. this.ports[1].postMessageEvent(this.sendFromEts);
    54. } else {
    55. console.error(`ports is null, Please initialize first`);
    56. }
    57. } catch (error) {
    58. console.error(`ErrorCode: ${error.code}, Message: ${error.message}`);
    59. }
    60. })
    61. Web({ src: $rawfile('xxx.html'), controller: this.controller })
    62. }
    63. }
    64. }
    • 前端页面代码。
    1. <!--xxx.html-->
    2. <!DOCTYPE html>
    3. <html>
    4. <head>
    5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
    6. <title>WebView Message Port Demo</title>
    7. </head>
    8. <body>
    9. <h1>WebView Message Port Demo</h1>
    10. <div>
    11. <input type="button" value="SendToEts" onclick="PostMsgToEts(msgFromJS.value);"/><br/>
    12. <input id="msgFromJS" type="text" value="send this message from HTML to ets"/><br/>
    13. </div>
    14. <p class="output">display received message send from ets</p>
    15. </body>
    16. <script>
    17. var h5Port;
    18. var output = document.querySelector('.output');
    19. window.addEventListener('message', function (event) {
    20. if (event.data === '__init_port__') {
    21. if (event.ports[0] !== null) {
    22. h5Port = event.ports[0]; // 1. 保存从ets侧发送过来的端口
    23. h5Port.onmessage = function (event) {
    24. // 2. 接收ets侧发送过来的消息.
    25. var msg = 'Got message from ets:';
    26. var result = event.data;
    27. if (typeof(result) === 'string') {
    28. console.info(`received string message from html5, string is: ${result}`);
    29. msg = msg + result;
    30. } else if (typeof(result) === 'object') {
    31. if (result instanceof ArrayBuffer) {
    32. console.info(`received arraybuffer from html5, length is: ${result.byteLength}`);
    33. msg = msg + 'lenght is ' + result.byteLength;
    34. } else {
    35. console.info('not support');
    36. }
    37. } else {
    38. console.info('not support');
    39. }
    40. output.innerHTML = msg;
    41. }
    42. }
    43. }
    44. })
    45. // 3. 使用h5Port往ets侧发送消息.
    46. function PostMsgToEts(data) {
    47. if (h5Port) {
    48. h5Port.postMessage(data);
    49. } else {
    50. console.error('h5Port is null, Please initialize first');
    51. }
    52. }
    53. </script>
    54. </html>
  • 相关阅读:
    Kotlin高仿微信-第21篇-个人信息-修改头像
    Mysql Explain
    Go语言 01
    linux虚拟机(Ubuntu)和电脑ping通
    docker离线搭建仓库
    吴恩达深度学习笔记(四)——深度学习的实践层面
    什么是spring mvc 模式
    35 LRU缓存
    android桌面插件每秒刷新
    老陈打码老陈打码
  • 原文地址:https://blog.csdn.net/HarmonyOSDev/article/details/132897428