var extend = require('raptor-util/extend'); function ensure(state, propertyName) { var proto = state.constructor.prototype; if (!(propertyName in proto)) { Object.defineProperty(proto, propertyName, { get: function() { return this.$__raw[propertyName]; }, set: function(value) { this.$__set(propertyName, value, false /* ensure:false */); } }); } } function State(component, initialState) { this.$__component = component; this.$__raw = initialState || {}; this.$__dirty = false; this.$__old = null; this.$__changes = null; this.$__forced = null; // An object that we use to keep tracking of state properties that were forced to be dirty if (initialState) { for(var key in initialState) { ensure(this, key); } } Object.seal(this); } State.prototype = { $__reset: function() { var self = this; self.$__dirty = false; self.$__old = null; self.$__changes = null; self.$__forced = null; }, $__replace: function(newState) { var state = this; var key; var rawState = this.$__raw; for (key in rawState) { if (!(key in newState)) { state.$__set(key, undefined, false /* ensure:false */, false /* forceDirty:false */); } } for (key in newState) { state.$__set(key, newState[key], true /* ensure:true */, false /* forceDirty:false */); } }, $__set: function(name, value, shouldEnsure, forceDirty) { var rawState = this.$__raw; if (shouldEnsure) { ensure(this, name); } if (forceDirty) { var forcedDirtyState = this.$__forced || (this.$__forced = {}); forcedDirtyState[name] = true; } else if (rawState[name] === value) { return; } if (!this.$__dirty) { // This is the first time we are modifying the component state // so introduce some properties to do some tracking of // changes to the state this.$__dirty = true; // Mark the component state as dirty (i.e. modified) this.$__old = rawState; this.$__raw = rawState = extend({}, rawState); this.$__changes = {}; this.$__component.$__queueUpdate(); } this.$__changes[name] = value; if (value === undefined) { // Don't store state properties with an undefined or null value delete rawState[name]; } else { // Otherwise, store the new value in the component state rawState[name] = value; } }, toJSON: function() { return this.$__raw; } }; module.exports = State;