Skip to content

Latest commit

 

History

History
199 lines (159 loc) · 8.06 KB

README.md

File metadata and controls

199 lines (159 loc) · 8.06 KB

Omnistate

  • Single, centralized state container with setters and getters backed by JSON
  • High performance with large state objects and high state change volume
  • Views use path strings to declare what state they need (using lodash get and set)
  • State is delivered to views via proxy mixins (read replica + setter) e.g. get = this.foo, set = this.setFoo(value) or updateFoo(reducerFn) or mergeFoo(keysAndValuesObject)
  • Views render independently of parents (but this does not preclude top-down)
  • State change pattern-matching drives "operations" (simple controllers outside of views)
  • In development/debug mode direct mutation of read-replica state is an error
  • Full history API to snapshot, record and replay all state changes. e.g. time traveling, infinite undo/redo

screen shot 2015-12-20 at 12 20 35 pm

screen shot 2015-12-20 at 12 22 51 pm

Still in Beta

This is not on npm yet as it is still in heavy development but if you want to try it out you can run the example or use it by requiring from your file system.

Run the Example

from the examples/todomvc or examples/basic folder...

npm install
npm start
open http://localhost:3000

Motivation

OmniState has been developed over the past couple years while working on a couple fairly complex applications. First an Angular 1.x App and another in React. In that time I have tried to make a system that avoids complexity, has a few simple ideas and performs well. It takes a unique approach to change observation, controllers, accessing state within views, how view renders are triggered. Much effort has also been put into making sure the system performs well with large state objects and a high volume of state changes. Omnistate is currently benchmarked at 180 random cell updates per second on a 100x100 table (In safari, see example).

Basic JSON objects are used instead of immutable data-structures and state is easily accessible and updateable. Change observation is performed by routing all changes through setters and diffing the new and old sate. Paths are used heavily leaveraging lodash set and get semantics. These paths help make the diffing more efficient by diffing only from the branch of state that has changed. These paths are also used to declare what state a view needs which leads to three very nice things. First the state required is fetched for you from the read-replica and made easily available as a mixin. Second a setter as it is likely you will want to change that state. Finally any time there are any state changes at or under that path the view will render but none of its parents will.

With OmniState you can make a complex app without this.props, this.state or defining a shouldComponentUpdate. Obviously this is not your standard approach to react but you are not prevented from using those things. Although it is different I have found this approach to be very nice to work with. It is similar in some respects to OM Next.

Component (declarative state access)

A custom component constructor is provided which makes accessing state easy. It also automatically renders the component if any of it's state changes, bypassing the traditional top-down rendering path. This is optional but can be a powerful tool for performance and scalability. In worst-case scenarios rendering performance is similar to raw js. See example code.

module.exports = omnistate.component("UserName", {
	proxies: {
		name: "user.name"	
	}
}, function() {
	return (<div className="userName">{this.name}</div>)
});

Our simple UserName component will now reflect any changes to user.name app state. How about a list of users? The parent can simply redefine the path via props.

module.exports = omnistate.component("Users", {
	proxies: {
		name: "users"	
	},
	userList: function(users) {
		return _.each(users, function(user, i) {
			return <li><UserName proxies={{name: "users."+i}}
		});
	}
	
}, function() {
	return <ul>{this.userList(this.users)}</ul>
});

Now if users[10].name is changed only that component will render. The ShouldComponentUpdate functions will not even be triggered for the parents or any siblings. This can be a big win when there are many siblings or complex parents to render. This of course is an optional feature that can be disabled by declaring the state as an accessor only (no direct rendering on change) Simply prefix the proxy path with @

module.exports = omnistate.component("UserName", {
	proxies: {
		name: "@user.name"
	}
}, function() {
	return (<div className="userName">{this.name}</div>)
});

The view will not force a direct render when the user.name changes. In practice there are few edge cases for needing this capability. See the simpl example code Alpha.jsx for such a case.

Controllers and Computed Values

Views react to changes in state. But how does state itself react to changes in state?

Controllers are much like views, they can be triggerd by binding to specific state changes declared in a proxies property. Both views and controllers can respond to all state changes by adding an onStateChange function. The differnce of course is that a controller is concerned only with updating app state or other non-rendering operations such as saving state changes to local storage as in the provided TodoMVC example.

import {debounce} from 'lodash'
import omni from 'omnistate'

var history = omni.state.history;

export default omni.controller("saveState", {

	// load saved app state
	load: function() {
		history.loadLatestCheckpoint();
	},

	save: function() {
		history.clearCheckpoints();
		history.checkpoint();
	},

	// triggered on ANY state change so we debounce
	onStateChange: debounce(function() {
		console.log("state saved", omni.state.replica);
		this.save();
	}, 500, {maxWait: 5000})
});

Computed values are built on top of controllers and provide a handy syntax for defining computed values. Computed values are one way so treat them as read only. Here is another example from the TodoMVC example. It takes the source of truth, 'todosById' and creates three different collections. Only the ids are stored in these secondary collections, this normalized approach has several benefits and OmniState works best in this manner. For larger projects consider using something like Normalizr to help with this approach, however with computed values it is trivial to setup.

computed({
	allTodoIds: [
		['todosById'],
		todosById => _(todosById).sortBy('id').pluck('id').value()
	],

	activeTodoIds: [
		['todosById'],
		todosById => _(todosById).filter({completed: false}).sortBy('id').pluck('id').value()
	],

	completedTodoIds: [
		['todosById'],
		todosById => _(todosById).filter({completed: true}).sortBy('id').pluck('id').value()
	]
});

But is it immutable?

This work is largely inspired by practical experience developing with Immutable.js backed application state. One of the annoyances there was the api boilerplate and constant serialization and deserialization of application state. Omnistate provides views and controllers with a read-replica of the app state. Copy-on-write process is used as all state changes go through setters that update the source of truth and the read replica.

The read replica is a plain JSON object, mutations to it will not trigger a state change! So don't assign values to app state read-replicas only use the OmniState api methods for updates. In fact if you initialize omnistate with the dev option true it will throw an error if at any time you mutate the read replica rather than useing the state setters interface.

More to Cover...

Lots more to do and cover but for now look at the TodoMVC example to learn more.

  • React Router integration
  • history api
  • OmniState Tools run the example to try it out!
  • not limited to React Router or even React for that matter
  • more on normalizr and falcor integration