flutter中使用redux之多界面互动
在上一篇文章,我们介绍了如何在flutter中使用redux。在上一篇文章的例子中,我们使用了单界面集成redux,但是在实际项目中,我们通常涉及多个模块,每个模块涉及多个界面,那么如何使用redux整合模块,并实现模块和界面间的消息传递呢?
例子:登录
接着上篇文章,这次的例子稍微复杂一点:
目标:
- 实现登录界面,实现基本登录逻辑
- 成功之后将结果通知到其他界面
先将AppState等状态有关的移动到reducers.dart,创建LoginPage.dart
LoginPage.dart代码如下:
import 'package:flutter/material.dart';
import 'package:flutter_redux/flutter_redux.dart';
import 'package:flutter_use_redux/reducers.dart';
import 'package:redux/redux.dart';
import 'dart:async';
import 'package:flutter/cupertino.dart';
typedef Future CallLogin(String account,String pwd);
class LoginPageState extends State<LoginPage>{
String _account;
String _pwd;
bool isLogin;
@override
void initState() {
super.initState();
}
@override
void dispose() {
super.dispose();
}
@override
Widget build(BuildContext context) {
return new Scaffold(
appBar: new AppBar(
title: new Text("登录"),
),
body: new Form(
onChanged: (){
print("changed");
},
onWillPop: () async{
return true;
},
child: new Padding(padding: new EdgeInsets.all(10.0),child: new Column(
children: <Widget>[
new TextFormField( decoration:new InputDecoration(labelText: "请输入账号") ,
onSaved: (String value){
_account = value;
}, ///保持一下输入的账号
validator: (String value)=> value.isEmpty ? "请输入账号" : null, ),
new TextFormField(decoration:new InputDecoration(labelText: "请输入密码"),
onSaved: (String value)=>_pwd = value, ///保持一下输入的密码
validator: (String value)=> value.isEmpty ? "请输入密码" : null),
new FormField(builder: (FormFieldState s){
return new Center(
child: new RaisedButton(onPressed: () async{
FormState state = Form.of(s.context);
if(state.validate()){
//如果验证成功,那么执行登录逻辑
state.save();
print("Login success $_account" );
//这里交给设置好的逻辑去执行操作
try{
await widget.callLogin(_account,_pwd);
showDialog(context: context,builder: (c){
return new CupertinoAlertDialog(
title: new Text("登录成功"),
actions: <Widget>[
new Center(
child: new RaisedButton(
onPressed:(){
Navigator.of(context).pop();
Navigator.of(context).pop();
},
child: new Text("ok"),
)
)
],
);
});
// Navigator.of(context).pop();
}catch(e){
showDialog(context: context,builder: (c){
return new CupertinoAlertDialog(
title: new Text("登录失败$e"),
actions: <Widget>[
new Center(
child: new RaisedButton(
onPressed:(){
Navigator.of(context).pop();
},
child: new Text("ok"),
)
)
],
);
});
///登录失败,提示一下用户
print("登录失败! $e");
}
}
},child: new Text("提交"),)
);
})
],
),)),
);
}
}
class LoginPage extends StatefulWidget{
CallLogin callLogin;
LoginPage({this.callLogin});
@override
State<StatefulWidget> createState() {
return new LoginPageState();
}
}
注意:这个组件其实并没有使用redux,登录逻辑使用外部传递过来的函数来处理:
CallLogin callLogin;
LoginPage({this.callLogin});
...
//执行登录逻辑
await widget.callLogin(_account,_pwd);
...
为什么要这么做呢?好处有哪些?
- 减少组件之间的依赖关系
- 减少本类的职责,将本类的职责变成只展示ui
- 本类可单独单元测试,单独工作,只要传进来一个模拟的逻辑函数
那么在哪里将登录逻辑传递进来呢?
···
routes: {
"login":(BuildContext context)=>new StoreConnector(builder: ( BuildContext context,Store<AppState> store ){
return new LoginPage(callLogin: (String account,String pwd) async{
print("正在登录,账号$account,密码:$pwd");
// 为了模拟实际登录,这里等待一秒
await new Async.Future.delayed(new Duration(milliseconds: 1000));
if(pwd != "123456"){
throw ("登录失败,密码必须是123456");
}
print("登录成功!");
store.dispatch(new LoginSuccessAction(account: account));
},);
}, converter: (Store<AppState> store){
return store;
}),
···
在导入这个登录组件到应用程序路由上面的时候,这个时候将逻辑和Store进行对接,这样做,就将逻辑和ui彻底的分开了。
这里涉及到异步操作,那么就会遇到所谓“副作用”问题。
随着项目的增大,reducer也会越来越大,那么有什么办法可以管理呢?
这些问题我们改天再分享。
附件:
老规矩,代码在这里:
https://github.com/jzoom/flut...
如有疑问,请加qq群854192563讨论
flutter中使用redux之多界面互动的更多相关文章
- 理解 Flutter 中的 Key
概览 在 Flutter 中,大概大家都知道如何更新界面视图: 通过修改 Stata 去触发 Widget 重建,触发和更新的操作是 Flutter 框架做的. 但是有时即使修改了 State,Flu ...
- 在React中使用Redux
这是Webpack+React系列配置过程记录的第六篇.其他内容请参考: 第一篇:使用webpack.babel.react.antdesign配置单页面应用开发环境 第二篇:使用react-rout ...
- 在Flutter中嵌入Native组件的正确姿势是...
引言 在漫长的从Native向Flutter过渡的混合工程时期,要想平滑地过渡,在Flutter中使用Native中较为完善的控件会是一个很好的选择.本文希望向大家介绍AndroidView的使用方式 ...
- mvp 在 flutter 中的应用
在 Android 应用程序开发过程中,我们经常会用到一些所谓的架构方法,如:mvp,mvvm,clean等.之所以这些方法会被推崇是因为他们可以大大的解耦我们的代码的功能模块,让我们的代码在项目中后 ...
- Flutter 中文文档网站 flutter.cn 正式发布!
在通常的对 Flutter 介绍中,最耳熟能详的是下面四个特点: 精美 (Beautiful):充分的赋予和发挥设计师的创造力和想象力,让你真正掌控屏幕上的每一个像素. ** 极速 (Fast)**: ...
- Flutter中管理路由栈的方法和应用
原文地址:https://www.jianshu.com/p/5df089d360e4 本文首先讲的Flutter中的路由,然后主要讲下Flutter中栈管理的几种方法. 了解下Route和Navig ...
- 在Flutter中构建布局
这是在Flutter中构建布局的指南.首先,您将构建以下屏幕截图的布局.然后回过头, 本指南将解释Flutter的布局方法,并说明如何在屏幕上放置一个widget.在讨论如何水平和垂直放置widget ...
- flutter中的生命周期函数
前言:生命周期是一个组件加载到卸载的整个周期,熟悉生命周期可以让我们在合适的时机做该做的事情,flutter中的State生命周期和android以及React Native的生命周期类似. 先看一张 ...
- flutter 中的AppBar
在flutter中的很多页面中,都会有下面这段代码: 对应就是下图中的红色线框区域,被称作AppBar顶部导航. 项目准备 在使用AppBar之前,我们先新建一个tabBar的项目: 然后在pages ...
随机推荐
- Redis设计与实现笔记 - hash
基本结构如下 初始状态一直使用 dictht[0],即 0 号哈希表 在发生扩容 rehash的时候,开始渐进式向 dictht[1]哈希表转移, 转移完成后交换 dicth[0] 与 dictht[ ...
- ArrayList 迭代器学习笔记
我们先来看一段代码: List<String> list = new ArrayList<>(); list.add("str1"); list.add(& ...
- php人民币小写转大写函数,不限长度,精确到分
原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 .作者信息和本声明.否则将追究法律责任.http://ustb80.blog.51cto.com/6139482/1035327 在打印 ...
- 原生js,jquery通过ajax获得后台json数据动态新增页面元素
一.原生js通过ajax获取json数据 因为IE浏览器对ajax对象的创建和其他浏览器不同,为了兼容全部浏览器,我用下面的代码: function createXMLHttpRequest(){ t ...
- python工业互联网监控项目实战2—OPC
OPC(OLE for Process Control)定义:指为了给工业控制系统应用程序之间的通信建立一个接口标准,在工业控制设备与控制软件之间建立统一的数据存取规范.它给工业控制领域提供了一种标准 ...
- Linux网络架设篇,虚拟机l系统中网卡设备名与配置文件不符如何处理?
很多情况下,当我们在虚拟机中安装好linux系统后,并不能成功连上网.当我们配置好相关IP地址后同样不能成功连接网络.并且会体会网卡名与配置名不符,这时候应该怎么办呢? 1.清空下面文件 /etc/u ...
- Python爬虫利器 cURL你用过吗?
hello,小伙伴们,今天给大家分享的开源项目是一个python爬虫利器,感兴趣的小伙伴看完这篇文章不妨去尝试一下,这个开源项目就是curlconverter,不知道小伙伴们分析完整个网站后去code ...
- 超过百万的StackOverflow Flutter 问题
老孟导读:今天分享StackOverflow上高访问量的20大问题,这些问题给我一种特别熟悉的感觉,我想你一定或多或少的遇到过,有的问题在stackoverflow上有几十万的阅读量,说明很多人都遇到 ...
- Java课程设计之——爬虫篇
主要使用的技术 Httplcient Jsoup 多线程 dao模式 网络爬虫简介 网络爬虫(又称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页追逐者),是一种按照一定的规则,自动地抓取 ...
- [算法]Miller-Robbin素数判定
目录 一.实现原理 二.应用 判断一个正整数是否为素数 三.小结 一.实现原理 我们以前都是怎么判断素数的呢: 试除法: 若一个正整数N为合数,则存在一个能整除N的数k,其中\(2\leqslant ...