|
|
@ -12,16 +12,10 @@ function Store(state) {
|
|
|
|
this._changeHandlers = [];
|
|
|
|
this._changeHandlers = [];
|
|
|
|
this._dependents = [];
|
|
|
|
this._dependents = [];
|
|
|
|
|
|
|
|
|
|
|
|
this._proto = blankObject();
|
|
|
|
this._computed = blankObject();
|
|
|
|
this._changed = blankObject();
|
|
|
|
this._sortedComputedProperties = [];
|
|
|
|
this._dependentProps = blankObject();
|
|
|
|
|
|
|
|
this._dirty = blankObject();
|
|
|
|
|
|
|
|
this._state = Object.create(this._proto);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
for (var key in state) {
|
|
|
|
this._state = assign({}, state);
|
|
|
|
this._changed[key] = true;
|
|
|
|
|
|
|
|
this._state[key] = state[key];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
assign(Store.prototype, {
|
|
|
|
assign(Store.prototype, {
|
|
|
@ -35,17 +29,6 @@ assign(Store.prototype, {
|
|
|
|
});
|
|
|
|
});
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
_makeDirty: function(prop) {
|
|
|
|
|
|
|
|
var dependentProps = this._dependentProps[prop];
|
|
|
|
|
|
|
|
if (dependentProps) {
|
|
|
|
|
|
|
|
for (var i = 0; i < dependentProps.length; i += 1) {
|
|
|
|
|
|
|
|
var dependentProp = dependentProps[i];
|
|
|
|
|
|
|
|
this._dirty[dependentProp] = this._changed[dependentProp] = true;
|
|
|
|
|
|
|
|
this._makeDirty(dependentProp);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
_init: function(props) {
|
|
|
|
_init: function(props) {
|
|
|
|
var state = {};
|
|
|
|
var state = {};
|
|
|
|
for (let i = 0; i < props.length; i += 1) {
|
|
|
|
for (let i = 0; i < props.length; i += 1) {
|
|
|
@ -65,51 +48,51 @@ assign(Store.prototype, {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
compute: function(key, deps, fn) {
|
|
|
|
_sortComputedProperties() {
|
|
|
|
var store = this;
|
|
|
|
var computed = this._computed;
|
|
|
|
var value;
|
|
|
|
var sorted = this._sortedComputedProperties = [];
|
|
|
|
|
|
|
|
var visited = blankObject();
|
|
|
|
|
|
|
|
|
|
|
|
store._dirty[key] = true;
|
|
|
|
function visit(key) {
|
|
|
|
|
|
|
|
if (visited[key]) return;
|
|
|
|
|
|
|
|
var c = computed[key];
|
|
|
|
|
|
|
|
|
|
|
|
for (var i = 0; i < deps.length; i += 1) {
|
|
|
|
if (c) {
|
|
|
|
var dep = deps[i];
|
|
|
|
c.deps.forEach(visit);
|
|
|
|
if (!this._dependentProps[dep]) this._dependentProps[dep] = [];
|
|
|
|
sorted.push(c);
|
|
|
|
this._dependentProps[dep].push(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object.defineProperty(this._proto, key, {
|
|
|
|
for (var key in this._computed) visit(key);
|
|
|
|
enumerable: true,
|
|
|
|
},
|
|
|
|
get: function() {
|
|
|
|
|
|
|
|
if (store._dirty[key]) {
|
|
|
|
compute: function(key, deps, fn) {
|
|
|
|
|
|
|
|
var store = this;
|
|
|
|
|
|
|
|
var value;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var c = {
|
|
|
|
|
|
|
|
deps: deps,
|
|
|
|
|
|
|
|
update: function(state, changed, dirty) {
|
|
|
|
var values = deps.map(function(dep) {
|
|
|
|
var values = deps.map(function(dep) {
|
|
|
|
if (dep in store._changed) changed = true;
|
|
|
|
if (dep in changed) dirty = true;
|
|
|
|
return store._state[dep];
|
|
|
|
return state[dep];
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (dirty) {
|
|
|
|
var newValue = fn.apply(null, values);
|
|
|
|
var newValue = fn.apply(null, values);
|
|
|
|
|
|
|
|
|
|
|
|
if (differs(newValue, value)) {
|
|
|
|
if (differs(newValue, value)) {
|
|
|
|
value = newValue;
|
|
|
|
value = newValue;
|
|
|
|
store._changed[key] = true;
|
|
|
|
changed[key] = true;
|
|
|
|
|
|
|
|
state[key] = value;
|
|
|
|
var dependentProps = store._dependentProps[key];
|
|
|
|
|
|
|
|
if (dependentProps) {
|
|
|
|
|
|
|
|
for (var i = 0; i < dependentProps.length; i += 1) {
|
|
|
|
|
|
|
|
var prop = dependentProps[i];
|
|
|
|
|
|
|
|
store._dirty[prop] = store._changed[prop] = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
store._dirty[key] = false;
|
|
|
|
c.update(this._state, {}, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
this._computed[key] = c;
|
|
|
|
},
|
|
|
|
this._sortComputedProperties();
|
|
|
|
set: function() {
|
|
|
|
|
|
|
|
throw new Error(`'${key}' is a read-only property`);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
},
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
onchange: function(callback) {
|
|
|
|
onchange: function(callback) {
|
|
|
@ -128,13 +111,16 @@ assign(Store.prototype, {
|
|
|
|
dirty = false;
|
|
|
|
dirty = false;
|
|
|
|
|
|
|
|
|
|
|
|
for (var key in newState) {
|
|
|
|
for (var key in newState) {
|
|
|
|
|
|
|
|
if (this._computed[key]) throw new Error("'" + key + "' is a read-only property");
|
|
|
|
if (differs(newState[key], oldState[key])) changed[key] = dirty = true;
|
|
|
|
if (differs(newState[key], oldState[key])) changed[key] = dirty = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!dirty) return;
|
|
|
|
if (!dirty) return;
|
|
|
|
|
|
|
|
|
|
|
|
this._state = assign(Object.create(this._proto), oldState, newState);
|
|
|
|
this._state = assign({}, oldState, newState);
|
|
|
|
|
|
|
|
|
|
|
|
for (var key in changed) this._makeDirty(key);
|
|
|
|
for (var i = 0; i < this._sortedComputedProperties.length; i += 1) {
|
|
|
|
|
|
|
|
this._sortedComputedProperties[i].update(this._state, changed);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for (var i = 0; i < this._changeHandlers.length; i += 1) {
|
|
|
|
for (var i = 0; i < this._changeHandlers.length; i += 1) {
|
|
|
|
this._changeHandlers[i](this._state, changed);
|
|
|
|
this._changeHandlers[i](this._state, changed);
|
|
|
@ -177,8 +163,6 @@ function combineStores(store, children) {
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
console.log('updates', updates);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
store.set(updates);
|
|
|
|
store.set(updates);
|
|
|
|
return store;
|
|
|
|
return store;
|
|
|
|
}
|
|
|
|
}
|
|
|
|