【Flutter学习】组件通信(父子、兄弟)
一,概述
flutter一个重要的特性就是组件化。组件分为两种状态,一种是StatefulWidget有状态组件,一种是StatelessWidget无状态组件。 无状态组件不能更新状态,有状态组件具有类似刷新的机制,可更改状态。
功能模块都可以通过继承两种状态组件实现功能模块封装。组件间通信,一般存在一下两种关系。
- 父子组件通信
- 兄弟组件通信
二, 通信实现方式
- 回调通信
- 需求“点击子组件,修改父组件的背景颜色与子组件背景颜色一致”
- 代码实现
- //父组件
- class ParentWidget extends StatefulWidget {
- final String title;
- ParentWidget({Key key,this.title}):super(key:key);
- @override
- State<StatefulWidget> createState() {
- return new ParentWidgetState();
- }
- }
- class ParentWidgetState extends State<ParentWidget> {
- Color containerBg = Colors.orange;
//回调函数 - void changeBackgroundColor(Color newColor){
- setState(() {
- containerBg = newColor;//修改状态
- });
- }
- @override
- Widget build(BuildContext context) {
- return new Scaffold(
- appBar: new AppBar(
- title: new Text(widget.title),
- ),
- body: new Center(
- child: new GestureDetector(
- onTap: (){
- changeBackgroundColor(Colors.orange);
- },
- child: new Container(
- width: 300,
- height: 300,
- color: containerBg,
- alignment: Alignment.center,
- child: new Row(
- mainAxisAlignment: MainAxisAlignment.spaceEvenly,
- children: <Widget>[
- new childrenA(childrenACallBack: changeBackgroundColor),
- new childrenB(childrenBCallBack: changeBackgroundColor),
- ],
- ),
- ),
- )
- ),
- );
- }
- }
- //子组件(组件A)
- class childrenA extends StatelessWidget {
//定义接收父类回调函数的指针
- final ValueChanged<Color> childrenACallBack;
- childrenA({Key key,this.childrenACallBack}):super(key:key);
- @override
- Widget build(BuildContext context) {
- return new GestureDetector(
- onTap: (){
//调用回调函数传值 - childrenACallBack(Colors.green);
- },
- child: new Container(
- width: 80,
- height: 80,
- color: Colors.green,
- child: new Text('ChildrenA'),
- ),
- );
- }
- }
- //子组件(组件B)
- class childrenB extends StatelessWidget {
- final ValueChanged<Color> childrenBCallBack;
- childrenB({Key key,this.childrenBCallBack}):super(key:key);
- @override
- Widget build(BuildContext context) {
- return new GestureDetector(
- onTap:(){
- childrenBCallBack(Colors.red);
- },
- child: new Container(
- width: 80,
- height: 80,
- color: Colors.red,
- child: new Text('ChildredB'),
- ),
- );
- }
- }
功能实现
- 使用场景:一般用于子组件对父组件传值。
InheritedWidget 数据共享
- 场景:业务开发中经常会碰到这样的情况,多个Widget需要同步同一份全局数据,比如点赞数、评论数、夜间模式等等。
- 代码实现:
- import 'package:flutter/material.dart';
- void main() => runApp(MyApp());
- class MyApp extends StatelessWidget {
- // This widget is the root of your application.
- @override
- Widget build(BuildContext context) {
- return MaterialApp(
- title: 'Flutter Demo',
- theme: ThemeData(
- primarySwatch: Colors.blue,
- ),
- home: new InheritedWidgetTestContainer(),
- );
- }
- }
- //模型数据
- class InheritedTestModel {
- final int count;
- const InheritedTestModel(this.count);
- }
- //哨所(自定义InheritedWidget类)
- class InheritedContext extends InheritedWidget {
- //构造函数
- InheritedContext({
- Key key,
- @required this.inheritedTestModel,
- @required this.increment,
- @required this.reduce,
- @required Widget child
- }):super(key:key,child:child);
- //变量
- final InheritedTestModel inheritedTestModel;
- final Function() increment;
- final Function() reduce;
- //静态方法
- static InheritedContext of(BuildContext context){
- InheritedContext contexts = context.inheritFromWidgetOfExactType(InheritedContext);
- return context.inheritFromWidgetOfExactType(InheritedContext);
- }
- //是否重建取决于Widget组件是否相同
- @override
- bool updateShouldNotify(InheritedContext oldWidget) {
- return inheritedTestModel != oldWidget.inheritedTestModel;
- }
- }
- class TestWidgetA extends StatelessWidget {
- @override
- Widget build(BuildContext context) {
- final inheritedContext = InheritedContext.of(context);
- return new Padding(
- padding: const EdgeInsets.only(left: 10.0,top: 10.0,right: 10.0),
- child: new RaisedButton(
- textColor: Colors.black,
- child: new Text('+'),
- onPressed:inheritedContext.increment
- ),
- );
- }
- }
- class TestWidgetB extends StatelessWidget {
- @override
- Widget build(BuildContext context) {
- final inheritedContext = InheritedContext.of(context);
- return new Padding(
- padding: const EdgeInsets.only(left: 10,top: 10,right: 10.0),
- child: new RaisedButton(
- textColor: Colors.black,
- child: new Text('-'),
- onPressed: inheritedContext.reduce
- ),
- );
- }
- }
- class TestWidgetC extends StatelessWidget {
- @override
- Widget build(BuildContext context) {
- final inheritedContext = InheritedContext.of(context);
- final inheritedTestModel = inheritedContext.inheritedTestModel;
- return new Padding(
- padding: const EdgeInsets.only(left: 10.0,top: 10.0,right: 10.0),
- child: new RaisedButton(
- textColor: Colors.black,
- child: new Text('${inheritedTestModel.count}'),
- onPressed: (){
- },
- ),
- );
- }
- }
- class InheritedWidgetTestContainer extends StatefulWidget {
- @override
- State<StatefulWidget> createState() {
- return new InheritedWidgetTestContainerState();
- }
- }
- class InheritedWidgetTestContainerState extends State<InheritedWidgetTestContainer> {
- InheritedTestModel _inheritedTestModel;
- _initData(){
- _inheritedTestModel = new InheritedTestModel(0);
- }
- @override
- void initState() {
- _initData();
- super.initState();
- }
- _incrementCount(){
- setState(() {
- _inheritedTestModel = new InheritedTestModel(_inheritedTestModel.count + 1);
- });
- }
- _reduceCount(){
- setState(() {
- _inheritedTestModel = new InheritedTestModel(_inheritedTestModel.count - 1);
- });
- }
- @override
- Widget build(BuildContext context) {
- return new InheritedContext(
- inheritedTestModel: _inheritedTestModel,
- increment: _incrementCount,
- reduce: _reduceCount,
- child: new Scaffold(
- appBar: new AppBar(
- title: new Text('InheritedWidgetTest'),
- ),
- body: new Center(
- child: new Column(
- children: <Widget>[
- new TestWidgetA(),
- new TestWidgetB(),
- new TestWidgetC(),
- ],
- ),
- )
- ),
- );
- }
- }
- 功能实现
使用场景
一般用于父组件对子组件的跨组件传值。
- Global Key通信
GlobalKey能够跨Widget访问状态。- 需求“点击A子组件,修改B子组件的背景颜色为指定的‘蓝色”
- 代码实现
- //父组件
- class ParentWidget extends StatefulWidget {
- @override
- State<StatefulWidget> createState() {
- return new ParentWidgetState();
- }
- }
- class ParentWidgetState extends State<ParentWidget> {
- @override
- Widget build(BuildContext context) {
- return new Scaffold(
- appBar: new AppBar(
- title: new Text('组件化'),
- ),
- body: new Center(
- child: new Container(
- color: Colors.grey,
- width: 200,
- height: 200,
- child: new Row(
- mainAxisAlignment: MainAxisAlignment.spaceEvenly,
- children: <Widget>[
- new SubWidgetA(key: subAkey),
- new SubWidgetB(key: subBkey)
- ],
- ),
- ),
- ),
- );
- }
- }
- //子组件A
- class SubWidgetA extends StatefulWidget {
- SubWidgetA({Key key}):super(key:key);
- @override
- State<StatefulWidget> createState() {
- return new SubWidgetAState();
- }
- }
- class SubWidgetAState extends State <SubWidgetA> {
- Color _backgroundColors = Colors.red;//红色
- void updateBackGroundColors(Color colos){
- setState(() {
- _backgroundColors = colos;
- });
- }
- @override
- Widget build(BuildContext context) {
- return new GestureDetector(
- onTap: (){
- subBkey.currentState.updateBackGroundColors(Colors.blue);
- setState(() {
- _backgroundColors = Colors.red;
- });
- },
- child: new Container(
- width: 80,
- height: 80,
- color:_backgroundColors,
- alignment: Alignment.center,
- child: new Text('SubWidgetA'),
- ),
- );
- }
- }
- //子组件B
- class SubWidgetB extends StatefulWidget {
- SubWidgetB({Key key}):super(key:key);
- @override
- State<StatefulWidget> createState() {
- // TODO: implement createState
- return new SubWidgetBState();
- }
- }
- class SubWidgetBState extends State<SubWidgetB> {
- Color _backgroundColors = Colors.green;//绿色
- void updateBackGroundColors(Color colos){
- setState(() {
- _backgroundColors = colos;
- });
- }
- @override
- Widget build(BuildContext context) {
- return new GestureDetector(
- onTap: (){
- subAkey.currentState.updateBackGroundColors(Colors.blue);
- setState(() {
- _backgroundColors = Colors.green;
- });
- },
- child: new Container(
- width: 80,
- height: 80,
- color: _backgroundColors,
- alignment: Alignment.center,
- child: new Text('SubWidgetB'),
- ),
- );
- }
- }
- //父组件
- 功能实现
- 使用场景:一般用于跨组件访问状态
- ValueNotifier通信
ValueNotifier是一个包含单个值的变更通知器,当它的值改变的时候,会通知它的监听。
- 定义ValueNotifierData类,继承ValueNotifier
- class ValueNotifierData extends ValueNotifier<String> {
- ValueNotifierData(value) : super(value);
- }
- class ValueNotifierData extends ValueNotifier<String> {
- 定义
_WidgetOne
,包含一个ValueNotifierData的实例。- class _WidgetOne extends StatefulWidget {
- _WidgetOne({this.data});
- final ValueNotifierData data;
- @override
- _WidgetOneState createState() => _WidgetOneState();
- }
- class _WidgetOne extends StatefulWidget {
_WidgetOneState
中给ValueNotifierData实例添加监听。- @override
- initState() {
- super.initState();
- widget.data.addListener(_handleValueChanged);
- info = 'Initial mesage: ' + widget.data.value;
- }
- void _handleValueChanged() {
- setState(() {
- info = 'Message changed to: ' + widget.data.value;
- });
- @override
- 在
ValueNotifierCommunication
组件中实例化_WidgetOne
,可以通过改变ValueNotifierData
实例的value来触发_WidgetOneState
的更新。- @override
- Widget build(BuildContext context) {
- ValueNotifierData vd = ValueNotifierData('Hello World');
- return Scaffold(
- appBar: AppBar(title: Text('Value Notifier Communication'),),
- body: _WidgetOne(data: vd),
- floatingActionButton: FloatingActionButton(child: Icon(Icons.refresh),onPressed: () {
- vd.value = 'Yes';
- }),
- );
- }
- @override
- 定义ValueNotifierData类,继承ValueNotifier
- 第三方插件
在这里运用event_bus来实现传值,用于组件与组件之间的传值。- event_bus
- 引入插件
- import 'package:event_bus/event_bus.dart';
event_bus用法。
新建消息监测类
- import 'package:event_bus/event_bus.dart';
- EventBus eventBus = new EventBus();
- class TransEvent{
- String text;
- TransEvent(this.text);
- }
- import 'package:event_bus/event_bus.dart';
监测类变化
- eventBus.on<TransEvent>().listen((TransEvent data) => show(data.text));
- void show(String val) {
- setState(() {
- data = val;
- });
- }
- eventBus.on<TransEvent>().listen((TransEvent data) => show(data.text));
触发消息变化
- eventBus.fire(new TransEvent('$inputText'));
- 使用场景:这样我们就可以根据这些来实现组件之间的传值。
- 引入插件
- event_bus
【Flutter学习】组件通信(父子、兄弟)的更多相关文章
- react第六单元(react组件通信-父子组件通信-子父组件通信-跨级组件的传参方式-context方式的传参)
第六单元(react组件通信-父子组件通信-子父组件通信-跨级组件的传参方式-context方式的传参) #课程目标 1.梳理react组件之间的关系 2.掌握父子传值的方法 3.掌握子父传值的方法 ...
- vue组件之间的通信, 父子组件通信,兄弟组件通信
组件通讯包括:父子组件间的通信和兄弟组件间的通信.在组件化系统构建中,组件间通信必不可少的. 父组件--> 子组件 1. 属性设置 父组件关键代码如下: <template> < ...
- vue组件通信&&v兄弟组件通信eventbus遇到的问题(多次触发、第一次不触发)
组件通讯包括:父子组件间的通信和兄弟组件间的通信.在组件化系统构建中,组件间通信必不可少的 (vuex以后再说). 父组件--> 子组件 1. 属性设置 父组件关键代码如下: <templ ...
- 计算属性、侦听属性、局部与全局组件使用、组件通信(父子互传)、ref属性、动态组件和keep-alive、插槽
今日内容概要 计算属性 侦听属性 局部组件和全局组件 组件通信之父传子 组件通信之子传父 ref属性(组件间通信) 动态组件和keep-alive 插槽 内容详细 1.计算属性 # 插值的普通函数,只 ...
- vue组件详解——组件通信
每天学习一点点 编程PDF电子书.视频教程免费下载:http://www.shitanlife.com/code 组件之间通信可以用下图表示: 组件关系可分为父子组件通信.兄弟组件通信.跨级组件通信. ...
- vue组件详解(三)——组件通信
组件之间通信可以用下图表示: 组件关系可分为父子组件通信.兄弟组件通信.跨级组件通信. 一.自定义事件 当子组件需要向父组件传递数据时,就要用到自定义事件. 子组件用$emit ()来触发事件,父组件 ...
- vue2.0 $emit $on组件通信
在vue1.0中父子组件通信使用$dispatch 和 $broadcast,但是在vue2.0中$dispatch 和 $broadcast 已经被弃用. 因为基于组件树结构的事件流方式实在是让人难 ...
- vue 2 使用Bus.js进行兄弟(非父子)组件通信 简单案例
vue2中废弃了$dispatch和$broadcast广播和分发事件的方法.父子组件中可以用props和$emit().如何实现非父子组件间的通信,可以通过实例一个vue实例Bus作为媒介,要相互通 ...
- vue2.0 父子组件通信 兄弟组件通信
父组件是通过props属性给子组件通信的来看下代码: 父组件: <parent> <child :child-com="content"></chil ...
随机推荐
- MaxCompute studio FAQ
1. 官方文档地址 https://help.aliyun.com/document_detail/50889.html 2. Show Table Detail 中文乱码 原因是Intellij A ...
- 使用C#获取IP地址方法
C#中如何获取IP地址?,看到问题的时候我也很纠结,纠结的不是这个问题是如何的难回答,而是纠结的是这些问题都是比较基本的常识,也是大家会经常用到的.但是却不断的有人问起,追根究底的原因估计就是没有好好 ...
- 七、单例设计模式共享数据分析、解决、call_once
一.设计模式大概谈 代码的一些写法,与常规的写法不太一样,程序灵活,维护起来很方便,但是别人接管.阅读代码很痛苦. 用设计模式理念写出来的代码很晦涩.<< head first>&g ...
- puppet(一种Linux、Unix、windows平台的集中配置管理系统)
puppet是一种Linux.Unix.windows平台的集中配置管理系统,使用自有的puppet描述语言,可管理配置文件.用户.cron任务.软件包.系统服务等.puppet把这些系统实体称之为资 ...
- 20 October in ss
Contest A: sum 快速读. B: 鬼谷子的钱袋(coin) 贪心. 按照类似二进制的方式准备钱袋:1, 2, 4, 8, ... 以此装入的钱袋数目记为 \(N\). 如果最后剩余不足以凑 ...
- Linux环境下对大小写敏感,linux环境升级node
linux对大小写敏感 在window下可以正常运行的代码,在linux环境下报错,找不到文件,因为window下对大小写不敏感,linux对大小写敏感 linux环境下node升级 1.安装nvm ...
- upc组队赛15 Made In Heaven【第K短路 A*】
Made In Heaven 题目描述 One day in the jail, F·F invites Jolyne Kujo (JOJO in brief) to play tennis with ...
- Java并发AtomicReference类
java.util.concurrent.atomic.AtomicReference类提供了可以原子读取和写入的底层对象引用的操作,还包含高级原子操作. AtomicReference支持对底层对象 ...
- (4.10)sql server导入导出
SQL Server快速导入数据分享 核心参考:官网~~https://docs.microsoft.com/zh-cn/sql/relational-databases/import-export/ ...
- webstorm的下载、破解、与汉化
其实很简单的事情,都被我弄复杂了倒腾了很久,特做个记录. 说明前提,版本为 webstorm 2018.1.4 一.下载webstorom 下载地址:当然去官网啊 https://www.jetbr ...