Advanced NgRx
by Gerard Sans | @gerardsans
data:image/s3,"s3://crabby-images/db259/db2598c6a5a1b7a94f9e4a4cc3dc8c25365a3f43" alt=""
Advanced NgRx
data:image/s3,"s3://crabby-images/95696/956965f226c419788772450a326baa3104791b34" alt=""
data:image/s3,"s3://crabby-images/95696/956965f226c419788772450a326baa3104791b34" alt=""
data:image/s3,"s3://crabby-images/95696/956965f226c419788772450a326baa3104791b34" alt=""
data:image/s3,"s3://crabby-images/95696/956965f226c419788772450a326baa3104791b34" alt=""
SANS
GERARD
Google Developer Expert
data:image/s3,"s3://crabby-images/80224/8022497242bf3a8f489fbbc63d32c246fda77efd" alt=""
Google Developer Expert
International Speaker
data:image/s3,"s3://crabby-images/07d8b/07d8b538ae01b1fdef8e1a86d9d804eb05f3565e" alt=""
Spoken at 100 events in 26 countries
data:image/s3,"s3://crabby-images/afda3/afda3d2500371b526613cd2a8b4eb01f648569de" alt=""
Blogger
Blogger
Community Leader
data:image/s3,"s3://crabby-images/11288/11288cf1d4eb23adac45eef615dd8c0c254f4f6e" alt=""
900
1.5K
Trainer
data:image/s3,"s3://crabby-images/fcd71/fcd71771b824e4bb3facf6c3c6490dee12f20daf" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/daa81/daa81bf22d9387af1af898d6829c1627ca7a6022" alt=""
Master of Ceremonies
data:image/s3,"s3://crabby-images/b0ccc/b0ccc9a93f700dda33a746e2f56eee52fae8af2a" alt=""
data:image/s3,"s3://crabby-images/79dc5/79dc53a5b8989a5178023af3f8204205b17e612d" alt=""
data:image/s3,"s3://crabby-images/37f72/37f7213c6e21cdcff8d639c9dc346a4c3261f7f3" alt=""
data:image/s3,"s3://crabby-images/0a7fe/0a7fe65c159b23faa2bdd6c27717aa853cd4b721" alt=""
data:image/s3,"s3://crabby-images/a202a/a202adeaadf495470bada152c4b54b635075b123" alt=""
data:image/s3,"s3://crabby-images/623ab/623ab959dc6de9e1712f8f045ff80aeb6c8d8659" alt=""
data:image/s3,"s3://crabby-images/bb500/bb500e1d10d7e4bc89481b25ba00dd9d4325acf5" alt=""
data:image/s3,"s3://crabby-images/bbf51/bbf5193173f429707ceb9955bb9c709d293055dc" alt=""
data:image/s3,"s3://crabby-images/68977/68977daa3df608460d92f94d579d78a70bb500a2" alt=""
data:image/s3,"s3://crabby-images/0a421/0a421b3b47d00e8ebfc8c983a099ec72433c8517" alt=""
data:image/s3,"s3://crabby-images/a1d15/a1d15ca80fb18306a363c9dc32ddaf2be5041743" alt=""
data:image/s3,"s3://crabby-images/8cf05/8cf05996f9c6e8b201b30f2042ba32e64f43fa53" alt=""
data:image/s3,"s3://crabby-images/95c15/95c1573ce7984e63d6322283aeff3be961c62223" alt=""
FREE 3h Workshop
bit.ly/cfp-sampa
data:image/s3,"s3://crabby-images/75d38/75d38bc8dbd65d45de6b94ef183c36d138af44e5" alt=""
NGRX
- v6.0.1 (Dec 2015)
- 129 contributors
- CLI integration
- 3K stars
- v3.2 (Jan 2018)
- 69 contributors
- Plugins
- 1K stars
data:image/s3,"s3://crabby-images/09ddc/09ddc71e3c2b793ff26b66d6b6e2456af8a5d3a6" alt=""
- v4 (June 2015)
- 611 contributors
- Inspired by Flux
- 43K stars
Overview
- State Management for Angular
- Inspired by Redux
- Implemented using RxJS
- Angular CLI integration via schematics
@ngrx/store life cycle
source: blog
data:image/s3,"s3://crabby-images/d5426/d54266bfdf41414014b954943b3624d325ba0226" alt=""
1
2
3
4
5
actions
store
A
A
A
S
S
S
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Components
Actions
Reducer
Match
Dispatch
Creates
State
Notify
Packages
@ngrx/store-devtools
@ngrx/store
@ngrx/schematics
@ngrx/router-store
@ngrx/effects
@ngrx/entity
Features
Utilities
Counter
Increment
Decrement
Reset
Total
Components
Actions
Reducer
Match
Dispatch
Creates
State
Notify
import { Action } from '@ngrx/store';
export enum CounterActionTypes {
Increment = '[Counter] Increment',
}
export class CounterIncrement implements Action {
readonly type = CounterActionTypes.Increment;
}
export type CounterActions = CounterIncrement;
src/app.component.ts
Increment Action
app/actions/counter.actions.ts
> new CounterIncrement();
{
type: "[Counter] Increment"
}
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
export function reducer(state = 0, action: CounterActions): number
{
switch (action.type) {
case CounterActionTypes.Increment:
return state + 1;
default:
return state;
}
}
src/app.component.ts
Counter Reducer
app/reducers/counter.reducer.ts
> reducer(undefined, { type: "@ngrx/store/init" })
0
> reducer(0, new CounterIncrement())
1
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
export enum CounterActionTypes {
Reset = '[Counter] Reset'
}
export class CounterReset implements Action {
readonly type = CounterActionTypes.Reset;
constructor(public payload: { value: number }) { }
}
export type CounterActions = CounterIncrement | CounterReset;
src/app.component.ts
Reset Action
app/actions/counter.actions.ts
> new CounterReset({ value: 0 });
{
type: "[Counter] Reset",
payload: { value: 0 }
}
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
export function reducer(state = 0, action: CounterActions): number
{
switch (action.type) {
case CounterActionTypes.Reset:
return action.payload.value;
default:
return state;
}
}
src/app.component.ts
Counter Reducer
app/reducers/counter.reducer.ts
> reducer(42, new CounterReset({ value: 0 });
0
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
Components
Actions
Reducer
Match
Dispatch
Creates
State
Notify
@Component({
template: `
<app-counter
(increment)="increment()" (decrement)="decrement()"
(reset)="reset()">
</app-counter>`
})
export class AppComponent {
constructor(private store: Store) { }
increment = () => this.store.dispatch(new CounterIncrement());
decrement = () => this.store.dispatch(new CounterDecrement());
reset = () => this.store.dispatch(new CounterReset({ value: 0 }));
}
src/app.component.ts
Dispatching Actions
app/app.component.ts
We pass events up to container
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
@Component({
template: `
<app-counter [total]="counter|async"></app-counter>`
})
export class AppComponent {
counter: Observable<number>;
constructor(private store: Store<fromStore.State>) {
this.counter = this.store.select(state => state.counter);
}
}
src/app.component.ts
Subscribing to the Store
app/app.component.ts
Maps value
not Observable
We will improve this part later
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
@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
better performance
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Components
Actions
Reducer
Match
Dispatch
Creates
State
Notify
import * as fromCounter from './counter.reducer';
export interface State {
"counter": number;
}
export const reducers: ActionReducerMap<State> = {
"counter": fromCounter.reducer,
};
src/app.component.ts
Counter State
app/reducers/index.ts
import { StoreModule } from '@ngrx/store';
import { reducers } from './reducers';
@NgModule({
imports: [
BrowserModule,
StoreModule.forRoot(reducers),
],
})
export class AppModule { }
src/app.component.ts
Store Setup
app/app.module.ts
Reusing reducers
data:image/s3,"s3://crabby-images/6203e/6203e8af410ebb818358a048f535230e6166a6d6" alt=""
data:image/s3,"s3://crabby-images/2ec76/2ec76a4cc2bdf610fa8c4054d4f0640c42066831" alt=""
{
counter: 7
}
Single Counter
{
type: "[Counter] Increment"
}
State
Action
User clicks
{
counter: 8
}
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Multiple Counters
{
"counter-1": 7,
"counter-2": 3,
}
{
type: "[Counter] Decrement",
payload: { target: "counter-2" }
}
counter-1
counter-2
State
Action
User clicks
{
"counter-1": 7,
"counter-2": 2,
}
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
import * as fromCounter from './counter.reducer';
import { namedReducer } from './named.reducer';
export interface State {
"counter-1" : number;
"counter-2" : number;
}
export const reducers: ActionReducerMap<State> = {
"counter-1": namedReducer(fromCounter.reducer, "counter-1"),
"counter-2": namedReducer(fromCounter.reducer, "counter-2")
};
src/app.component.ts
Counter State
app/reducers/index.ts
export function namedReducer(reducer: any, target: string) {
return (state: number, action: CounterActions) => {
// ignore action and return current state
if (action.payload && action.payload.target != target)
return state;
// otherwise use original reducer
return reducer(state, action);
}
}
src/app.component.ts
Meta-reducer
app/reducers/named.reducer.ts
namedReducer(fromCounter.reducer, "counter-2")
{
type: "[Counter] Decrement"
payload: { target: "counter-2" }
}
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
Selectors
data:image/s3,"s3://crabby-images/20b14/20b14f33ec67844851cb9b2f9631b5439fdc0a65" alt=""
State
Property Selectors
Computed Selectors
s
s
s
Property Selectors
-
Helpers to access Store​
-
​Avoid Components tight coupling with Store
-
Colocated with Reducers
store.select(state => state.total);
store.select(state => state.pagination.page);
Tight coupling
Components
State
Selectors: loose coupling
store.select(getCounter());
store.select(getPage());
Components
Selectors
State
A small change won't break all Components now
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
// counter
export const getCounter =
(state: State): number => state.counter;
// todos
export const getTodos =
(state: State): Array<Todo> => state.todos;
export const getFilter =
(state: State): string => state.currentFilter;
src/app.component.ts
Property Selectors
app/reducers/index.ts
State
Property Selectors
Computed Selectors
s
s
s
Computed Selectors
-
Compute values using other Selectors
-
​​createSelector, createFeatureSelector
-
-
Memoised for performance
-
Colocated with Reducers
// State
{
todos: [{ id: 1, text: 'Learn ngrx', complete: false }],
currentFilter: "SHOW_ALL"
}
// Visible Todos
[{ id: 1, text: 'Learn ngrx', complete: false }]
src/app.component.ts
Example: Visible Todos Selector
// State
{
todos: [{ id: 1, text: 'Learn ngrx', complete: false }],
currentFilter: "SHOW_COMPLETED"
}
// Visible Todos
[]
createSelector
export const getTodos = state => state.todos;
export const getFilter = state => state.currentFilter;
export const getVisibleTodos = createSelector(
getTodos,
getFilter,
projector
);
export const getTodos = state => state.todos;
export const getFilter = state => state.currentFilter;
export const getVisibleTodos = createSelector(
getTodos,
getFilter,
(todos, filter) => visibleTodos(todos, filter)
);
export const getTodos = state => state.todos;
export const getFilter = state => state.currentFilter;
export const getVisibleTodos = createSelector(
getTodos,
getFilter,
visibleTodos
);
The projector function is just the calculation we want to run
Refactored!
todos is getTodos(state)
filter is getFilter(state)
export const getTodos = state => state.todos;
export const getFilter = state => state.currentFilter;
Using createSelector
app/reducers/index.ts
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
src/app.component.ts
Selector: Visible Todos
function visibleTodos(todos: Array<Todo>, filter: string) {
switch (filter) {
case 'SHOW_ACTIVE':
return todos.filter(t => !t.completed);
case 'SHOW_COMPLETED':
return todos.filter(t => t.completed);
case 'SHOW_ALL':
default:
return todos;
}
};
import { getVisibleTodos } from './reducers'
@Component({
template: `<todo *ngFor="let todo of todos | async"></todo>`
})
export class TodoComponent {
todos: Observable<Array<Todo>>;
constructor(private store: Store<fromStore.State>) {
this.todos = this.store.select(getVisibleTodos);
}
}
src/app.component.ts
Using getVisibleTodos
app/todos.component.ts
createFeatureSelector
export const getBooksState =
createFeatureSelector<BooksState>('books');
// {
// search: ...
// books: ...
// collection: ...
// }
This is always from the Root
Using createFeatureSelector
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
import { StoreModule } from '@ngrx/store';
import { reducers } from './reducers';
@NgModule({
imports: [
CommonModule,
StoreModule.forFeature('books', reducers),
],
})
export class BooksModule { }
src/app.component.ts
Store Setup
app/books/books.module.ts
Books Module can be lazy loaded
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
Memoisation
data:image/s3,"s3://crabby-images/7f135/7f135d85b3c0ebf99b8d30afc2f58a1fed2ee087" alt=""
Memoisation
-
Created to improve performance of Machine Learning algorithms
-
Targets repetitive calls trading space for speed
-
Uses a Cache to store Results
-
1-slot and multi-slot
src/app.component.ts
1-slot Memoisation
function memoise(fn) {
let cache, $args;
return function() {
if (sameArgs($args, arguments)){
console.log('cached');
return cache;
} else {
console.log('calculating...');
cache = fn(...arguments);
$args = arguments;
return cache;
}
}
};
const add = (a,b) => a+b;
> $add(1,1)
calculating...
2
first execution
we keep results and arguments
> const $add = memoise(add);
> $add(1,1)
cached
2
next execution
with same arguments
we use cache
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
src/app.component.ts
1-slot Memoisation Limitations
> $add(1,1)
calculating...
2
> $add(2,2)
calculating...
4
> $add(1,1)
calculating...
2
Unless we do consecutive repetitive calls (recursive) is not very effective
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
😱
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
src/app.component.ts
Multi-slot Memoisation
function memoise(fn) {
let cache = {};
return function() {
const key = arguments.join('-');
if (key in cache) {
return cache[key];
}
else {
cache[key] = fn(...arguments);
return cache[key];
}
}
};
const add = (a,b) => a+b;
we use the key 1-1 to cache results
> $add(1,1)
calculating...
2
> $add(2,2)
calculating...
4
> $add(1,1)
cached
2
we create a unique key using the arguments
Use JSON.stringify for a generic approach
cache
{"1-1": 2, "2-2": 4}
this time
we hit the cache
data:image/s3,"s3://crabby-images/e27cc/e27cca46e7036b472dc67e3d1a0e81a95a564d9c" alt=""
😃
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
@ngrx/store Selectors
-
Uses 1-slot Memoisation
-
​createSelector, createFeatureSelector
-
- Applied to projector​ function
-
Allows replacing default memoisation
- reselect (1-slot)
- moize (multi-slot)
src/app.component.ts
1-slot Memoisation: getVisibleTodos
> const state = {
todos: [{ id: 1, text: 'Learn ngrx', complete: false }],
currentFilter: "SHOW_ALL"
};
[{ id: 1, text: 'Learn ngrx', complete: false }]
> getVisibleTodos(state);
filter function executed
[]
> state = { ...state, currentFilter: "SHOW_COMPLETED" }
> getVisibleTodos(state);
> state = { ...state, currentFilter: "SHOW_ALL" }
> getVisibleTodos(state);
[{ id: 1, text: 'Learn ngrx', complete: false }]
executed again
and again...
😱
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
import { createSelectorFactory } from '@ngrx/store';
import moize from "moize";
const customMemoizer = fn => {
const memoized = moize.deep(fn, {maxSize: 5, maxAge: 60000});
return { memoized, reset: () => memoized.clear() }
};
const $createSelector = createSelectorFactory(customMemoizer);
export const getFilteredTodos = $createSelector(
selectTodosEntities,
selectCurrentFilter,
getVisibleTodos
);
src/app.component.ts
Multi-slot getVisibleTodos
We can use deepEqual for arguments
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
reset will clear
the cache
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
src/app.component.ts
Multi-slot Memoisation: getVisibleTodos
> const state = {
todos: [{ id: 1, text: 'Learn ngrx', complete: false }],
currentFilter: "SHOW_ALL"
};
[{ id: 1, text: 'Learn ngrx', complete: false }]
> getVisibleTodos(state);
filter function executed
[]
> state = { ...state, currentFilter: "SHOW_COMPLETED" }
> getVisibleTodos(state);
> state = { ...state, currentFilter: "SHOW_ALL" }
> getVisibleTodos(state);
[{ id: 1, text: 'Learn ngrx', complete: false }]
executed again
Cached!
😃
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
data:image/s3,"s3://crabby-images/e0b03/e0b0303f378126c963d381fe8feb38e13ced1189" alt=""
More
data:image/s3,"s3://crabby-images/79494/794949bdcee47f0605075f992640a1b4c2f1b80b" alt=""
@MikeRyanDev
@robwormald
@brandontroberts
data:image/s3,"s3://crabby-images/e6caf/e6caf26b6622ef4095e30313a3de0918f61529ae" alt=""
data:image/s3,"s3://crabby-images/14a4a/14a4a60a9972566ad3872016f019670513e9cf4c" alt=""
Rob Wormald
Mike Ryan
Brandon Roberts
@toddmotto
data:image/s3,"s3://crabby-images/601b8/601b8495faca2f3c061748a3041b5b7f48873267" alt=""
Todd Motto
data:image/s3,"s3://crabby-images/cff38/cff38e170b3f33c62c15004459fa702a8ed21fda" alt=""
data:image/s3,"s3://crabby-images/da497/da49770bf85576c54d861cca9ee0f88c319278d5" alt=""
data:image/s3,"s3://crabby-images/da497/da49770bf85576c54d861cca9ee0f88c319278d5" alt=""
Advanced NgRx
By Gerard Sans
Advanced NgRx
State Management is key to build modern Web Apps
- 6,507