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 ...
随机推荐
- Jmeter 压力测试笔记(2)--问题定位
事情已经出了,是该想办法解决的时候了. 经过运维和DBA定位: 数据库读写分离中,读库延时超过了30秒,导致所有请求都压在主库.另外所有数据库都连接数都被占满,但活跃请求数量缺不多. 数据库16K的连 ...
- 面试刷题37:微服务是什么?springcloud,springboot是什么?
面试中被问到为什么要使用微服务架构?springcloud的核心组件有哪些? 拿我们国家的兵种来说,如何把战争这个单体架构微服务化,就是根据适用的场景,拆分出不同的兵种(微服务) 然后每个兵种之间通过 ...
- Nginx-高性能的反向代理服务器
Nginx Nginx作为一款反向代理服务器,现在大多数网站都有使用,自己在项目中几乎都有用到,自己的网站也使用到了它. 了解Nginx 上面图可以直观的看出Nginx的用处,可以将请求转发至Web服 ...
- Array(数组)对象-->shift() 方法
1.定义和用法 shift() 方法用于把数组的第一个元素从其中删除,并返回第一个元素的值. 语法: array.shift() 注意: 此方法改变数组的长度! 举例: var arr = [1,2, ...
- springIoc中的单列对象的分析
最近有个同事去面试,其中有一个问题是关于spring单例的.本篇博文就发表一下小编我自己的理解~~. 使用过spring的程序猿应该都知道,我们的bean(controller.service和Dao ...
- java web之Filter详解
java web之Filter详解 2012-10-20 0 个评论 作者:chenshufei2 收藏 我要投稿 .概念: Filter也称之为过滤器,它是Servlet技术中比较激动人心的技术,W ...
- Property [*****] not found on type [com.erp.pojo.ErpSupplier]
我实体类里用的是 springboot 里@Slf4j @Data 注解式写的 这样可以减少代码量 ,但是遇到一个问题影响我好长时间 出现了这个错误 Property [*****] not ...
- [总结]最小生成树之Kruskal算法
目录 一.最小生成树的相关知识 1. 树的性质 2. 生成树 3. 最小生成树 4. 最小生成树的性质 二.Kruskal算法求最小生成树 1. 核心思想 2. 具体流程 3. 图示 4. 代码实施 ...
- ios 中使用 animation-play-state: paused 属性失效的问题
前言 因为要做一个播放器的播放图片旋转动画,像这样子 当音乐播放就转动,停止就暂停. 开始于是很自然地想到了使用Css3的 animation 动画属性CSS3 animation(动画) 属性 an ...
- L3 多层感知机
**本小节用到的数据下载 1.涉及语句 import d2lzh1981 as d2l 数据1 : d2lzh1981 链接:https://pan.baidu.com/s/1LyaZ84Q4M75G ...