update state with new computed properties

pull/1395/head
Rich Harris 8 years ago
parent ba7fc6b580
commit c8d55aa90a

@ -20,24 +20,24 @@ function Store(state, options) {
} }
assign(Store.prototype, { assign(Store.prototype, {
_add: function(component, props) { _add(component, props) {
this._dependents.push({ this._dependents.push({
component: component, component: component,
props: props props: props
}); });
}, },
_init: function(props) { _init(props) {
var state = {}; const state = {};
for (var i = 0; i < props.length; i += 1) { for (let i = 0; i < props.length; i += 1) {
var prop = props[i]; const prop = props[i];
state['$' + prop] = this._state[prop]; state['$' + prop] = this._state[prop];
} }
return state; return state;
}, },
_remove: function(component) { _remove(component) {
var i = this._dependents.length; let i = this._dependents.length;
while (i--) { while (i--) {
if (this._dependents[i].component === component) { if (this._dependents[i].component === component) {
this._dependents.splice(i, 1); this._dependents.splice(i, 1);
@ -46,14 +46,52 @@ assign(Store.prototype, {
} }
}, },
_sortComputedProperties: function() { _set(newState, changed) {
var computed = this._computed; const previous = this._state;
var sorted = this._sortedComputedProperties = []; this._state = assign(assign({}, previous), newState);
var visited = blankObject();
var currentKey; for (let i = 0; i < this._sortedComputedProperties.length; i += 1) {
this._sortedComputedProperties[i].update(this._state, changed);
}
this.fire('state', {
changed,
previous,
current: this._state
});
const dependents = this._dependents.slice(); // guard against mutations
for (let i = 0; i < dependents.length; i += 1) {
const dependent = dependents[i];
const componentState = {};
dirty = false;
for (let j = 0; j < dependent.props.length; j += 1) {
const prop = dependent.props[j];
if (prop in changed) {
componentState['$' + prop] = this._state[prop];
dirty = true;
}
}
if (dirty) dependent.component.set(componentState);
}
this.fire('update', {
changed,
previous,
current: this._state
});
},
_sortComputedProperties() {
const computed = this._computed;
const sorted = this._sortedComputedProperties = [];
const visited = blankObject();
let currentKey;
function visit(key) { function visit(key) {
var c = computed[key]; const c = computed[key];
if (c) { if (c) {
c.deps.forEach(dep => { c.deps.forEach(dep => {
@ -71,26 +109,25 @@ assign(Store.prototype, {
} }
} }
for (var key in this._computed) { for (const key in this._computed) {
visit(currentKey = key); visit(currentKey = key);
} }
}, },
compute: function(key, deps, fn) { compute(key, deps, fn) {
var store = this; let value;
var value;
var c = { const c = {
deps: deps, deps,
update: function(state, changed, dirty) { update: (state, changed, dirty) => {
var values = deps.map(function(dep) { const values = deps.map(dep => {
if (dep in changed) dirty = true; if (dep in changed) dirty = true;
return state[dep]; return state[dep];
}); });
if (dirty) { if (dirty) {
var newValue = fn.apply(null, values); const newValue = fn.apply(null, values);
if (store._differs(newValue, value)) { if (this._differs(newValue, value)) {
value = newValue; value = newValue;
changed[key] = true; changed[key] = true;
state[key] = value; state[key] = value;
@ -99,63 +136,33 @@ assign(Store.prototype, {
} }
}; };
c.update(this._state, {}, true);
this._computed[key] = c; this._computed[key] = c;
this._sortComputedProperties(); this._sortComputedProperties();
const state = assign({}, this._state);
const changed = {};
c.update(state, changed, true);
this._set(state, changed);
}, },
fire: fire, fire,
get: get, get,
on: on, on,
set: function(newState) { set(newState) {
var oldState = this._state, const oldState = this._state;
changed = this._changed = {}, const changed = this._changed = {};
dirty = false; let dirty = false;
for (var key in newState) { for (const key in newState) {
if (this._computed[key]) throw new Error("'" + key + "' is a read-only property"); if (this._computed[key]) throw new Error(`'${key}' is a read-only property`);
if (this._differs(newState[key], oldState[key])) changed[key] = dirty = true; if (this._differs(newState[key], oldState[key])) changed[key] = dirty = true;
} }
if (!dirty) return; if (!dirty) return;
this._state = assign(assign({}, oldState), newState); this._set(newState, changed);
for (var i = 0; i < this._sortedComputedProperties.length; i += 1) {
this._sortedComputedProperties[i].update(this._state, changed);
}
this.fire('state', {
changed: changed,
current: this._state,
previous: oldState
});
var dependents = this._dependents.slice(); // guard against mutations
for (var i = 0; i < dependents.length; i += 1) {
var dependent = dependents[i];
var componentState = {};
dirty = false;
for (var j = 0; j < dependent.props.length; j += 1) {
var prop = dependent.props[j];
if (prop in changed) {
componentState['$' + prop] = this._state[prop];
dirty = true;
}
}
if (dirty) dependent.component.set(componentState);
}
this.fire('update', {
changed: changed,
current: this._state,
previous: oldState
});
} }
}); });

@ -1,6 +1,8 @@
import { Store } from '../../../../store.js'; import { Store } from '../../../../store.js';
export default { export default {
'skip-ssr': true,
store: new Store(), store: new Store(),
html: ` html: `

Loading…
Cancel
Save