Flutter和Android的混合跳转

1、项目特点

项目是Flutter作为主工程,将Android module或SDK作为模块嵌入到flutter中,与通常所熟悉的Android(或iOS)工程将flutter 为module嵌入到工程中有所不同。

2、业务需求

任意界面间的跳转,不管是flutter页面Android页面亦或是SDK中的页面,需要实现相互间的任意跳转且按照顺序返回(例如进入时A-B-C-D-C-D-B,返回时是B-D-C-D-C-B-A,不考虑启动模式)

3、实现方式探索

我们都知道Android 要打开一个Flutter页面(简称为A页面吧),需要借助与FlutterEngine来开启一个新的任务栈。

startActivity(FlutterActivity.withNewEngine().initialRoute("/second").build(ThreeActivity.this));

这样可以开启在Flutter代码中注册路由为“/second”的A页面,但这个页面加载速度会有些慢,因为每次都需要创建新的FlutterEngine,因此我们可以优化代码为:

public class MyApplication extends FlutterApplication {@Overridepublic void onCreate() {super.onCreate();FlutterEngine flutterEngineInit = new FlutterEngine(this);//flutterEngineInit.// 开始执行Dart代码以预热FlutterEngineflutterEngineInit.getNavigationChannel().setInitialRoute("/second");flutterEngineInit.getDartExecutor().executeDartEntrypoint(DartExecutor.DartEntrypoint.createDefault());// 缓存FlutterActivity要使用的FlutterEngineFlutterEngineCache.getInstance().put("/second", flutterEngineInit);}
}

自定义Application并在AndroidManifest.xml中注册,在自定义的application的onCreate()方法中将页面提前绑定到创建好的FlutterEngine上,并放入到缓存中

    <applicationandroid:label="untitled"android:name=".MyApplication"android:icon="@mipmap/ic_launcher"android:usesCleartextTraffic="true"></application>

这样在打开A页面时就可以,从缓存中获取,加载体验会明显优于使用.withNewEngine()创建并打开A页面

        jumpTv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {startActivity(FlutterActivity.withCachedEngine("/second").build(ThreeActivity.this));}});

注意:需要在AndroidManifest.xml中注册FlutterActivity

       <activityandroid:name="io.flutter.embedding.android.FlutterActivity"android:theme="@style/LaunchTheme"android:configChanges="orientation|keyboardHidden|keyboard|screenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"android:hardwareAccelerated="true"android:windowSoftInputMode="adjustResize"/>

到这里你可以在Android中打开A页面了。

4、遇到问题

但有一个很大的问题,如果A页面需要与Android交互怎么办?每次打开A页面都是放在一个新的FlutterEngine中,FlutterEngine是相互隔离的,channel无法互通。而我们的项目作为一个flutter项目,APP启动时,已经创建了FlutterEngine,且channel全部建立在这个engine中,上述方法打开的页面是无法使用这些channel;并且flutter B页面中也可以通过navigation.push打卡A页面,那A页面所在的FlutterEngine,即为B页面所在的FlutterEngine。如此情况A页面的在调用channel时会报如下错误:

2023-11-15 15:56:51.504 1528-10332/com.example.untitled E/flutter: [ERROR:flutter/lib/ui/ui_dart_state.cc(209)] Unhandled Exception: MissingPluginException(No implementation found for method jumpToAndroidWebviewPage on channel samples.flutter.jumpto.android)#0      MethodChannel._invokeMethod (package:flutter/src/services/platform_channel.dart:165:7)<asynchronous suspension>

提示我在A页面中调用的channel,没有找到对应的实现。这是因为Android通过FlutterEngine开启的A页面,新的任务栈中没有对应的实现。如何解决这个问题呢?

MethodChannel一般会是这样写:


public class MainActivity extends FlutterActivity  implements MethodChannel.MethodCallHandler{@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);}@Overridepublic void configureFlutterEngine(@NonNull FlutterEngine flutterEngine) {super.configureFlutterEngine(flutterEngine);MethodChannel commonMethodChannel = new MethodChannel(flutterEngine.getDartExecutor(), "samples.flutter.jumpto.android");commonMethodChannel.setMethodCallHandler(this);}@Overridepublic void onMethodCall(@NonNull MethodCall call, @NonNull MethodChannel.Result result) {}
}

我注意到在创建channel的时候,接收两个参数一个是messenger,一个是name。name好理解就是channel名称,自己定义。而messenger,我们是通过flutterEngine.getDartExecutor()来获取的一个BinaryMessenger的实现类对象。 也就是说我们只要能拿到每次打开A页面的FlutterEngine对象,那就好解决这个问题了。

我们创建一个channel的管理类,方便我们在每个engine中实现channel(代码有优化空间)

public class ChannelManage implements MethodChannel.MethodCallHandler, EventChannel.StreamHandler{private FlutterEngine flutterEngine;private Context context;EventChannel.EventSink jumpevents;public ChannelManage(FlutterEngine flutterEngine, Context context){this.flutterEngine = flutterEngine;this.context = context;MethodChannel commonMethodChannel = new MethodChannel(flutterEngine.getDartExecutor(), "samples.flutter.jumpto.android");commonMethodChannel.setMethodCallHandler(this);}@Overridepublic void onMethodCall(@NonNull MethodCall call, @NonNull MethodChannel.Result result) {if (call.method.equals("jumpToAndroidPage")) {Intent intent=new Intent(context, ContentActivity.class);context.startActivity(intent);result.success("跳转");}else if(call.method.equals("jumpToAndroidSDKPage")){Intent intent=new Intent(context, JumpActivity.class);context.startActivity(intent);result.success("跳转");}else if(call.method.equals("jumpToAndroidWebviewPage")){Intent intent=new Intent(context, ThreeActivity.class);context.startActivity(intent);result.success("跳转");} else {result.notImplemented();}}@Overridepublic void onListen(Object arguments, EventChannel.EventSink events) {events.success("Android");jumpevents=events;}@Overridepublic void onCancel(Object arguments) {}
}

步骤3中的代码改下:

我们在application中已经缓存了A页面和绑定的FlutterEngine,从缓存那取出来,获取对应的messenger设置给ChannelManage,这样A页面的Channel也可以使用啦。

        jumpTv.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View view) {new ChannelManage(FlutterEngineCache.getInstance().get("/second"),ThreeActivity.this);startActivity(FlutterActivity.withCachedEngine("/second").build(ThreeActivity.this));}});

但是不要高兴太早,还记得我们要的需求不?任意界面间的跳转,不管是flutter页面Android页面亦或是SDK中的页面。

按照上面的方式假如有一个Android activityB 通过上述方法打开了flutter A页面,A页面又通过Channel新开了activityB页面,activityB又打开了A页面,(这里例子比较极端,现实中A页面和activityB中间可能有其他页面,但不妨碍会出现循环打开的场景)那会是什么样子呢?

童年Andy 2023-11-11 16.19.28

我发现在返回到A页面倒数第二次打开的地方,屏幕上没有任何界面,且不能响应系统的手势放回。这肯定是不行啊,没有按原路返回啊,更大的问题是APP不能用了啊。

这是怎么回事呢?

其实是因为用了FlutterEngineCache,我们每次取到的都是一个FlutterEngine,那么当多次使用时,FlutterEngine就被放在了最后一次出现的位置,原位置将是一个空的engine,那这怎么玩。

再回过头来看下,如果每次都新建engine开启A页面呢?

回顾下上面的代码(中间写的废话太多了)

startActivity(FlutterActivity.withNewEngine().initialRoute("/second").build(ThreeActivity.this));

这肯定没问题了吧,试下。靠,Channel又报错了.......。

那就想法搞到每次的FlutterEngine不就行了吗?

看API....,哎呀,这玩意没有get FlutterEngine的API啊。这怎么玩。

但是如果我写一个MFlutterActivity extends FlutterActivity,然后重写configureFlutterEngine方法不就可以了吗?(MainActivity不就是这样搞的吗)

startActivity(MFlutterActivity.withNewEngine().initialRoute("/second").build(ThreeActivity.this));
public class MFlutterActivity extends FlutterActivity {@Overrideprotected void onCreate(@Nullable Bundle savedInstanceState) {super.onCreate(savedInstanceState);}@Overridepublic void configureFlutterEngine(@NonNull FlutterEngine flutterEngine) {new ChannelManage(flutterEngine,this);}}

注意:需要在AndroidManifest.xml中注册MFlutterActivity

来一把试试,结果不行。这里的configureFlutterEngine不会回调,不应该啊,MainActivity为什么就可以呢?

想不通啊想不通。翻源代码。

发现如下注释(走了很多弯路,不赘述了,也尝试过反射获取engine对象,我没有成功,小伙伴们可以试试)

    /*** Constructor that allows this {@code NewEngineIntentBuilder} to be used by subclasses of* {@code FlutterActivity}.** <p>Subclasses of {@code FlutterActivity} should provide their own static version of {@link* #withNewEngine()}, which returns an instance of {@code NewEngineIntentBuilder} constructed* with a {@code Class} reference to the {@code FlutterActivity} subclass, e.g.:** <p>{@code return new NewEngineIntentBuilder(MyFlutterActivity.class); }*/public NewEngineIntentBuilder(@NonNull Class<? extends FlutterActivity> activityClass) {this.activityClass = activityClass;}

啥意思?这意思是不是要自己重写withNewEngine方法?干吧,直接把系统的代码粘贴复制一下试试。

/*** Creates an {@link NewEngineIntentBuilder}, which can be used to configure an {@link Intent} to* launch a {@code FlutterActivity} that internally creates a new {@link* io.flutter.embedding.engine.FlutterEngine} using the desired Dart entrypoint, initial route,* etc.** @return The engine intent builder.*/@NonNullpublic static NewEngineIntentBuilder withNewEngine() {return new NewEngineIntentBuilder(FlutterActivity.class);}

APP起来,起来吧!!!

童年Andy 2023-11-11 14.30.39

5、总结:

实现flutter和Activity间的任意跳转一定要实现对Channel的统一管理和自定义MFlutterActivity继承FlutterActivity,用MFlutterActivity去执行加载flutter页面的代码开启新的engine,且一定要重写withNewEngine方法,否则Activity的生命周期以及configureFlutterEngine是不会执行的,也就没办法拿到engine,以便为统一管理的Channel设置messenger。

6、原代码

最后上demo代码,自取。

flutter_android_jump: 实现flutter工程下嵌入SDK、Android module后,flutter与Android界面间的任意跳转和通信

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/163484.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

工作中死循环害死人

背景&#xff1a;研发的一段代码&#xff0c;循环一直没有跳出&#xff0c;导致其他依赖逻辑有问题&#xff0c;生产事故导致9万左右数据不正常。 这里while&#xff08;true&#xff09;真的不要轻易用 &#xff0c;后来研发改动限制mysql的id切分步长&#xff0c;控制不会有数…

去大连发展还是去苏州

公司要搬到苏州&#xff0c;你是跟随公司去苏州发展&#xff0c;还是留在大连另寻出路&#xff1f;

【RtpRtcp】1: webrtc m79:audio的ChannelReceive 创建并使用

m79中,RtpRtcp::Create 的调用很少 不知道谁负责创建ChannelReceiveclass ChannelReceive : public ChannelReceiveInterface,public MediaTransportAudioSinkInterface {接收编码后的音频帧:接收rtcp包:

linux端无法magic上网,该怎么处理

可以打开网址&#xff1a;登录 — iKuuu VPN 不用重新安装&#xff0c;再次配置一下即可。

深入了解前馈网络、CNN、RNN 和 Hugging Face 的 Transformer 技术!

一、说明 本篇在此对自然语言模型做一个简短总结&#xff0c;从CNN\RNN\变形金刚&#xff0c;和抱脸的变形金刚库说起。 二、基本前馈神经网络&#xff1a; 让我们分解一个基本的前馈神经网络&#xff0c;也称为多层感知器&#xff08;MLP&#xff09;。此代码示例将&#xff1…

Web应用系统的小安全漏洞及相应的攻击方式

1 写作目的 本文讲述一个简单的利用WebAPI来进行一次基本没有破坏力的“黑客”行为。 主要目的如下&#xff1a; 了解什么叫安全漏洞知道什么是api了解一些获取api的工具通过对API的认识了解白盒接口测试基本概念和技术 免责声明&#xff1a; 本文主要是以学习交流为目的…

C++ ,VCPKG那些事

玩过C都知道&#xff0c;熟悉三方库对开发工作的重要性&#xff0c;寻找同步更新、稳定、权威的库源更是每一位开发者经常要做的功课&#xff0c;诸如赫赫有名的boost,google SDK、腾迅sdk、阿里库&#xff0c;vcpkg等等&#xff0c;这里要说的就是VCPKG&#xff0c;以下是记录…

盘点63个Python登录第三方源码Python爱好者不容错过

盘点63个Python登录第三方源码Python爱好者不容错过 学习知识费力气&#xff0c;收集整理更不易。 知识付费甚欢喜&#xff0c;为咱码农谋福利。 链接&#xff1a;https://pan.baidu.com/s/1l7oooH9YovHmWzQ_58FRdg?pwd8888 提取码&#xff1a;8888 项目名称 A headless…

11-23 SSM4

Ajax 同步请求 &#xff1a;全局刷新的方式 -> synchronous请求 客户端发一个请求&#xff0c;服务器响应之后你客户端才能继续后续操作&#xff0c;请求二响应完之后才能发送后续的请求&#xff0c;依次类推 有点&#xff1a;服务器负载较小&#xff0c;但是由于服务器相应…

Vue3+Ts实现聊天机器人(chatBot-附代码)

一&#xff1a;项目介绍 本次实验主要涉及到的技术是 Vue3 Ts&#xff0c;当然其中也有部分是 Vue2 格式的代码以及 json 和 CSS 布局等。本来是想仿照 文心一言 来开发的一个聊天机器人案例。结果由于时间不足&#xff0c;可能只是做出来了一个半成品。不过核心功能是有的。由…

浅谈安科瑞智能照明系统在马来西亚国家石油公司项目的应用

摘要&#xff1a;随着社会经济的发展及网络技术、通信技术的提高&#xff0c;人们对照明设计提出了新的要求&#xff0c;它不仅要控制照明光源的发光时间、 亮度&#xff0c;而且与其它系统来配合不同的应用场合做出相应的灯光场景。本文介绍了马亚西亚石油公司智能照明项目的应…

tp8 使用rabbitMQ(2)工作队列

代码的参数说明在 第一小节的代码中&#xff0c;如果需要可移步到第一节中查看 工作队列 工作队列&#xff08;又称&#xff1a;任务队列——Task Queues&#xff09;是为了避免等待一些占用大量资源、时间的操作。当我们把任务&#xff08;Task&#xff09;当作消息发送到队列…

推荐一款png图片打包plist工具pngPackerGUI_V2.0

png图片打包plist工具&#xff0c;手把手教你使用pngPackerGUI_V2.0 此软件是在pngpacker_V1.1软件基础之后&#xff0c;开发的界面化操作软件&#xff0c;方便不太懂命令行的小白快捷上手使用。1.下载并解压缩软件&#xff0c;得到如下目录&#xff0c;双击打开 pngPackerGUI.…

《第一行代码:Android》第三版-2.4.3循环语句 for循环

本节主要讲for 循环&#xff0c;主要就是创建个区间&#xff0c;然后用for 来遍历。 /*** You can edit, run, and share this code.* play.kotlinlang.org*/fun main() {println("Hello, world!!!")for(i in 1..10)//表示 0 到10 但是包括10 &#xff0c;即数学上的…

使用paddleocr进行OCR文字识别

1 OCR介绍 OCR&#xff08;Optical Character Recognition&#xff09;即光学字符识别&#xff0c;是一种将不同类型的文档&#xff08;如扫描的纸质文件、PDF文件或图像文件中的文本&#xff09;转换成可编辑和可搜索的数据的技术。OCR技术能够识别和转换印刷或手写文字&…

Python接口自动化测试——如何搭建测试环境

前言 接口测试的方式有很多&#xff0c;比如可以用工具&#xff08;jmeter,postman&#xff09;之类&#xff0c;也可以自己写代码进行接口测试&#xff0c;工具的使用相对来说都比较简单&#xff0c;重点是要搞清楚项目接口的协议是什么&#xff0c;然后有针对性的进行选择&a…

NF是哪个国家品牌?韩国NF-耐福功放芯片介绍

NF是韩国一家专注于数字音频芯片领域公司旗下的数字功放IC品牌&#xff0c;2008年开始进入中国市场&#xff0c;ISweek工采网作为其代理商&#xff1b;在国内简称&#xff1a;韩国NF&#xff0c;又被称耐福、耐福NTP功放芯片&#xff1b;国内长虹、TCL、海信、小豹AI音箱、腾讯…

大数据-之LibrA数据库系统告警处理(ALM-37001 MPPDBServer实例Redo日志缺失)

告警解释 当DN主实例有未同步到DN备实例的xlog日志被删除时&#xff0c;产生该告警。 告警属性 告警ID 告警级别 可自动清除 37001 严重 是 告警参数 参数名称 参数含义 ServiceName 产生告警的服务名称 RoleName 产生告警的角色名称 HostName 产生告警的主机名…

shell从服务列表中读取参数并检查服务

一、shell脚本 #!/usr/bin/env bash# 读取文档数据 while read -r line do# 提取服务名和命令service_name$(echo "$line" | awk {print $1})command$(echo "$line" | awk {$1""; print substr($0,2)})#echo "服务名: $service_name"…

SQL进阶学习

1.[NISACTF 2022]join-us sql报错注入和联合注入 过滤&#xff1a; as IF rand() LEFT by updatesubstring handler union floor benchmark COLUMN UPDATE & sys.schema_auto_increment_columns && 11 database case AND right CAST FLOOR left updatexml DATABA…