by Gerard Sans | @gerardsans
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
data:image/s3,"s3://crabby-images/ce4c3/ce4c3d1542b17c5516ae3dd192b779e1986379e6" alt=""
data:image/s3,"s3://crabby-images/ce4c3/ce4c3d1542b17c5516ae3dd192b779e1986379e6" alt=""
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
Google Developer Expert
data:image/s3,"s3://crabby-images/d5028/d50287fef9219023ed44c25e67269a8e8e414d17" alt=""
Google Developer Expert
International Speaker
data:image/s3,"s3://crabby-images/6d998/6d9980e0986cd753c822406512932befad00ee2d" alt=""
Spoken at 92 events in 25 countries
data:image/s3,"s3://crabby-images/96f08/96f08685e62fcae42a6dace37257b4ae7cf2fa53" alt=""
Blogger
Blogger
Community Leader
data:image/s3,"s3://crabby-images/0ccfa/0ccfa68c8b7e25e17aaf53bb8d1b5552e1592a92" alt=""
900
1.5K
Trainer
data:image/s3,"s3://crabby-images/9160a/9160ac729600f072b387ea94a74e157adf8123da" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/288f4/288f47d6ec76d747f0fab9df48586b0427885699" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/5512c/5512cae4175c37575f75c11d0f94a306c3834a46" alt=""
data:image/s3,"s3://crabby-images/c8616/c8616c8606b434bef34c7747cc27c297bfd89320" alt=""
data:image/s3,"s3://crabby-images/442bd/442bd41736b6958f25d3e5b7579a7086d7c77eb6" alt=""
data:image/s3,"s3://crabby-images/22ec5/22ec5f02fcc1b1abc1e06b132a33956773f761df" alt=""
data:image/s3,"s3://crabby-images/4f045/4f045fc01f6cc5789eced8a3ac05830528a87f55" alt=""
data:image/s3,"s3://crabby-images/9d0be/9d0be778c3aa4bee393a87565f9a9cabd382518c" alt=""
data:image/s3,"s3://crabby-images/d1e64/d1e64803f22111d0fc8375882a3ffc1c160b8461" alt=""
data:image/s3,"s3://crabby-images/ae584/ae584cb97523bb4a600288bd308c77065c3ce014" alt=""
data:image/s3,"s3://crabby-images/7d9c1/7d9c19069236561a65b8bd5e1de290eccc174351" alt=""
data:image/s3,"s3://crabby-images/e4ab0/e4ab0987e22bedb014bf9c2687a28dc98d22d421" alt=""
data:image/s3,"s3://crabby-images/85a99/85a99a413deddecc13d383432f27fc6b6c3aaff6" alt=""
data:image/s3,"s3://crabby-images/0b90e/0b90ec77fdd3e8478577dfee09e903ac8b50f310" alt=""
NGXS
Overview
-
Angular Developer Friendly
- Decorators
- Dependency Injection
- Life cycle hooks
- Minimal Setup
- Extensible via plugins
- Community driven
Components
Actions
State
Mutate
Dispatch
Notify
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
Observables
actions
states
A
A
A
S
S
S
Components
Actions
State
Mutate
Dispatch
Notify
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
Plugins
Actions
State
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
Counter
Increment
Decrement
Reset
Total
Components
Actions
State
Mutate
Dispatch
Notify
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
import { State } from '@ngxs/store';
@State<number>({
name: 'counter',
defaults: 0
})
export class CounterState { }
src/app.component.ts
Counter State
app/store/counter.state.ts
Components
Actions
State
Mutate
Dispatch
Notify
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
export class Increment {
static readonly type = "[Counter] Increment";
}
// const action = new Increment();
// {
// type: "[Counter] Increment"
// }
src/app.component.ts
Increment Action
app/store/counter.actions.ts
import { State, Action, StateContext } from '@ngxs/store';
import { Increment } from '../store/counter.actions';
@State<number>({ name: 'counter', defaults: 0 })
export class CounterState {
@Action(Increment)
Increment(store: StateContext<number>) {
const counter = store.getState();
store.setState(counter+1);
}
}
src/app.component.ts
Increment Action
app/store/counter.state.ts
export class Reset {
static readonly type = "[Counter] Reset";
constructor(public payload: { value: number }) { }
}
// const action = new Reset({ value: 0 });
// {
// type: "[Counter] Reset",
// payload: {
// value: 0
// }
// }
src/app.component.ts
Reset Action
app/store/counter.actions.ts
import { State, Action, StateContext } from '@ngxs/store';
import { Reset } from '../store/counter.actions';
@State<number>({ name: 'counter', defaults: 0 })
export class CounterState {
@Action(Reset)
Reset(store: StateContext<number>, action: Reset) {
const counter = store.getState();
store.setState(action.payload.value);
}
}
src/app.component.ts
Reset Action
app/store/counter.state.ts
Components
Actions
State
Mutate
Dispatch
Notify
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
@Component({
template: `
<app-counter
(increment)="increment()" (decrement)="decrement()"
(reset)="reset()">
</app-counter>`
})
export class AppComponent {
constructor(private store: Store) { }
increment = () => this.store.dispatch(new Increment());
decrement = () => this.store.dispatch(new Decrement());
reset = () => this.store.dispatch(new Reset({ value: 0 }));
}
src/app.component.ts
Dispatching Actions
app/app.component.ts
@Component({
template: `
<app-counter [total]="counter|async"></app-counter>`
})
export class AppComponent {
@Select(state => state.counter) counter: Observable<number>;
}
src/app.component.ts
Subscribing to the Store
app/app.component.ts
@Component({
selector: 'app-counter',
template: `
<div class="total">{{total}}</div>
<button (click)="increment.emit()">+</button>
<button (click)="decrement.emit()">-</button>
<button (click)="reset.emit()">C</button>`,
changeDetection: ChangeDetectionStrategy.OnPush
})
export class CounterComponent {
@Input() total: number;
@Output() increment = new EventEmitter();
...
}
src/app.component.ts
Counter
app/counter/counter.component.ts
Plugins
Logger
DevTools
Storage
Forms
WebSockets
Router
import { NgxsModule } from '@ngxs/store';
import { NgxsReduxDevtoolsPluginModule }
from '@ngxs/devtools-plugin';
@NgModule({
imports: [
NgxsModule.forRoot([]),
NgxsReduxDevtoolsPluginModule.forRoot()
]
})
export class AppModule {}
src/app.component.ts
Adding DevTools Plugin
import { NgxsModule } from '@ngxs/store';
import { NgxsFormPluginModule }
from '@ngxs/form-plugin';
@NgModule({
imports: [
NgxsModule.forRoot([]),
NgxsFormPluginModule.forRoot(),
NgxsReduxDevtoolsPluginModule.forRoot()
]
})
export class AppModule {}
src/app.component.ts
Adding Forms Plugin
@State({
name: "app",
defaults: {
form: {
model: undefined,
dirty: false,
status: "",
errors: {}
}
}
})
export class AppState {}
src/app.component.ts
Adding initial form state
@Component({
selector: 'form',
template: `
<form ngxsForm="app.form" novalidate
[formGroup]="form" (ngSubmit)="onSubmit()">
<input type="text" formControlName="email" />
<button type="submit">Subscribe</button>
</form>`
})
export class FormComponent { }
src/app.component.ts
Tracking form changes
import { NgxsModule } from '@ngxs/store';
import { NgxsRouterPluginModule }
from '@ngxs/router-plugin';
@NgModule({
imports: [
NgxsModule.forRoot([]),
NgxsRouterPluginModule.forRoot()
]
})
export class AppModule {}
src/app.component.ts
Adding Router Plugin
Advanced NGXS
Todo App
data:image/s3,"s3://crabby-images/062df/062df670a0ba0f6cc2bdb1cf922d6cc8a16d1fc5" alt=""
Selectors
{
todos: [{ id: 1, text: 'todo 1', complete: false }],
currentFilter: "SHOW_ALL"
}
// Filtered Todos
[{ id: 1, text: 'todo 1', complete: false }]
src/app.component.ts
Example: Todos State Selector
{
todos: [{ id: 1, text: 'todo 1', complete: false }],
currentFilter: "SHOW_COMPLETED"
}
// Filtered Todos
[]
import { FilterState } from '../filter/filter.state';
@State<Todo[]>({ name: 'todos', defaults: [] })
export class TodosState {
@Selector([FilterState]) static todos(state, currentFilter) {
let t = state.slice().reverse();
switch (currentFilter) {
case 'SHOW_ACTIVE': return t.filter(t => !t.completed);
case 'SHOW_COMPLETED': return t.filter(t => t.completed);
case 'SHOW_ALL':
default: return t;
};
}
}
src/app.component.ts
Selector: Filtered Todos
Comparison
- v3.1.3 (3 months)
- 53 contributors
- Plugins
- 1K stars
data:image/s3,"s3://crabby-images/1a86e/1a86e9d1b182b991a4989c13b543f35898f29306" alt=""
- v6.0.1 (2 years)
- 119 contributors
- CLI integration
- 3K stars
More
data:image/s3,"s3://crabby-images/23376/2337627cb059165184f857bad1eb7f2ee3db2779" alt=""
Austin McDaniel
ngxs.io
data:image/s3,"s3://crabby-images/27dff/27dffabaa639ef571d9ea9156f8bfb588dec2f08" alt=""
Medium
Thanks
data:image/s3,"s3://crabby-images/ce4c3/ce4c3d1542b17c5516ae3dd192b779e1986379e6" alt=""
data:image/s3,"s3://crabby-images/ce4c3/ce4c3d1542b17c5516ae3dd192b779e1986379e6" alt=""
NGXS: Redux implemented in 2018
By Gerard Sans
NGXS: Redux implemented in 2018
In this talk, we are going to look into NGXS, a new State Management library released earlier this year. We will introduce its main concepts and cover all the good stuff up to v3.1 release: state/actions life cycle, advanced selectors, snapshots, plugins and more!
- 3,573