Flutter

Flutter 之 通知 Notification (五十二)

2022-05-05  本文已影响0人  maskerII

通知(Notification)是Flutter中一个重要的机制,在widget树中,每一个节点都可以分发通知,通知会沿着当前节点向上传递,所有父节点都可以通过NotificationListener来监听通知。Flutter中将这种由子向父的传递通知的机制称为通知冒泡(Notification Bubbling)。通知冒泡和用户触摸事件冒泡是相似的,但有一点不同:通知冒泡可以中止,但用户触摸事件不行。

注意:通知冒泡和Web开发中浏览器事件冒泡原理是相似的,都是事件从出发源逐层向上传递,我们可以在上层节点任意位置来监听通知/事件,也可以终止冒泡过程,终止冒泡后,通知将不会再向上传递。

1. 监听通知

Flutter中很多地方使用了通知,如前面介绍的 Scrollable 组件,它在滑动时就会分发滚动通知(ScrollNotification),而 Scrollbar 正是通过监听 ScrollNotification 来确定滚动条位置的。


class MSNotificationDemo1 extends StatelessWidget {
  const MSNotificationDemo1({Key? key}) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text("NotificationDemo1")),
      body: NotificationListener(
        onNotification: (notification) {
          switch (notification.runtimeType) {
            case ScrollStartNotification:
              print("开始滚动");
              break;
            case ScrollUpdateNotification:
              print("正在滚动");
              break;
            case ScrollEndNotification:
              print("结束滚动");
              break;
            default:
          }
          return true;
        },
        child: ListView.builder(
          itemBuilder: (ctx, index) {
            return ListTile(
              title: Text("$index"),
            );
          },
          itemCount: 100,
          itemExtent: 56,
        ),
      ),
    );
  }
}

image.png

上例中的滚动通知如ScrollStartNotification、ScrollUpdateNotification等都是继承自ScrollNotification类,不同类型的通知子类会包含不同的信息,比如ScrollUpdateNotification有一个scrollDelta属性,它记录了移动的位移。

上例中,我们通过NotificationListener来监听子ListView的滚动通知的,NotificationListener定义如下:

  const NotificationListener({
    Key? key,
    required this.child,
    this.onNotification,
  }) :

我们可以看到:

//指定监听通知的类型为滚动结束通知(ScrollEndNotification)
NotificationListener<ScrollEndNotification>(
  onNotification: (notification) {
    // 只有滚动结束时才会回调
    print(notification.runtimeType);
    return true;
  },
  child: ListView.builder(
    itemBuilder: (ctx, index) {
      return ListTile(title: Text("$index"));
    },
    itemCount: 100,
  ),
),

上面代码运行后便只会在滚动结束时在控制台打印出通知的信息。

typedef NotificationListenerCallback<T extends Notification> = bool Function(T notification);

它的返回值类型为布尔值,当返回值为true时,阻止冒泡,其父级Widget将再也收不到该通知;当返回值为false 时继续向上冒泡通知。

Flutter的UI框架实现中,除了在可滚动组件在滚动过程中会发出ScrollNotification之外,还有一些其它的通知,如SizeChangedLayoutNotificationKeepAliveNotificationLayoutChangedNotification等,Flutter正是通过这种通知机制来使父元素可以在一些特定时机来做一些事情。

2. 自定义通知

除了 Flutter 内部通知,我们也可以自定义通知,下面我们看看如何实现自定义通知:

  1. 定义一个通知类,要继承自Notification类;
class MSCustomNotification extends Notification {
  MSCustomNotification(this.msg);
  final String msg;
}
  1. 分发通知。

Notification有一个dispatch(context)方法,它是用于分发通知的,我们说过context实际上就是操作Element的一个接口,它与Element树上的节点是对应的,通知会从context对应的Element节点向上冒泡。

示例


class MSCustomNotification extends Notification {
  MSCustomNotification(this.msg);
  final String msg;
}


class MSNotificationDemo3 extends StatefulWidget {
  const MSNotificationDemo3({Key? key}) : super(key: key);

  @override
  State<MSNotificationDemo3> createState() => _MSNotificationDemo3State();
}

class _MSNotificationDemo3State extends State<MSNotificationDemo3> {
  String _msg = "";
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text("NotificationDemo3")),
      // 监听指定通知 MSCustomNotification
      body: NotificationListener<MSCustomNotification>(
        onNotification: (notification) {
          _msg += notification.msg;
          setState(() {});
          return true; // 阻止冒泡
        },
        child: Center(
          child: Column(
            mainAxisAlignment: MainAxisAlignment.center,
            children: [
              // 不能这样写 
              // 此处context是根Context,而NotificationListener监听的是子Widget
              // ElevatedButton(
              //   onPressed: () => MSCustomNotification("Hi").dispatch(context),
              //   child: Text("Send Notification"),
              // ),
              Builder(
                builder: (context) {
                  return ElevatedButton(
                    onPressed: () {
                      // 分发通知
                      MSCustomNotification("Hi ").dispatch(context);
                    },
                    child: Text("Send Notification"),
                  );
                },
              ),
              Text(
                _msg,
                textScaleFactor: 1.5,
              ),
            ],
          ),
        ),
      ),
    );
  }
}

上面代码中,我们每点一次按钮就会分发一个MSCustomNotification类型的通知,我们在Widget根上监听通知,收到通知后我们将通知通过Text显示在屏幕上。

注意:代码中注释的部分是不能正常工作的,因为这个context是根Context,而NotificationListener是监听的子树,所以我们通过Builder来构建ElevatedButton,来获得按钮位置的context。

image.png

3. 阻止通知冒泡

我们将上面的例子改为:

class NotificationRouteState extends State<NotificationRoute> {
  String _msg="";
  @override
  Widget build(BuildContext context) {
    //监听通知
    return NotificationListener< MSCustomNotification >(
      onNotification: (notification){
        print(notification.msg); //打印通知
        return false;
      },
      child: NotificationListener< MSCustomNotification >(
        onNotification: (notification) {
          setState(() {
            _msg+=notification.msg+"  ";
          });
          return false; 
        },
        child: ...//省略重复代码
      ),
    );
  }
}

上列中两个NotificationListener进行了嵌套,子NotificationListeneronNotification回调返回了false,表示不阻止冒泡,所以父NotificationListener仍然会受到通知,所以控制台会打印出通知信息;如果将子NotificationListeneronNotification回调的返回值改为true,则父NotificationListener便不会再打印通知了,因为子NotificationListener已经终止通知冒泡了。

4. 冒泡原理

我们在上面介绍了通知冒泡的现象及使用,现在我们更深入一些,介绍一下Flutter框架中是如何实现通知冒泡的。
我们从通知分发的的源头出发,然后再顺藤摸瓜。由于通知是通过Notification的dispatch(context)方法发出的,那我们先看看dispatch(context)方法中做了什么,下面是相关源码

  void dispatch(BuildContext? target) {
    // The `target` may be null if the subtree the notification is supposed to be
    // dispatched in is in the process of being disposed.
    target?.visitAncestorElements(visitAncestor);
  }

dispatch(context)中调用了当前context的visitAncestorElements方法,该方法会从当前Element开始向上遍历父级元素;visitAncestorElements有一个遍历回调参数,在遍历过程中对遍历到的父级元素都会执行该回调。遍历的终止条件是:已经遍历到根Element或某个遍历回调返回false。源码中传给visitAncestorElements方法的遍历回调为visitAncestor方法,我们看看visitAncestor方法的实现:

//遍历回调,会对每一个父级Element执行此回调
bool visitAncestor(Element element) {
  //判断当前element对应的Widget是否是NotificationListener。
  
  //由于NotificationListener是继承自StatelessWidget,
  //故先判断是否是StatelessElement
  if (element is StatelessElement) {
    //是StatelessElement,则获取element对应的Widget,判断
    //是否是NotificationListener 。
    final StatelessWidget widget = element.widget;
    if (widget is NotificationListener<Notification>) {
      //是NotificationListener,则调用该NotificationListener的_dispatch方法
      if (widget._dispatch(this, element)) 
        return false;
    }
  }
  return true;
}

visitAncestor会判断每一个遍历到的父级Widget是否是NotificationListener,如果不是,则返回true继续向上遍历,如果是,则调用NotificationListener的_dispatch方法,我们看看_dispatch方法的源码:

  bool _dispatch(Notification notification, Element element) {
    // 如果通知监听器不为空,并且当前通知类型是该NotificationListener
    // 监听的通知类型,则调用当前NotificationListener的onNotification
    if (onNotification != null && notification is T) {
      final bool result = onNotification(notification);
      // 返回值决定是否继续向上遍历
      return result == true; 
    }
    return false;
  }

我们可以看到NotificationListener的onNotification回调最终是在_dispatch方法中执行的,然后会根据返回值来确定是否继续向上冒泡。

注意:1. Context上也提供了遍历Element树的方法。2. 我们可以通过Element.widget得到element节点对应的widget

上一篇下一篇

猜你喜欢

热点阅读