使用flutter_boost混合開發時,android端返回鍵返回時數據回傳問題

  • flutter_boost版本(1.17.1, 1.22.4)

在使用flutter_boost進行混合開發時,有時候需要在關閉頁面時向前一頁面回傳數據,原生端處理時一般是通過startActivityForResult啟動頁面,然后在頁面的onActivityResult中接收回傳內容。flutter_boost提供以下回傳方式:

  • Flutter->Native
    flutter中打開native頁面,并從native回傳數據。常見場景例如flutter中需要選擇文件或圖片上傳時,需要打開原生頁面選擇,然后把文件路徑回傳給flutter。通過如下代碼以startActivityForResult方式啟動native頁面,native頁面關閉時setResult即可
FlutterBoost.singleton.open('url').then((result){...})

// native側setResult
Map map = new HashMap<String, String>();
map.put("a", "a");
Intent intent = getIntent().putExtra(IFlutterViewContainer.RESULT_KEY, (Serializable) map);
setResult(0, intent);
  • FlutterA->FlutterB
FlutterBoost.singleton
  .open('FlutterB')
  .then((Map<dynamic, dynamic> value) {
    // 回傳數據處理
  });

// FlutterB close并回傳:
final BoostContainerSettings settings = BoostContainer.of(context).settings;
FlutterBoost.singleton
  .close(settings.uniqueId, result: <String, dynamic>{'result': 'data from FlutterB'});

目前在FlutterA->FlutterB情況下,回傳數據時,發現android端在物理/系統虛擬返回鍵返回時,沒辦法調用FlutterBoost.singleton.close回傳數據。查看boost源碼,back鍵返回時處理流程大概如下:

BoostFlutterActivity中onBackPressed

->FlutterActivityAndFragmentDelegate中onBackPressed

->mSyncer.onBackPressed

// BoostFlutterActivity.java
private FlutterActivityAndFragmentDelegate delegate;
public void onBackPressed() {
    this.delegate.onBackPressed();
}

// FlutterActivityAndFragmentDelegate.java
protected IOperateSyncer mSyncer;

public void onBackPressed() {
    this.mSyncer.onBackPressed();
    this.ensureAlive();
}

public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
    Log.v("FlutterActivityAndFragmentDelegate", "Creating FlutterView.");
    this.mSyncer = FlutterBoost.instance().containerManager().generateSyncer(this);
    ...
}

// FlutterBoost.java
static FlutterBoost sInstance = null;
private FlutterViewContainerManager mManager;
public static FlutterBoost instance() {
    if (sInstance == null) {
        sInstance = new FlutterBoost();
    }

    return sInstance;
}

public IContainerManager containerManager() {
    return sInstance.mManager;
}
// FlutterViewContainerManager.java
public IOperateSyncer generateSyncer(IFlutterViewContainer container) {
    Utils.assertCallOnMainThread();
    ContainerRecord record = new ContainerRecord(this, container);
    if (this.mRecordMap.put(container, record) != null) {
        Debuger.exception("container:" + container.getContainerUrl() + " already exists!");
    }

    this.mRefs.add(new FlutterViewContainerManager.ContainerRef(record.uniqueId(), container));
    return record;
}
// ContainerRecord.java
public void onBackPressed() {
    Utils.assertCallOnMainThread();
    if (this.mState == 0 || this.mState == 4) {
        Debuger.exception("state error");
    }

    HashMap<String, String> map = new HashMap();
    map.put("type", "backPressedCallback");
    map.put("name", this.mContainer.getContainerUrl());
    map.put("uniqueId", this.mUniqueId);
    FlutterBoost.instance().channel().sendEvent("lifecycle", map);
}

查看部分源碼可見按下back鍵后,flutter_boost native側會通過channel("flutter_boost")向flutter側發送名為lifecycle的事件,類型為backPressedCallback,flutter_boost的flutter側在FlutterBoost單例初始化時,會初始化ContainerCoordinator,其中會注冊"lifecycle"的監聽。部分代碼如下:

  // flutter_boost.dart
  FlutterBoost() {
    ContainerCoordinator(_boostChannel);
  }

  // container_coordinator.dart
  ContainerCoordinator(BoostChannel channel) {
    assert(_instance == null);

    _instance = this;

    channel.addEventListener("lifecycle",
        (String name, Map arguments) => _onChannelEvent(arguments));

    channel.addMethodHandler((MethodCall call) => _onMethodCall(call));
  }

flutter_boost channel在收到事件時,會執行相關回調方法。back鍵按下后,lifecycle的回調會觸發,可見backPressedCallback類型的處理如下部分代碼所示:

Future<dynamic> _onChannelEvent(dynamic event) {
    if (event is Map) {
      Map map = event;
      final String type = map['type'];

      Logger.log('onEvent $type');

      switch (type) {
        //Handler back key pressed event.
        case 'backPressedCallback':
          {
            final String id = map['uniqueId'];
            FlutterBoost.containerManager
                ?.containerStateOf(id)
                ?.performBackPressed();
          }
          break;
        ...
      }
    }

    return Future<dynamic>(() {});
  }

跟進performBackPressed可見,其中backPressedHandler在state初始化時設置為maybePop,maybePop中會有頁面關閉的相關處理

  void performBackPressed() {
    Logger.log('performBackPressed');

    backPressedHandler?.call();
  }

  @override
  void initState() {
    super.initState();
    backPressedHandler = () => maybePop();
  }

  @override
  Future<bool> maybePop<T extends Object>([T result]) async {
    if(routerHistory.isEmpty){
      pop(result);
      return true;
    }

    final Route<T> route = routerHistory.last;
    final RoutePopDisposition disposition = await route.willPop();
    if (mounted) {
      switch (disposition) {
        case RoutePopDisposition.pop:
          pop(result);
          return true;
          break;
        case RoutePopDisposition.doNotPop:
          return false;
          break;
        case RoutePopDisposition.bubble:
          pop(result);
          return true;
          break;
      }
    }
    return false;
  }

了解以上流程后,我們大概可以明白為何back鍵返回時,無法回傳內容了,因為maybePop未傳入相關result內容。其實想想可可以理解,因為框架也不知道我們具體要回傳什么。查看上面maybePop的源碼,我們會發現一個關鍵的地方

final RoutePopDisposition disposition = await route.willPop();

按下back時頁面是否pop,完全取決于willPop的返回值。route默認為MaterialPageRoute,其繼承自PageRoute,PageRoute繼承自ModalRoute,在ModalRoute中我們可以看到willPop的具體實現。可見WillPopCallbacks非空且其中callback返回值為false時,返回值為RoutePopDisposition.doNotPop,maybePop中頁面才不會關閉。

  @override
  Future<RoutePopDisposition> willPop() async {
    final _ModalScopeState<T> scope = _scopeKey.currentState;
    assert(scope != null);
    for (final WillPopCallback callback in List<WillPopCallback>.from(_willPopCallbacks)) {
      if (await callback() != true)
        return RoutePopDisposition.doNotPop;
    }
    return await super.willPop();
  }

了解了以上流程,我們可以在FlutterB中添加WillPopScope攔截,onWillPop中返回false即可,這樣flutter_boost不會關閉當前頁面,我們在onWillPop中調用Navigator的pop關閉頁面和回傳數據即可。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,646評論 6 533
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,595評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,560評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,035評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,814評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,224評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,301評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,444評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,988評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,804評論 3 355
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,998評論 1 370
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,544評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,237評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,665評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,927評論 1 287
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,706評論 3 393
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,993評論 2 374

推薦閱讀更多精彩內容