likes
comments
collection
share

Flutter 封装:最佳实践 —— AppLifecycleState 状态监听封装

作者站长头像
站长
· 阅读数 47

备注:Flutter v3.13 版本以下推荐使用;以上使用官方实现 AppLifecycleListener;

一、需求来源

项目中多页面需要同时监听 AppLifecycleState 根据状态实现不同的业务逻辑;尝试多种实现,使用起来比较麻烦,为了开发效率,随封装一个。代码不复杂,核心在于极简的封装方法。

效果如下:

//进入后台
[log] 2024-01-21 09:44:59.923812 MyHomePage onInactive
[log] 2024-01-21 09:44:59.926700 AppLifecycleStateObserverDemo onInactive
[log] 2024-01-21 09:45:00.906268 MyHomePage onHidden
[log] 2024-01-21 09:45:00.906750 AppLifecycleStateObserverDemo onHidden
[log] 2024-01-21 09:45:00.907225 MyHomePage onPause
[log] 2024-01-21 09:45:00.907635 AppLifecycleStateObserverDemo onPause

//进入前台
[log] 2024-01-21 09:45:12.167072 MyHomePage onHidden
[log] 2024-01-21 09:45:12.167362 AppLifecycleStateObserverDemo onHidden
[log] 2024-01-21 09:45:12.167607 MyHomePage onInactive
[log] 2024-01-21 09:45:12.167819 AppLifecycleStateObserverDemo onInactive
[log] 2024-01-21 09:45:13.071467 MyHomePage onResume
[log] 2024-01-21 09:45:13.072509 AppLifecycleStateObserverDemo onResume

// MyHomePage 为主页面;
// AppLifecycleStateObserverDemo 为特定业务界面;

二、使用示例

import 'package:flutter/material.dart';
import 'package:flutter_templet_project/util/AppLifecycleObserver.dart';
import 'package:flutter_templet_project/util/debug_log.dart';

class AppLifecycleStateObserverDemo extends StatefulWidget {

  AppLifecycleStateObserverDemo({
    super.key, 
    this.title
  });

  final String? title;

  @override
  State<AppLifecycleStateObserverDemo> createState() => _AppLifecycleStateObserverDemoState();
}

class _AppLifecycleStateObserverDemoState extends State<AppLifecycleStateObserverDemo> with
    AppLifecycleObserverMixin {

  final _scrollController = ScrollController();


  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title ?? "$widget"),
        actions: ['done',].map((e) => TextButton(
          child: Text(e,
            style: TextStyle(color: Colors.white),
          ),
          onPressed: () => debugPrint(e),)
        ).toList(),
      ),
      body: buildBody(),
    );
  }

  buildBody() {
    return Scrollbar(
      controller: _scrollController,
      child: SingleChildScrollView(
        controller: _scrollController,
        child: Column(
          children: [
            Text("$widget"),
          ],
        ),
      ),
    );
  }

  /*************** AppLifecycleObserverMixin ***************/
  
  @override
  Future<void> onResume() async {
    // TODO: implement onResume
    DDLog("$widget onResume");
  }

  @override
  Future<void> onInactive() async {
    // TODO: implement onInactive
    DDLog("$widget onInactive");
  }

  @override
  Future<void> onPause() async {
    // TODO: implement onPause
    DDLog("$widget onPause");
  }

  @override
  Future<void> onDetached() async {
    // TODO: implement onDetached
    DDLog("$widget onDetached");
  }
  
  @override
  Future<void> onHidden() async {
    // TODO: implement onHidden
    ddLog("$widget onHidden");
  }
}

三、源码

1、AppLifecycleObserver 源码

/// app 前后台生命周期函数监听
class AppLifecycleObserver extends WidgetsBindingObserver{

  AppLifecycleObserver({
    required this.onResume,
    required this.onInactive,
    required this.onPause,
    required this.onDetached,
    required this.onHidden,
    this.data,
  });

  final AsyncCallback onResume;
  final AsyncCallback onInactive;
  final AsyncCallback onPause;
  final AsyncCallback onDetached;
  final AsyncCallback onHidden;
  final dynamic data;

  @override
  void didChangeAppLifecycleState(AppLifecycleState state) {
    // DDLog('${Get.currentRoute} APP状态监听:$state, ${data ?? ""}');

    switch (state) {
      case AppLifecycleState.resumed:
        onResume();
        break;
      case AppLifecycleState.inactive:
        onInactive();
        break;
      case AppLifecycleState.paused:
        onPause();
        break;
      case AppLifecycleState.detached:
        onDetached();
        break;
     case AppLifecycleState.hidden:
       onHidden();
        break;
    }
  }
}

2、AppLifecycleObserverMixin 源码

/// app 前后台生命周期函数混入封装
mixin AppLifecycleObserverMixin<T extends StatefulWidget> on State<T>{
  late final _lifecycleEvent = AppLifecycleObserver(
    onResume: onResume,
    onInactive: onInactive,
    onPause: onPause,
    onDetached: onDetached,
  );

  @override
  void initState() {
    super.initState();
    WidgetsBinding.instance.addObserver(_lifecycleEvent);
  }

  @override
  void dispose() {
    WidgetsBinding.instance.removeObserver(_lifecycleEvent);
    super.dispose();
  }

  Future<void> onResume() async {
    throw UnimplementedError("❌: $this 未实现 onResume");
  }

  Future<void> onInactive() async {
    throw UnimplementedError("❌: $this 未实现 onInactive");
  }

  Future<void> onPause() async {
    throw UnimplementedError("❌: $this 未实现 onPause");
  }

  Future<void> onDetached() async {
    throw UnimplementedError("❌: $this 未实现 onDetached");
  }
  
  Future<void> onHidden() async {
   throw UnimplementedError("❌: $this 未实现 onHidden");
  }

}

四、总结

1、Flutter 3.13 之后官方实现了一个 AppLifecycleListener

使用效果和 AppLifecycleObserverMixin 相同,基于官方优先原则(未来维护性考虑),建议大家使用官方 AppLifecycleListener实现功能。

class AppLifecycleListener with WidgetsBindingObserver, Diagnosticable {
  /// Creates an [AppLifecycleListener].
  AppLifecycleListener({
    WidgetsBinding? binding,
    this.onResume,
    this.onInactive,
    this.onHide,
    this.onShow,
    this.onPause,
    this.onRestart,
    this.onDetach,
    this.onExitRequested,
    this.onStateChange,
  })  : binding = binding ?? WidgetsBinding.instance,
        _lifecycleState = (binding ?? WidgetsBinding.instance).lifecycleState {
    this.binding.addObserver(this);
  }
  。。。
2、AppLifecycleObserverMixin 生命周期方法是一组,即使不使用也要全部实现(基于维护性考虑);如果未实现会日志报错,提示到具体页面;
3、v3.13 以下使用请删除 onHidden 相关代码;

github

转载自:https://juejin.cn/post/7325693010570100786
评论
请登录