效果
通常Flutter與Android頁面交互是各自獨占整個手機屏幕咳胃,但有些情況下無法滿足需求植康,比如:Flutter頁面中嵌套Android地圖旷太,F(xiàn)lutter中嵌套相機預(yù)覽圖等,這些都需要在手機屏幕中既有Flutter頁面也有Android頁面销睁,有些時候Flutter中沒有提供相關(guān)插件或者插件不滿足需求供璧,這時候就需要開發(fā)者自定義插件,開發(fā)者可以參考本文中的方法去進行自定義冻记。本文具體demo效果如下:開發(fā)
- 首先創(chuàng)建flutter項目睡毒,在項目中定義好flutter需要展示布局:
@override
Widget build(BuildContext context) {
return Scaffold(
body: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: <Widget>[
Expanded(
child: Center(
child: Text(
'Android按鈕點擊了 $_counter 次',
style: const TextStyle(fontSize: 17.0)),
),
),
Container(
padding: const EdgeInsets.only(bottom: 15.0, left: 5.0),
child: Row(
children: <Widget>[
Image.asset('assets/flutter-mark-square-64.png', scale: 1.5),
const Text('Flutter', style: TextStyle(fontSize: 30.0)),
],
),
),
],
),
floatingActionButton: FloatingActionButton(
onPressed: _sendFlutterIncrement,
child: const Icon(Icons.add),
),
);
}
復(fù)制代碼
上述代碼所呈現(xiàn)的布局就是效果圖中Flutter那一部分(上半部分),設(shè)置FloatingActionButton是為了呈現(xiàn)Flutter與Android的交互過程冗栗, "Android按鈕點擊了 $_counter 次"呈現(xiàn)的是交互結(jié)果演顾,Image.asset()則顯示的是Flutter的logo(標記這是flutter中的布局)。之所以這樣做是因為Flutter與Android頁面相互嵌套通產(chǎn)伴隨著數(shù)據(jù)交互隅居。
- 創(chuàng)建Android中的布局:
<io.flutter.embedding.android.FlutterView
android:id="@+id/flutter_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_weight="1"
/>
<androidx.coordinatorlayout.widget.CoordinatorLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_weight="1"
android:background="@color/grey"
>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
>
<TextView
android:id="@+id/button_tap"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:text="@string/button_tap"
...
/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/android"
...
/>
</LinearLayout>
<com.google.android.material.floatingactionbutton.FloatingActionButton
android:id="@+id/button"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
...
/>
</androidx.coordinatorlayout.widget.CoordinatorLayout>
復(fù)制代碼
io.flutter.embedding.android.FlutterView就是需要展示的flutter布局也就是第一步中編寫的布局钠至,剩下的部分和第一步的邏輯是一樣的,有用來展示交互結(jié)果的TextView(@+id/button_tap)胎源,標記Android頁面的TextView(@string/android)棉钧,用來交互的按鈕FloatingActionButton。
- 定義通信渠道
- Flutter:
static const String _channel = 'increment';
static const String _pong = 'pong';
static const String _emptyMessage = '';
static const BasicMessageChannel<String> platform =
BasicMessageChannel<String>(_channel, StringCodec());
int _counter = 0;
@override
void initState() {
super.initState();
platform.setMessageHandler(_handlePlatformIncrement);
}
Future<String> _handlePlatformIncrement(String message) async {
setState(() {
_counter++;
});
return _emptyMessage;
}
void _sendFlutterIncrement() {
platform.send(_pong);
}
復(fù)制代碼
代碼中通信渠道使用的是BasicMessageChannel涕蚤,沒有用MethodChannel和EventChannel是因為BasicMessageChannel可以隨時隨地進行任何通信宪卿,而另外兩種則各自有各自的局限性,這里就不做解釋了万栅,稍后會有文章專門介紹這三種通信渠道佑钾。_channel 是通信渠道的名稱,這個是唯一且固定的烦粒,一旦定義好休溶,Android端也要使用相同的名稱,否則兩者無法對接,導(dǎo)致通信失敗邮偎。_pong是Flutter向Android傳遞的消息內(nèi)容管跺,Android每次接收的內(nèi)容為"pong",相應(yīng)的Flutter按鈕點擊次數(shù)就+1禾进,消息內(nèi)容和類型用戶都可以自定義豁跑,只要定義好BasicMessageChannel的泛型和消息編碼機制(文中使用的是StringCodec)即可。如果消息的內(nèi)容比較多泻云,開發(fā)者可以使用Json進行消息傳遞艇拍。_counter是flutter接收Android按鈕的點擊次數(shù),Android按鈕每點擊一次_counter就+1宠纯。相關(guān)變量或常量定義完后卸夕,開發(fā)者需要在initState()中進行消息接收處理,因為BasicMessageChannel是雙向通信婆瓜,platform.setMessageHandler(_handlePlatformIncrement)就是對接收到的消息進行處理快集,_handlePlatformIncrement方法說明了消息的類型是String,消息是異步廉白,_counter+1后調(diào)用setState()刷新布局后相應(yīng)展示的Android按鈕點擊次數(shù)就+1了个初。platform.send(_pong)就是Flutter按鈕點擊完后調(diào)用這個方法,然后BasicMessageChannel將消息發(fā)送到Android猴蹂。
- Android:
private static FlutterEngine flutterEngine;
private FlutterView flutterView;
private int counter;
private static final String CHANNEL = "increment";
private static final String EMPTY_MESSAGE = "";
private static final String PING = "ping";
private BasicMessageChannel<String> messageChannel;
...
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
if (flutterEngine == null) {
flutterEngine = new FlutterEngine(this, args);
flutterEngine.getDartExecutor().executeDartEntrypoint(
DartEntrypoint.createDefault()
);
}
...
flutterView = findViewById(R.id.flutter_view);
flutterView.attachToFlutterEngine(flutterEngine);
messageChannel = new BasicMessageChannel<>(flutterEngine.getDartExecutor(), CHANNEL, StringCodec.INSTANCE);
messageChannel.
setMessageHandler(new MessageHandler<String>() {
@Override
public void onMessage(String s, Reply<String> reply) {
onFlutterIncrement();
reply.reply(EMPTY_MESSAGE);
}
});
FloatingActionButton fab = findViewById(R.id.button);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
sendAndroidIncrement();
}
});
...
復(fù)制代碼
CHANNEL 是通信渠道的名稱也是渠道的標識符院溺,一定要和flutter統(tǒng)一,否則無法通信磅轻。BasicMessageChannel是通信渠道珍逸,如果使用了和flutter端不一樣的,也是無法通信的聋溜。EMPTY_MESSAGE是Android端收到Flutter消息后給Flutter的回復(fù)谆膳,表示讓Flutter知道Android收到消息了。Android端收到消息后在setMessageHandler中進行消息處理:將flutter點擊按鈕次數(shù)+1( onFlutterIncrement())勤婚,同時回復(fù)確認收到的消息(reply.reply(EMPTY_MESSAGE))摹量。FloatingActionButton發(fā)生點擊事件后調(diào)用sendAndroidIncrement方法就會向Flutter發(fā)送消息說自己點擊了一次按鈕:
private void sendAndroidIncrement() {
messageChannel.send(PING);
}
復(fù)制代碼
PING就是Android向Flutter發(fā)送的消息內(nèi)容,F(xiàn)lutter收到消息后就對Android點擊按鈕次數(shù)+1馒胆。如果傳遞的消息比較多缨称,還需要對具體的消息進行判斷來確認需要做哪些處理,本文中只傳遞一種內(nèi)容的消息祝迂,所以對消息的參數(shù)和方法沒有做判斷睦尽。代碼中flutterView即是需要展示的Flutter布局,flutterEngine則是flutter引擎(說法不統(tǒng)一)型雳, flutterView.attachToFlutterEngine(flutterEngine)則是為flutterView注入生命和能量当凡,否則flutterView就是空空沒有生命和內(nèi)容的控件山害。flutterEngine和AppCompatActivity的生命周期是綁定在一起:
@Override
protected void onResume() {
super.onResume();
flutterEngine.getLifecycleChannel().appIsResumed();
}
@Override
protected void onPause() {
super.onPause();
flutterEngine.getLifecycleChannel().appIsInactive();
}
@Override
protected void onStop() {
super.onStop();
flutterEngine.getLifecycleChannel().appIsPaused();
}
@Override
protected void onDestroy() {
flutterView.detachFromFlutterEngine();
super.onDestroy();
}
復(fù)制代碼
Android中一旦出現(xiàn)了對生命周期的綁定,就是說只要按要求來沿量,就不會出現(xiàn)亂七八糟的問題浪慌,即使有問題也不是它的問題。
總結(jié)
- Flutter與Android的交互在不斷迭代中已經(jīng)變得比較完善朴则,最新Flutter版本中已經(jīng)比Flutter早期的版中簡單很多权纤。
- 如果能避免Flutter與Android的相互嵌套就盡量避免,因為兩者的嵌套很耗能乌妒,可能出現(xiàn)卡頓汹想、死機、高耗電等問題撤蚊。
作者:VanGogh
鏈接:https://juejin.im/post/6885918668018941965