code小生 一个专一大前端领域的技术平台公众号回复Android
加入安卓技术群前端
做者:言就尔宿1992
连接:https://www.jianshu.com/p/9425dd2baacc
声明:本文已获言就尔宿1992
受权发表,转发等请联系原做者受权java
Flutter 添加到现有项目 (Android)android
最近一直在看Flutter 的内容, 加上近期更新的Flutter1.12 有一些Flutter 的api 发生了改变, 因此 某些和android 交互的地方 就发生了变化c#
好比开始使用的新的插件api 旧的 PluginRegistry.Registrar 不会当即被弃用, 而使用了新的FlutterPlugin, FlutterActivity 的 从io.flutter.app.FlutterActivity 到 io.flutter.app.FlutterActivityapi
接下来 就是 添加到现有项目步骤 基于androdx性能优化
建立一个原生的Android项目 直接经过AndroidStudio 建立一个androd 项目app
而后在AS 中 经过 new Project 选择 FlutterModule 建立一个Flutter module 建立的时候记得 勾选androidxless
再 androd项目中 app 下 build.gradle 中添加async
android{ compileOptions { sourceCompatibility 1.8 targetCompatibility 1.8 } }
而后 再setting.gradle 中添加ide
setBinding(new Binding([gradle:this])) evaluate(new File( settingsDir.parentFile, '你的flutter module 的名字/.android/include_flutter.groovy' ))
编译
而后成功以后再android 项目中的 build.gradle 中添加
implementation project(path: ':flutter')
编译
调整Flutter 库中关联的.android 中的flutter 的库 中的build.gradle 的 compileSdkVersion minsdkVersion 以及 targetSdkVersion和主项目保持一致,
我用的是 在acticity 中 添加FlutterFragement 首先 Activity 继承 FragmentActivity
对了若是 Android 项目中是androdx 的话 须要再 Flutter项目中修改 pubspec.yaml
moudule 下的 androidX 是否为true;
接下来是我 Flutter 项目中的代码 就是 官方生成的代码 本身稍微改了下
import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; void main() => runApp(MyApp()); class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { return MaterialApp( title: 'Flutter Demo', theme: ThemeData( primarySwatch: Colors.blue, ), home: MyHomePage(title: 'Flutter Demo Home Page'), ); } } class MyHomePage extends StatefulWidget { MyHomePage({Key key, this.title}) : super(key: key); final String title; @override _MyHomePageState createState() => _MyHomePageState(); } class _MyHomePageState extends State<MyHomePage> { static const platform = const MethodChannel("com.battery.io/battery"); String _battery = "获取电池电量"; @override Widget build(BuildContext context) { return Scaffold( appBar: AppBar( title: Text(widget.title), ), body: Center( child: Column( mainAxisAlignment: MainAxisAlignment.center, children: <Widget>[ Text( _battery ), ], ), ), floatingActionButton: FloatingActionButton( onPressed: _getBatteryLevel, tooltip: 'Increment', child: Icon(Icons.add), ), // This trailing comma makes auto-formatting nicer for build methods. ); } Future<Null> _getBatteryLevel() async { String battery = "获取电池电量"; try { int result = await platform.invokeMethod("getBatteryLevel"); battery = "获取到的电量 $result"; } on PlatformException catch (e) { // 未获取到电池电量 battery = "未获取到电池电量"; } setState(() { _battery = battery; }); } }
其实 这篇文章主要就是记录了 fragment 和flutter 的交互
此处写了一个方法 经过Flutter 中的方法来调用 android 中的方法 我是经过 android 中添加一个Fragment 来加载Flutter 的界面
先上代码 --> Activity
package com.storm.stormandflutterdemo import android.app.Activity import android.content.Intent import android.os.Bundle import androidx.fragment.app.FragmentActivity import com.storm.stormandflutterdemo.Utils.LogUtils import com.storm.stormandflutterdemo.flutterplugin.BatteryPlugin import io.flutter.embedding.android.FlutterFragment import io.flutter.embedding.engine.FlutterEngine import io.flutter.embedding.engine.FlutterEngineCache import io.flutter.embedding.engine.dart.DartExecutor class CustomActivity : FragmentActivity() { companion object { val TAG_FLUTTER_FRAGMENT = "flutter_fragment" val TAG = "CustomActivity" fun startSelf(activity: Activity) { var intent = Intent(activity, CustomActivity::class.java) activity.startActivity(intent) } } var flutterFragment: FlutterFragment? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_custom) startFlutter() } private fun startFlutter() { initPlugin() var fm = supportFragmentManager flutterFragment = fm.findFragmentByTag(MainActivity.TAG_FLUTTER_FRAGMENT) as FlutterFragment? if (flutterFragment == null) { var newFlutterFragment = FlutterFragment.withCachedEngine("my_flutter").build<FlutterFragment>() flutterFragment = newFlutterFragment fm.beginTransaction() .add(R.id.fl_content, flutterFragment!!, MainActivity.TAG_FLUTTER_FRAGMENT).commit() LogUtils.d(TAG, "是否未null ? --> ${null == flutterFragment}") flutterFragment!!.onAttach(this) } } private fun initPlugin() { var flutterEngine = FlutterEngine(this) flutterEngine.dartExecutor.executeDartEntrypoint(DartExecutor.DartEntrypoint.createDefault()) FlutterEngineCache.getInstance() .put("my_flutter",flutterEngine) BatteryPlugin.registerWith(flutterEngine) } override fun onResume() { super.onResume() } }
此处 有个问题 就是我当时再建立FlutterFragment 老是爆红 开始的时候 发现导入的包不同
FlutterFragment 用的是 v4包,而我用的是androidx 的包, 其实我能够改为v4包的 可是考虑到之后 api 29 之后 的 会维护androidX 而放弃androd v4 v7 各个版本的维护 因此本身就钻牛角尖, 索性都换成androdX 的去寻找方法
查过来查过去 FlutterFragment 的继承的Fragment 始终是v4 包下的Fragment 即便我删除了这个FlutterModule 从新建立的开始勾选了androidX 仍是 v4包 , 在Android项目中 中 仍是爆红...
妈蛋 结果就是 我试着运行 居然正常运行 ... enen . 这边还爆红 也Flutter 加载出来了 .........###Fuck说
另外一方面就是 Flutter1.12 之后更换了api 由于我也是最近开始看 , 有的地方不是很懂 各位大神以为有什么不对的地方 留言一块儿讨论共同进步, 以前的若是要加载插件 , 就是在当前的activity 或者是fragement 中 注册 传入 PluginRegistry.Registrar 记得以前的 FlutterActivity 或者FlutterFragment 都实现了 这个接口, 因此直接传入this 拿到activity 的引用 , 不过在1.12 以后 改为了新的 直接 传入 FlutterEngine 来注册交互的插件。
其实initPlugin 这个方法 放入 Application 中也能够 经过一个tag , 而后 建立 FlutterFragment 的时候经过tag 来建立 , 而后插件注册
插件-->
package com.storm.stormandflutterdemo.flutterplugin import androidx.annotation.Keep import androidx.annotation.NonNull import io.flutter.embedding.engine.FlutterEngine import io.flutter.embedding.engine.plugins.shim.ShimPluginRegistry import io.flutter.plugin.common.MethodChannel import java.lang.reflect.Method /** * * @author storm_z * @date @{DATE} * @email zq329051@outlook.com * * @Describe */ @Keep object BatteryPlugin { public const val CHANNEL = "com.battery.io/battery" fun registerWith(@NonNull flutterEngine: FlutterEngine){ val shimPluginRegistry = ShimPluginRegistry(flutterEngine) val registrar = shimPluginRegistry.registrarFor("com.battery.io") val context = registrar.activeContext() val channel = MethodChannel(registrar.messenger(), CHANNEL) var value = 0 ; channel.setMethodCallHandler { call, result -> when(call.method){ "getBatteryLevel" -> { value += 20 result.success(value) } } } } }
此次 主要是想要吧 Flutter 放入到 android 的 Fragment 中来尝试 , 看了不少博客 和官方文档 都是介绍怎么在 Fragment 中使用可是交互方面 都是1.12 版本以前的 , 或者说在 FlutterActivity 中的 , 因此 本身在捣鼓出来以后就记录一下 而后有什么问题 一块儿讨论 共同进步 不多写这个 之后慢慢规范 . 去尝试 FlutterView 了 ......
1 已有 Android 项目集成 Flutter 寻坑记
2 Flutter 体验记
3 Flutter 应用性能优化最佳实践
4 Mac版最详细的Flutter开发环境搭建
5 Ubuntu 20.04 强悍来袭,看看都有哪些新特性?
若是你有写博客的好习惯
点个在看,小生感恩❤️