diff --git a/doc/article/en-US/aurelia-store-plugin.md b/doc/article/en-US/aurelia-store-plugin.md index 940faaa..cc46a36 100644 --- a/doc/article/en-US/aurelia-store-plugin.md +++ b/doc/article/en-US/aurelia-store-plugin.md @@ -1093,6 +1093,106 @@ The result will be that a new state is emitted and your subscriptions will recei > Keep in mind that this feature should be used with caution as it might introduce unintended side effects, especially in collaboration with the Redux DevTools, as the reset states are not tracked by it. +## Two-way binding and state management +Two-way binding definitely has its merits when it comes to easily develop applications. By having Aurelia take care of synchronizing of the View and ViewModel a lot of work gets done for free. +This behavior though becomes troublesome when working with a central state management. Remember, the primary vision is to do state modifications only via dispatching actions. This way we guarantee a single source of truth and no side-effects. +Yet exactly the later is what a two-way binding introduces if you bind your VM directly to state objects. So instead of two-way binding to state objects you should: + +* Create a new property, reflecting the state part you'd like to modify and bind to that. +* When changes are made, persist the changes by dispatching an action which uses the binding models values +* After dispatching make sure to update the binding model to match any updates to the object happening during dispatch + + + + + /* + * state: { + * firstName: string, + * lastName: string + * } + */ + + // app.ts + ... + + function updateUser(state: State, model: State) { + return Object.assign({}, state, model); + } + + @autoinject() + @inlineView(` + + `) + export class App { + public state: State; + public model: State; + + constructor(private store: Store) { + this.store.registerAction("updateUser", updateUser); + } + + bind() { + this.store.state.subscribe((newState) => { + this.state = newState; + this.model = Object.assign({}, newState); // take care of shallow cloning + }); + } + + updateUser() { + this.store.dispatch(updateUser, this.model); + } + } + + + + /* + * state: { + * firstName: string, + * lastName: string + * } + */ + + // app.js + ... + + function updateUser(state, model) { + return Object.assign({}, state, model); + } + + @autoinject() + @inlineView(` + + `) + export class App { + constructor(private store: Store) { + this.store.registerAction("updateUser", updateUser); + } + + bind() { + this.store.state.subscribe((newState) => { + this.state = newState; + this.model = Object.assign({}, newState); // take care of shallow cloning + }); + } + + updateUser() { + this.store.dispatch(updateUser, this.model); + } + } + + + +Alternatively you can resort to only using `one-time`, `to-view`, `one-way` and handle the updates yourself. + + ## Recording a navigable history of the stream of states Since the whole concept of this plugin is to stream states over time, it makes sense to also keep track of the historical changes. Aurelia Store supports this feature by turning on the history support during the plugin registration. @@ -1693,9 +1793,6 @@ If you've ever worked with Redux then you know for sure about the [Redux Devtool There are tons of [great articles](https://codeburst.io/redux-devtools-for-dummies-74566c597d7) to get you started. Head over to [DevTools browser extension page](https://github.com/zalmoxisus/redux-devtools-extension) for instructions on how to install the extension, start your Aurelia Store plugin project and see how it works. -> Info -> Because of the way Aurelia Binding works with `bind`, `one-way`, `two-way` or `to-view`, your state could be manipulated outside of the knowledge of the store, showing an incorrect current state in Redux DevTools. If your state object for a view model does not initialize every property (e.g. your initial state for a view model is an empty object such as `currentState = {}`), and you are binding properties in your view using one of the binding commands mentioned previously (such as `