ngrx/store effects 使用总结2:列表展示
第一个计数器案例:http://www.cnblogs.com/axel10/p/8589122.html
完成了计数器案例后,现在开始比较能够完整的展示angular2+开发流程的案例:在线获取用户列表并展示,包含点击删除功能。希望在开始之前,你对typescript和rxjs有一定了解。
再重复一遍开发流程:
开始 -> 编写数据模型 -> 编写action -> 编写redurces并配置到相应module -> 编写services -> 编写effects并配置到相应module -> 创建组件 -> 组件绑定数据模型 -> 渲染
第一步:编写数据模型(app/models/user.ts)
export class User { id: number; name: string; }
第二步:编写action(app/actions/num.ts)
import {Action} from '@ngrx/store';
import {User} from '../models/user'; export enum ListActionType {
Load = 'LOAD',
LoadSuccess = 'LOAD_SUCCESS',
RemoveUser = 'REMOVE_USER',
RemoveUserSuccess = 'REMOVE_USER_SUCCESS',
RemoveUserError = 'REMOVE_USER_ERROR'
} export class Load implements Action {
readonly type = ListActionType.Load;
} export class LoadSuccess implements Action {
readonly type = ListActionType.LoadSuccess; constructor(public payload: User[]) {
}
} export class RemoveUser implements Action {
readonly type = ListActionType.RemoveUser; constructor(public payload: number) {
}
} export class RemoveUserSuccess implements Action {
readonly type = ListActionType.RemoveUserSuccess;
} export class RemoveUserError implements Action {
readonly type = ListActionType.RemoveUserError;
}
第三步:编写redurcers(app/redurces/list.ts)
import {Action} from '@ngrx/store'; import {User} from '../models/user'; import {ListActionType, LoadSuccess, RemoveUser} from '../actions/list'; export interface State { loading: boolean; loaded: boolean; list: User[]; } const initialState: State = { loading: false, loaded: false, list: [] }; export const list = (state = initialState, action: Action) => { switch (action.type) { case ListActionType.Load: console.log('load....'); return { ...state, loading: true, }; case ListActionType.LoadSuccess: console.log('load success'); const myA = (<LoadSuccess>action).payload; console.log(myA); return { ...state, loaded: true, loading: false, list: myA }; case ListActionType.RemoveUser: console.log('remove user'); const userId = (<RemoveUser>action).payload; state.list = state.list.filter(function (item) { return item.id !== userId; }); return {...state}; default: return state; } };
配置redurcer(app/app.module.ts)
imports: [ BrowserModule, RouterModule.forRoot(routes), StoreModule.forRoot({ modelNum, list}), //配置redurcer ],
第四步:编写services(app/services/ListService.ts)
import {Injectable} from '@angular/core'; import {HttpClient} from '@angular/common/http'; import {Observable} from 'rxjs/Observable'; import {User} from '../models/user'; @Injectable() export class ListService { public getList(): Observable<any> { return this.http.get<{ users: User[] }>('/api/users.json'); } public removeUser(): Observable<any> { return Observable.create(function (observer) { observer.next('true'); }); } constructor(private http: HttpClient) { } }
第五步:编写effects:
import {Injectable} from '@angular/core'; import {Actions, Effect, ofType} from '@ngrx/effects'; import {Observable} from 'rxjs/Observable'; import {Action} from '@ngrx/store'; import {ListActionType, Load, LoadSuccess, RemoveUser, RemoveUserError, RemoveUserSuccess} from '../actions/list'; import {catchError, map, switchMap} from 'rxjs/operators'; import {ListService} from '../services/ListService'; import {of} from 'rxjs/observable/of'; @Injectable() export class ListEffects { @Effect() loadList$: Observable<Action> = this.action$.pipe( //rxjs写法。loadList$是effect名,在外部没有用到,可以随便起。 ofType<Load>(ListActionType.Load), switchMap(action => { return this.listService.getList().pipe(map( users => { return new LoadSuccess(users); } )); }) ); @Effect() removeUser$: Observable<Action> = this.action$.pipe( ofType<RemoveUser>(ListActionType.RemoveUser), switchMap(_ => { return this.listService.removeUser().pipe( map(res => { console.log(res); if (res === 'true') { return new RemoveUserSuccess(); } else { return new RemoveUserError(); } }), catchError(err => of(new RemoveUserError())) ); }) ); constructor(private action$: Actions, private listService: ListService) { } }
记得在app.module.ts配置effects和HttpClient:
imports: [ BrowserModule, RouterModule.forRoot(routes), StoreModule.forRoot({modelNum, list}), EffectsModule.forRoot([ListEffects]), HttpClientModule ],
第六步:创建组件
ng g component list
第七步:组件绑定数据模型(连带完成第八步)
组件ts文件:
import {Component, OnInit} from '@angular/core'; import {Store} from '@ngrx/store'; import * as list from '../actions/list'; import {State} from '../reducers/list'; import {User} from '../models/user'; @Component({ selector: 'app-list', templateUrl: './list.component.html', styleUrls: ['./list.component.scss'] }) export class ListComponent implements OnInit { public list: User[]; constructor(private store: Store<any>) { this.store.select('list').subscribe(_list => { if (_list) { console.log(_list); console.log(_list.list); this.list = _list.list; } }); } removeUser(id) { console.log(id); this.store.dispatch(new list.RemoveUser(id)); } ngOnInit() { this.store.dispatch(new list.Load()); } }
组件html文件:
<div> list 请尝试点击上半部分的li。 <ul> <li *ngFor="let item of list" (click)="removeUser(item.id)">{{item.name}}</li> </ul> <app-list-body></app-list-body> <br/> <a routerLink="/model">to counter demo</a> </div>
最后配置路由:
import {Routes} from '@angular/router'; import {IndexComponent} from './index/index.component'; import {ModelDemoComponent} from './model-demo/model-demo.component'; import {ListComponent} from './list/list.component'; export const routes: Routes = [ { path: 'list', component: ListComponent }, { path: 'model', component: ModelDemoComponent }, { path: '**', redirectTo: 'list' } ];
到此本案例结束。如果想要更加复杂的案例可以到ngrx的github获取。https://github.com/ngrx/platform
ngrx/store effects 使用总结2:列表展示的更多相关文章
- ngrx/store effects 使用总结1:计数器
本教程案例github:https://github.com/axel10/ngrx_demo-counter-and-list angular2+ 的学习成本应该是三大框架中最高的一个,教程及案例稀 ...
- [Angular2] @Ngrx/store and @Ngrx/effects learning note
Just sharing the learning experience related to @ngrx/store and @ngrx/effects. In my personal opinio ...
- Angular应用架构设计-3:Ngrx Store
这是有关Angular应用架构设计系列文章中的一篇,在这个系列当中,我会结合这近两年中对Angular.Ionic.甚至Vuejs等框架的使用经验,总结在应用设计和开发过程中遇到的问题.和总结的经验, ...
- Vuex 教程案例:计数器以及列表展示
本案例github:https://github.com/axel10/Vuex_demo-Counter-and-list 本篇教程将以计数器及列表展示两个例子来讲解Vuex的简单用法. 从安装到启 ...
- Winform开发主界面菜单的动态树形列表展示
我在之前很多文章里面,介绍过Winform主界面的开发,基本上都是标准的界面,在顶部放置工具栏,中间区域则放置多文档的内容,但是在顶部菜单比较多的时候,就需要把菜单分为几级处理,如可以在顶部菜单放置一 ...
- [Angular 2] ngrx/store
@ngrx/store builds on the concepts made popular by Redux and supercharges it with the backing of RxJ ...
- [Angular 2] Using ngrx/store and Reducers for Angular 2 Application State
ngrx/store is a library that simplifies common RxJS patterns for managing state and gives you an eas ...
- JSP中列表展示,隔行变色以及S标签的使用
1.java代码 /** * 列表展示,隔行变色以及S标签的使用 * * @return */ public String list() { List<User> list = new A ...
- NgRx/Store 4 + Angular 5使用教程
这篇文章将会示范如何使用NgRx/Store 4和Angular5.@ngrx/store是基于RxJS的状态管理库,其灵感来源于Redux.在NgRx中,状态是由一个包含action和reducer ...
随机推荐
- Spark MLlib(下)--机器学习库SparkMLlib实战
1.MLlib实例 1.1 聚类实例 1.1.1 算法说明 聚类(Cluster analysis)有时也被翻译为簇类,其核心任务是:将一组目标object划分为若干个簇,每个簇之间的object尽可 ...
- Mysql密码加密方式
以Mysql 4.1版本为分界线,两种加密方式 Mysql323加密:(16位) select old_password('root'); //Mysql自带加密函数old_password(str ...
- sql优化系列3(收集来源http://bbs.csdn.net/topics/250004467)
如何加快查询速度? 1.升级硬件 2.根据查询条件,建立索引,优化索引.优化访问方式,限制结果集的数据量. 3.扩大服务器的内存 4.增加服务器CPU个数 5.对于大的数据库不要设置数据库自动增长 ...
- 2 Model层 - 模型查询
1.简介 查询集表示从数据库中获取的对象集合 查询集可以含有零个.一个或多个过滤器 过滤器基于所给的参数限制查询的结果 从Sql的角度,查询集和select语句等价,过滤器像where和limit子句 ...
- Java web学习总结
javaweb学习总结(十四)——JSP原理 孤傲苍狼 2014-07-24 09:38 阅读:46603 评论:37 JavaWeb学习总结(十三)——使用Session防止表单重复提交 孤 ...
- IOS开发学习笔记034-UIScrollView-xib实现分页
通过xib实现分页功能的封装 1.首先实现xib UIView 的尺寸为300*130,因为准备的图片为600*260. scrollView属性设置如下: 2.新建一个和xib同名的类 2.1 类方 ...
- Mac教程macOS教程 苹果电脑教程
第1 章 初识MacOS 01 菜单栏 02 键盘 03 聚焦(Spotlight)
- File IO(NIO.2):什么是路径?
简介 文件系统以某种形式的媒体(通常为一个或多个硬盘驱动器)存储和组织文件,使得它们可以容易地被检索.目前使用的大多数文件系统将文件存储在树形(或分层)结构中.在树的顶部是一个(或多个)根节点.在根节 ...
- CodeForces839B[思维] Codeforces Round #428 (Div. 2)
#include <bits/stdc++.h> using namespace std; int n, k; ; ], cnt[]; void solve() { int t; cnt[ ...
- event对象的兼容性
最近在调试项目的时候,发现IE和Chrome都显示正常,就是FireFox异常,F12查看控制台,发现报错:window.event is undefined.检查代码中定义的事件方法中获取事件对象直 ...