Cordova源码深入分析-第三讲

2024-05-15 20:38

本文主要是介绍Cordova源码深入分析-第三讲,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

上一讲讲解了js端,调用到java端的代码逻辑

这一讲,主要介绍,js->native带有callback的形式,是如何调用回去的

上一篇介绍到了调用照相机代码的地方,在最后调用了:sendPluginResult

这里调用一下:

    @Overridepublic void sendPluginResult(PluginResult cr, String callbackId) {nativeToJsMessageQueue.addPluginResult(cr, callbackId);}

NativeToJsMessageQueue.java

    public void addPluginResult(PluginResult result, String callbackId) {if (callbackId == null) {LOG.e(LOG_TAG, "Got plugin result with no callbackId", new Throwable());return;}// Don't send anything if there is no result and there is no need to// clear the callbacks.boolean noResult = result.getStatus() == PluginResult.Status.NO_RESULT.ordinal();boolean keepCallback = result.getKeepCallback();if (noResult && keepCallback) {return;}JsMessage message = new JsMessage(result, callbackId);if (FORCE_ENCODE_USING_EVAL) {StringBuilder sb = new StringBuilder(message.calculateEncodedLength() + 50);message.encodeAsJsMessage(sb);message = new JsMessage(sb.toString());}enqueueMessage(message);//将消息放到队列中,里面含有}
    private void enqueueMessage(JsMessage message) {synchronized (this) {if (activeBridgeMode == null) {LOG.d(LOG_TAG, "Dropping Native->JS message due to disabled bridge");return;}queue.add(message);//将消息加入队列中if (!paused) { //如果没有pause则直接执行,有pause的话,下次触发执行activeBridgeMode.onNativeToJsMessageAvailable(this);//开始执行}}}

根据不同的bridge走不同代码,这里先列举一种情况

      @Overridepublic void onNativeToJsMessageAvailable(final NativeToJsMessageQueue queue) {cordova.getActivity().runOnUiThread(new Runnable() {public void run() {String js = queue.popAndEncodeAsJs();//取出刚才加入进去的消息,并且转换为jsif (js != null) {engine.loadUrl("javascript:" + js, false);}}});}

这里是组装js的代码:

    public String popAndEncodeAsJs() {synchronized (this) {int length = queue.size();if (length == 0) {return null;}int totalPayloadLen = 0;int numMessagesToSend = 0;for (JsMessage message : queue) {int messageSize = message.calculateEncodedLength() + 50; // overestimate.if (numMessagesToSend > 0 && totalPayloadLen + messageSize > MAX_PAYLOAD_SIZE && MAX_PAYLOAD_SIZE > 0) {break;}totalPayloadLen += messageSize;numMessagesToSend += 1;}boolean willSendAllMessages = numMessagesToSend == queue.size();StringBuilder sb = new StringBuilder(totalPayloadLen + (willSendAllMessages ? 0 : 100));// Wrap each statement in a try/finally so that if one throws it does// not affect the next.for (int i = 0; i < numMessagesToSend; ++i) {JsMessage message = queue.removeFirst();if (willSendAllMessages && (i + 1 == numMessagesToSend)) {message.encodeAsJsMessage(sb);} else {sb.append("try{");message.encodeAsJsMessage(sb);sb.append("}finally{");}}if (!willSendAllMessages) {sb.append("window.setTimeout(function(){cordova.require('cordova/plugin/android/polling').pollOnce();},0);");}for (int i = willSendAllMessages ? 1 : 0; i < numMessagesToSend; ++i) {sb.append('}');}String ret = sb.toString();Log.v("XPC","ret="+ret);//ret=cordova.callbackFromNative('Camera1024838304',false,9,["No Image Selected"],false);return ret;}}

看一下encodeAsJsMessage

        void encodeAsJsMessage(StringBuilder sb) {if (pluginResult == null) {sb.append(jsPayloadOrCallbackId);} else {int status = pluginResult.getStatus();boolean success = (status == PluginResult.Status.OK.ordinal()) || (status == PluginResult.Status.NO_RESULT.ordinal());sb.append("cordova.callbackFromNative('").append(jsPayloadOrCallbackId).append("',").append(success).append(",").append(status).append(",[");buildJsMessage(sb);sb.append("],").append(pluginResult.getKeepCallback()).append(");");}}

这样就调回了js端的cordova.js端

    callbackFromNative: function(callbackId, isSuccess, status, args, keepCallback) {try {var callback = cordova.callbacks[callbackId];//根据callbackId找到我们之前的函数,这个在上一讲中,写到过,if (callback) {if (isSuccess && status == cordova.callbackStatus.OK) {callback.success && callback.success.apply(null, args);//这里会直接调用camera.js的成功和失败方法中} else if (!isSuccess) {callback.fail && callback.fail.apply(null, args);}/*elseNote, this case is intentionally not caught.this can happen if isSuccess is true, but callbackStatus is NO_RESULTwhich is used to remove a callback from the list without calling the callbackstypically keepCallback is false in this case*/// Clear callback if not expecting any more resultsif (!keepCallback) {delete cordova.callbacks[callbackId];}}}catch (err) {var msg = "Error in " + (isSuccess ? "Success" : "Error") + " callbackId: " + callbackId + " : " + err;console && console.log && console.log(msg);cordova.fireWindowEvent("cordovacallbackerror", { 'message': msg });throw err;}},
本讲结束!!!!下一讲java端的插件系统是如何建立关联的










这篇关于Cordova源码深入分析-第三讲的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/992866

相关文章

Java汇编源码如何查看环境搭建

《Java汇编源码如何查看环境搭建》:本文主要介绍如何在IntelliJIDEA开发环境中搭建字节码和汇编环境,以便更好地进行代码调优和JVM学习,首先,介绍了如何配置IntelliJIDEA以方... 目录一、简介二、在IDEA开发环境中搭建汇编环境2.1 在IDEA中搭建字节码查看环境2.1.1 搭建步

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL

如何在Visual Studio中调试.NET源码

今天偶然在看别人代码时,发现在他的代码里使用了Any判断List<T>是否为空。 我一般的做法是先判断是否为null,再判断Count。 看了一下Count的源码如下: 1 [__DynamicallyInvokable]2 public int Count3 {4 [__DynamicallyInvokable]5 get

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

Spring 源码解读:自定义实现Bean定义的注册与解析

引言 在Spring框架中,Bean的注册与解析是整个依赖注入流程的核心步骤。通过Bean定义,Spring容器知道如何创建、配置和管理每个Bean实例。本篇文章将通过实现一个简化版的Bean定义注册与解析机制,帮助你理解Spring框架背后的设计逻辑。我们还将对比Spring中的BeanDefinition和BeanDefinitionRegistry,以全面掌握Bean注册和解析的核心原理。

音视频入门基础:WAV专题(10)——FFmpeg源码中计算WAV音频文件每个packet的pts、dts的实现

一、引言 从文章《音视频入门基础:WAV专题(6)——通过FFprobe显示WAV音频文件每个数据包的信息》中我们可以知道,通过FFprobe命令可以打印WAV音频文件每个packet(也称为数据包或多媒体包)的信息,这些信息包含该packet的pts、dts: 打印出来的“pts”实际是AVPacket结构体中的成员变量pts,是以AVStream->time_base为单位的显

kubelet组件的启动流程源码分析

概述 摘要: 本文将总结kubelet的作用以及原理,在有一定基础认识的前提下,通过阅读kubelet源码,对kubelet组件的启动流程进行分析。 正文 kubelet的作用 这里对kubelet的作用做一个简单总结。 节点管理 节点的注册 节点状态更新 容器管理(pod生命周期管理) 监听apiserver的容器事件 容器的创建、删除(CRI) 容器的网络的创建与删除

red5-server源码

red5-server源码:https://github.com/Red5/red5-server