Flutter部件内部状态管理小结之实现Vue的v-model功能


Posted in Javascript onJune 11, 2019

Flutter部件内部状态管理

本文是 Flutter 部件内部状态管理的小结,从部件的基础开始,到部件的状态管理,并且在过程中实现一个类似 Vue 的 v-model 的功能。

widget 基础

widget(部件)

如 React 里万物皆组件, Java 里万物皆对象, Flutter 里,能看到的一切都是 widget(部件),如按钮、文本框等等。

Flutter 内部已经为我们做了一些基础的 widget ,例如:

  • Text : 这个就是一个文本部件,里面用于放置文本
  • Row , Column : 行列布局部件
  • Container : 可以理解为 HTML 里的 div

状态

状态可以理解为 widget 内拥有的成员变量

无状态 widget

无状态是指该成员变量不可改变,即使用 final 修饰符,为常量,创建无状态组件步骤如下:

创建一个类继承 StatelessWidget

实现 build 方法(类比写 HTML + CSS )

import 'package:flutter/material.dart';

class StateLessDemoWidget extends StatelessWidget {
 /// 如果定义非final修饰的成员变量,会提示
 /// This class (or a class which this class inherits from) is marked as '@immutable', but one or more of its instance fields are not final: StateLessDemoWidget.listInde
 /// int [listIndex];
 @override
 Widget build(BuildContext context) {
 return Container();
 }
}

有状态 widget

有状态则是指该 widget 内部的成员变量可以不使用 final 修饰符,并通过 setState() 方法改变成员变量的值时,引起 widget 的状态改变并重绘(例如文本框绑定变量A,通过改变变量A修改文本框的值,这里的变量A可以理解为可变状态)

有状态 widget 通过实现 StatefulWidget 和 State 两个子类实现,步骤如下:

  • 创建一个 Widget 类,继承 StatefulWidget
  • 创建一个继承 State 的类
  • 在 State 类里创建状态(成员变量),和实现 build 方法(画界面)
import 'package:flutter/material.dart';

/// 创建一个继承[StatefulWidget]的widget类
/// 这个类的效果类似收藏,有个空心的爱心图标
/// 点击一下爱心填充红色,再点击一次就取消填充
class FavoriteWidget extends StatefulWidget {
 @override
 _FavoriteWidgetState createState() => _FavoriteWidgetState();
}

/// [State]类,FavoriteWidget的具体实现
/// 包含一个状态[_isFavorited]
class _FavoriteWidgetState extends State<FavoriteWidget> {
 bool _isFavorited = true;

 @override
 Widget build(BuildContext context) {
 return Row(
 mainAxisSize: MainAxisSize.min,
 children: [
 Container(
 padding: EdgeInsets.all(0),
 child: IconButton(
 /// [_isFavorited] 为 true 时使用图标 Icons.star 否则使用 Icons.star_border
 icon: (_isFavorited ? Icon(Icons.star) : Icon(Icons.star_border)),

 color: Colors.red[500],

 /// 当点击时,改变状态
 onPressed: _toggleFavorite,
 ),
 ),
 ],
 );
 }

 /// 改变[_isFavorited]的状态
 void _toggleFavorite() {
 setState(() {
 if (_isFavorited) {
 _isFavorited = false;
 } else {
 _isFavorited = true;
 }
 });
 }
}

widget 管理内部状态的三种方式

widget 自己管理状态

其实和上边的 有状态部件 示例一样的,直接上代码

import 'package:flutter/material.dart';

class TapboxA extends StatefulWidget {
 @override
 _TapboxAState createState() => _TapboxAState();
}
/// [TapboxA]拥有状态[_active],通过[_handleTap]方法管理[_active]
class _TapboxAState extends State<TapboxA> {
 bool _active = false;

 void _handleTap() {
 setState(() {
 _active = !_active;
 });
 }
 Widget build(BuildContext context) {
 return GestureDetector(
 onTap: _handleTap,
 child: Center(
 child: Text(
 _active ? 'Active' : 'Inactive',
 ),
 ),
 );
 }
}

父widget 管理 子widget 状态

这一小节将实现一个和 Vue 的 v-model 一样的功能,父部件管理子部件的步骤如下:

  • 创建一个无状态部件作为子部件,构造函数中需要接收状态的值和状态改变时的回调函数
  • 创建一个有状态部件作为父部件,并定义需要管理的状态
  • 在父部件中创建子部件,并绑定状态和回调事件
import 'package:flutter/material.dart';

/// 父部件管理TapboxB的状态
///
/// 父部件[ParentWidget]定义了[_active]状态,并和[TapboxB]的[active]绑定
/// 当[TapboxB]被点击时,通过[onChanged]方法通知父部件,父部件修改[_active]的值
/// 也就间接修改了[TapboxB]的[active]
///
/// 如果了解过Vue的v-model原理的话比较好理解,其实这个和Vue的v-model一样的

//------------------------ ParentWidget --------------------------------

class ParentWidget extends StatefulWidget {
 @override
 _ParentWidgetState createState() => _ParentWidgetState();
}

class _ParentWidgetState extends State<ParentWidget> {
 bool _active = false;

 void _handleTapboxChanged(bool newValue) {
 setState(() {
 _active = newValue;
 });
 }

 @override
 Widget build(BuildContext context) {
 return Container(
 child: TapboxB(
 active: _active,
 onChanged: _handleTapboxChanged,
 ),
 );
 }
}

//------------------------- TapboxB ----------------------------------

// 注意[TapboxB]是无状态部件,所有成员变量使用 final 修饰
class TapboxB extends StatelessWidget {
 // 必须传递onChanged,用于通知父部件
 TapboxB({Key key, this.active: false, @required this.onChanged})
 : super(key: key);
 final bool active;
 final ValueChanged<bool> onChanged;

 void _handleTap() {
 // 通知父部件修改 active 的值
 onChanged(!active);
 }

 Widget build(BuildContext context) {
 return GestureDetector(
 // 被点击时
 onTap: _handleTap,
 child: Center(
 child: Text(
 // active 改变时修改文本的内容
 active ? 'Active' : 'Inactive',
 ),
 ),
 );
 }
}

混合管理

在这种管理模式下,和父管理子部件的区别是,子部件也需要管理自己的状态,所以子部件也将是有状态部件(区别)。

步骤如下:

  1. 创建一个【有状态部件】作为子部件,构造函数中需要接收状态的值和状态改变时的回调函数
  2. 为子部件定义子部件内部的状态(多的一步)
  3. 创建一个有状态部件作为父部件,并定义需要管理的状态
  4. 在父部件中创建子部件,并绑定状态和回调事件
import 'package:flutter/material.dart';
//---------------------------- ParentWidget ----------------------------

class ParentWidget extends StatefulWidget {
 @override
 _ParentWidgetState createState() => _ParentWidgetState();
}

class _ParentWidgetState extends State<ParentWidget> {
 bool _active = false;

 void _handleTapboxChanged(bool newValue) {
 setState(() {
 _active = newValue;
 });
 }

 @override
 Widget build(BuildContext context) {
 return Container(
 child: TapboxC(
 active: _active,
 onChanged: _handleTapboxChanged,
 ),
 );
 }
}

//----------------------------- TapboxC ------------------------------
class TapboxC extends StatefulWidget {
 TapboxC({Key key, this.active: false, @required this.onChanged})
 : super(key: key);

 final bool active;
 final ValueChanged<bool> onChanged;

 _TapboxCState createState() => _TapboxCState();
}

class _TapboxCState extends State<TapboxC> {
 /// [TapboxC]管理内部的边框是否高亮,通过状态[_highlight]来修改
 bool _highlight = false;

 void _handleTapDown(TapDownDetails details) {
 setState(() {
 _highlight = true;
 });
 }

 void _handleTapUp(TapUpDetails details) {
 setState(() {
 _highlight = false;
 });
 }

 void _handleTapCancel() {
 setState(() {
 _highlight = false;
 });
 }

 void _handleTap() {
 // 通知父组件
 widget.onChanged(!widget.active);
 }

 Widget build(BuildContext context) {
 return GestureDetector(
 // onXXX都是事件处理
 onTapDown: _handleTapDown,
 onTapUp: _handleTapUp,
 onTap: _handleTap,
 onTapCancel: _handleTapCancel,
 child: Container(
 child: Center(
 child: Text(widget.active ? 'Active' : 'Inactive',
 style: TextStyle(fontSize: 32.0, color: Colors.white)),
 ),
 width: 200.0,
 height: 200.0,
 decoration: BoxDecoration(
 color: widget.active ? Colors.lightGreen[700] : Colors.grey[600],
 // 边框是否高亮显示
 border: _highlight
 ? Border.all(
  color: Colors.teal[700],
  width: 10.0,
 )
 : null,
 ),
 ),
 );
 }
}

总结

以上所述是小编给大家介绍的Flutter部件内部状态管理小结之实现Vue的v-model功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!
如果你觉得本文对你有帮助,欢迎转载,烦请注明出处,谢谢!

Javascript 相关文章推荐
Add a Table to a Word Document
Jun 15 Javascript
javascript一些不错的函数脚本代码
Sep 10 Javascript
基于jquery的一个图片hover的插件
Apr 24 Javascript
js注意img图片的onerror事件的分析
Jan 01 Javascript
javascript Window及document对象详细整理
Jan 12 Javascript
jquery中邮箱地址 URL网站地址正则验证实例代码
Sep 15 Javascript
append和appendTo的区别以及appendChild用法
Dec 24 Javascript
实现网页页面跳转的几种方法(meta标签、js实现、php实现)
May 20 Javascript
ANGULARJS中使用JQUERY分页控件
Sep 16 Javascript
JS得到当前时间的方法示例
Mar 24 Javascript
JavaScript门道之标准库
May 26 Javascript
DatePickerDialog 自定义样式及使用全解
Jul 09 Javascript
JavaScript动态检测密码强度原理及实现方法详解
Jun 11 #Javascript
聊聊Vue 中 title 的动态修改问题
Jun 11 #Javascript
vue+element模态框中新增模态框和删除功能
Jun 11 #Javascript
vue.js中导出Excel表格的案例分析
Jun 11 #Javascript
ES6 Proxy实现Vue的变化检测问题
Jun 11 #Javascript
webpack实践之DLLPlugin 和 DLLReferencePlugin的使用教程
Jun 10 #Javascript
vue2 中二级路由高亮问题及配置方法
Jun 10 #Javascript
You might like
php使用pdo连接mssql server数据库实例
2014/12/25 PHP
Laravel Validator 实现两个或多个字段联合索引唯一
2019/05/08 PHP
JavaScript 字符串乘法
2009/08/20 Javascript
基于Jquery的$.cookie()实现跨越页面tabs导航实现代码
2011/03/03 Javascript
javascript 事件处理程序介绍
2012/06/27 Javascript
jquery插件之信息弹出框showInfoDialog(成功/错误/警告/通知/背景遮罩)
2013/01/09 Javascript
UpdatePanel和Jquery冲突的解决方法
2013/04/01 Javascript
js格式化货币数据实现代码
2013/09/04 Javascript
浅析基于WEB前端页面的页面内容搜索的实现思路
2014/06/10 Javascript
jQuery+Ajax+PHP+Mysql实现分页显示数据实例讲解
2015/09/27 Javascript
jQuery插件实现可输入和自动匹配的下拉框
2016/10/24 Javascript
weui框架实现上传、预览和删除图片功能代码
2017/08/24 Javascript
解决vue select当前value没有更新到vue对象属性的问题
2018/08/30 Javascript
vue微信分享的实现(在当前页面分享其他页面)
2019/04/16 Javascript
jquery多级树形下拉菜单的实例代码
2019/07/09 jQuery
jquery自定义组件实例详解
2020/12/31 jQuery
django 2.0更新的10条注意事项总结
2018/01/05 Python
python导出hive数据表的schema实例代码
2018/01/22 Python
python文件拆分与重组实例
2018/12/10 Python
Python列表常见操作详解(获取,增加,删除,修改,排序等)
2019/02/18 Python
Django中Middleware中的函数详解
2019/07/18 Python
简单了解Django应用app及分布式路由
2019/07/24 Python
python-numpy-指数分布实例详解
2019/12/07 Python
谈谈python垃圾回收机制
2020/09/27 Python
python实现马丁策略回测3000只股票的实例代码
2021/01/22 Python
给酒店员工的表扬信
2014/01/11 职场文书
生物制药自我鉴定
2014/01/25 职场文书
焦裕禄精神心得体会
2014/09/02 职场文书
2015年质量月活动总结报告
2015/03/27 职场文书
繁星春水读书笔记
2015/06/30 职场文书
校园歌手大赛主持词
2015/07/03 职场文书
为什么MySQL 删除表数据 磁盘空间还一直被占用
2021/10/16 MySQL
Python实现Matplotlib,Seaborn动态数据图
2022/05/06 Python
Windows Server 2008配置防火墙策略详解
2022/06/28 Servers
MyBatis在注解上使用动态SQL方式(@select使用if)
2022/07/07 Java/Android
Python中np.random.randint()参数详解及用法实例
2022/09/23 Python