Skip to content

Commit f7581e3

Browse files
committed
Fix Stateful#deliver() so that dcl.superCall() works as expected.
In other words, in the code below, computeProperties() and refreshRendering() should be called *before* the ... code runs: deliver: dcl.superCall(function (sup) { return function () { sup.apply(this, arguments); ... }; })
1 parent 307dd90 commit f7581e3

File tree

3 files changed

+53
-5
lines changed

3 files changed

+53
-5
lines changed

.jshintrc

+1-1
Original file line numberDiff line numberDiff line change
@@ -66,7 +66,7 @@
6666
"maxlen": 120,
6767
"indent": 4,
6868
"maxerr": 250,
69-
"predef": [ "require", "define", "intern" ],
69+
"predef": [ "require", "define", "intern", "WeakMap" ],
7070
"quotmark": "double",
7171
"maxcomplexity": 10
7272
}

Stateful.js

+20-4
Original file line numberDiff line numberDiff line change
@@ -103,7 +103,7 @@ define([
103103
configurable: false,
104104
value: proto[prop]
105105
});
106-
106+
107107
// Setup ES5 getter and setter for this property, unless it already has custom ones.
108108
var descriptor = Object.getOwnPropertyDescriptor(proto, prop);
109109
if (!descriptor.set) {
@@ -196,7 +196,7 @@ define([
196196
*/
197197
_set: function (name, value) {
198198
var shadowPropName = propNames(name).s;
199-
199+
200200
// Add the shadow property to the instance, masking what's in the prototype.
201201
// Use Object.defineProperty() so it's hidden from for(var key in ...) and Object.keys().
202202
Object.defineProperty(this, shadowPropName, {
@@ -278,8 +278,8 @@ define([
278278
advise.after(this, "_notify", function (args) {
279279
h.notify(args[0], args[1]);
280280
}),
281-
advise.after(this, "deliver", h.deliver.bind(h)),
282-
advise.after(this, "discardChanges", h.discardChanges.bind(h))
281+
advise.after(this, "_deliver", h.deliver.bind(h)),
282+
advise.after(this, "_discardChanges", h.discardChanges.bind(h))
283283
];
284284

285285
return {
@@ -306,16 +306,32 @@ define([
306306
_notify: function () {
307307
},
308308

309+
/**
310+
* Don't call this directly, it's a hook-point to register calls to Notifier#deliver().
311+
* @private
312+
*/
313+
_deliver: function () {
314+
},
315+
316+
/**
317+
* Don't call this directly, it's a hook-point to register calls to Notifier#discardChanges().
318+
* @private
319+
*/
320+
_discardChanges: function () {
321+
},
322+
309323
/**
310324
* Synchronously deliver change records to all listeners registered via `observe()`.
311325
*/
312326
deliver: function () {
327+
this._deliver();
313328
},
314329

315330
/**
316331
* Discard change records for all listeners registered via `observe()`.
317332
*/
318333
discardChanges: function () {
334+
this._discardChanges();
319335
}
320336
}, {
321337
enumerable: false

tests/unit/Stateful.js

+32
Original file line numberDiff line numberDiff line change
@@ -387,6 +387,38 @@ define(function (require) {
387387
}), 10);
388388
},
389389

390+
"overriding Stateful#deliver()": function () {
391+
var log = [];
392+
var MyStateful = dcl(Stateful, {
393+
// Properties.
394+
foo: undefined,
395+
bar: undefined,
396+
397+
// Override Stateful#deliver() to test that sup.apply() calls observe() callbacks.
398+
deliver: dcl.superCall(function (sup) {
399+
return function () {
400+
log.push("deliver before");
401+
sup.apply(this, arguments);
402+
log.push("deliver after");
403+
};
404+
})
405+
});
406+
var stateful = new MyStateful({
407+
foo: "Foo0",
408+
bar: "Bar0"
409+
});
410+
stateful.observe(function () {
411+
log.push("observer 1");
412+
});
413+
stateful.observe(function () {
414+
log.push("observer 2");
415+
});
416+
stateful.foo = "Foo1";
417+
stateful.bar = "Bar1";
418+
stateful.deliver();
419+
assert.deepEqual(log, ["deliver before", "observer 1", "observer 2", "deliver after"], "deliver()");
420+
},
421+
390422
"observe filter": function () {
391423
// Check to make sure reported changes are consistent between platforms with and without Object.observe()
392424
// native support

0 commit comments

Comments
 (0)