Flutter(六)Android與Flutter混合開發(fā)(Hybird)

因為筆者本身主要從事是Android開發(fā)咒林,所以很多角度都是作為一個Android開發(fā)者學習Flutter的角度出發(fā)徒仓,IOS或者H5的開發(fā)同學可以選擇性閱讀

目錄

前言

如果我們目前的項目是Android的悲伶,但是接下來我們希望部分頁面可以使用Flutter進行開發(fā),甚至我們希望在Native頁面中嵌入FlutterUI組件醉者,那么我們該如何實現(xiàn)呢啼器?

創(chuàng)建Flutter Module

假設你現(xiàn)在Android項目的目錄的結構是這樣的

xxx/flutter_hybrid/FlutterHybridAndroid 

這時候如果你想創(chuàng)建一個Flutter模塊玛追,使得Android模塊和Flutter模塊之間可以進行交互舔痕,我們可以通過Android Studio新建一個Flutter Module,具體過程是:File —> New —> New Module 豹缀,之后選擇Flutter Module,指定Project Location的路徑為

xxx/flutter_hybrid

也就是說慨代,最終你的項目結構會是這樣的

- flutter_hybrid  
  - flutter_module  
  - FlutterHybridAndroid 

接下來在Android Module的build.gradle文件中添加flutter依賴

//FlutterHybridAndroid/app/build.gradle
...
dependencies {
   implementation project(':flutter')
...
}

Android啟動Flutter頁面的兩種方式

先創(chuàng)建一個Flutter頁面

這里比較重要的是window.defaultRouteName這個字段邢笙,這個字段可以接收從Native傳遞過來的參數(shù)(下文我們會介紹原生傳遞參數(shù)的方法),也就是說通過這個字段我們就可以進行Flutter頁面的路由的分發(fā)

import 'dart:ui';

import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'FlutterHybird',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      //window.defaultRouteName可以接收從Native傳過來的參數(shù)
      home: _widgetForRoute(window.defaultRouteName),
    );
  }
}

Widget _widgetForRoute(String route) {
  switch (route) {
    case 'route1':
      return Center(
        child: Text("route1"),
      );
    case 'route2':
      return Center(
        child: Text("route2"),
      );
    default:
      return Center(
        child: Text('Unknown route: $route', textDirection: TextDirection.ltr),
      );
  }
}
  1. 直接啟動一個FlutterActivity

我們可以直接在Android的MainActivity中啟動一個FlutterActivity侍匙,這里的initialRoute方法中傳遞的參數(shù)就對應Flutter層的window.defaultRouteName

findViewById(R.id.jump).setOnClickListener(v -> {
        startActivity(FlutterActivity.withNewEngine()
              .initialRoute("route2")
              .build(MainActivity.this));
});

注意:需要在AndroidManifest.xml注冊FlutterActivity

<activity
    android:name="io.flutter.embedding.android.FlutterActivity"
 android:configChanges="orientation|keyboardHidden|keyboard|screenSize|locale|layoutDirection|fontScale|screenLayout|density|uiMode"
    android:hardwareAccelerated="true"
    android:windowSoftInputMode="adjustResize" />
  1. 啟動復寫后的FlutterActivity(推薦)

自己創(chuàng)建一個FlutterAppActivity繼承自FlutterActivity

public class FlutterAppActivity extends FlutterActivity {
    public final static String INIT_PARAMS = "initParams";
    private String initParams;

    public static void start(Context context, String initParams) {
        Intent intent = new Intent(context, FlutterAppActivity.class);
        intent.putExtra(INIT_PARAMS, initParams);
        context.startActivity(intent);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        initParams = getIntent().getStringExtra(INIT_PARAMS);
    }

    /**
     * 重載該方法來傳遞初始化參數(shù)
     *
     * @return
     */
    @NonNull
    @Override
    public String getInitialRoute() {
        return initParams == null ? super.getInitialRoute() : initParams;
    }
}

MainActivity中啟動FlutterAppActivity(另外別忘了在AndroidManifest.xml中注冊FlutterAppActivity

findViewById(R.id.jump).setOnClickListener(v -> {
    FlutterAppActivity.start(MainActivity.this,"route2");
});

兩種啟動方式的區(qū)別

如果單純只是想打開一個Flutter頁面氮惯,兩種方式實際上基本沒有太大區(qū)別,第一種方式也許還會更簡單一點想暗。但是妇汗,在Flutter開發(fā)中,我們往往還需要開發(fā)一些Native插件供Flutter調用说莫,如果使用復寫FlutterActivity的方式更有利于我們在FlutterActivity中注冊我們的Native插件杨箭,所以實際開發(fā)中一般推薦使用第二種方式

擴展思考

initialRoute從名稱上看起來是Flutter提供給我們進行Native與Flutter交互的路由跳轉的,但是實際上他就是一個字符串储狭,我們不僅僅可以傳遞一個路由名稱互婿,有時候我們也可以通過這個參數(shù)傳遞一串JSON數(shù)據(jù),然后在Flutter端進行解析辽狈,這樣我們就可以通過這個參數(shù)做更多的事情

在Android原生頁面中嵌入FlutterUI組件

activity_main.xml

FrameLayout用于承載Flutter組件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:background="@color/colorAccent"
        android:gravity="center"
        android:text="Hello World!" />

    <FrameLayout
        android:id="@+id/container"
        android:layout_width="wrap_content"
        android:layout_height="0dp"
        android:layout_weight="1">

    </FrameLayout>

</LinearLayout>

MainActivity.java

使用FragmentManagerFlutterFragment添加到FrameLayout容器中

FlutterFragment fragment = FlutterFragment.withNewEngine().initialRoute("route1").build();
getSupportFragmentManager()
        .beginTransaction()
        .replace(R.id.container, fragment)
        .commit();

運行結果

上半部分是原生的TextView慈参,下半部分是Flutter的Text組件

總結

本節(jié)主要介紹了Native和Flutter之間的頁面跳轉,以及同一個頁面中Native與Flutter組件的組合刮萌。接下來會介紹如何編寫Android插件與Flutter進行數(shù)據(jù)交互

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末驮配,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌壮锻,老刑警劉巖琐旁,帶你破解...
    沈念sama閱讀 206,126評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異躯保,居然都是意外死亡旋膳,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,254評論 2 382
  • 文/潘曉璐 我一進店門途事,熙熙樓的掌柜王于貴愁眉苦臉地迎上來验懊,“玉大人,你說我怎么就攤上這事尸变∫逋迹” “怎么了?”我有些...
    開封第一講書人閱讀 152,445評論 0 341
  • 文/不壞的土叔 我叫張陵召烂,是天一觀的道長碱工。 經(jīng)常有香客問我,道長奏夫,這世上最難降的妖魔是什么怕篷? 我笑而不...
    開封第一講書人閱讀 55,185評論 1 278
  • 正文 為了忘掉前任,我火速辦了婚禮酗昼,結果婚禮上廊谓,老公的妹妹穿的比我還像新娘。我一直安慰自己麻削,他們只是感情好蒸痹,可當我...
    茶點故事閱讀 64,178評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著呛哟,像睡著了一般叠荠。 火紅的嫁衣襯著肌膚如雪捂掰。 梳的紋絲不亂的頭發(fā)上蚁飒,一...
    開封第一講書人閱讀 48,970評論 1 284
  • 那天,我揣著相機與錄音兢卵,去河邊找鬼鳖孤。 笑死借帘,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的淌铐。 我是一名探鬼主播肺然,決...
    沈念sama閱讀 38,276評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼腿准!你這毒婦竟也來了际起?” 一聲冷哼從身側響起拾碌,我...
    開封第一講書人閱讀 36,927評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎街望,沒想到半個月后校翔,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,400評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡灾前,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 35,883評論 2 323
  • 正文 我和宋清朗相戀三年防症,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片哎甲。...
    茶點故事閱讀 37,997評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡蔫敲,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出炭玫,到底是詐尸還是另有隱情奈嘿,我是刑警寧澤,帶...
    沈念sama閱讀 33,646評論 4 322
  • 正文 年R本政府宣布吞加,位于F島的核電站裙犹,受9級特大地震影響,放射性物質發(fā)生泄漏衔憨。R本人自食惡果不足惜叶圃,卻給世界環(huán)境...
    茶點故事閱讀 39,213評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望践图。 院中可真熱鬧掺冠,春花似錦、人聲如沸平项。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,204評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽闽瓢。三九已至,卻和暖如春心赶,著一層夾襖步出監(jiān)牢的瞬間扣讼,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,423評論 1 260
  • 我被黑心中介騙來泰國打工缨叫, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留椭符,地道東北人。 一個月前我還...
    沈念sama閱讀 45,423評論 2 352
  • 正文 我出身青樓耻姥,卻偏偏與公主長得像销钝,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子琐簇,可洞房花燭夜當晚...
    茶點故事閱讀 42,722評論 2 345