forked from knockout/knockout
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdomNodeDisposalBehaviors.js
70 lines (58 loc) · 3.22 KB
/
domNodeDisposalBehaviors.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
describe('DOM node disposal', {
before_each: function () {
testNode = document.createElement("div");
},
'Should run registered disposal callbacks when a node is cleaned': function () {
var didRun = false;
ko.utils.domNodeDisposal.addDisposeCallback(testNode, function() { didRun = true });
value_of(didRun).should_be(false);
ko.cleanNode(testNode);
value_of(didRun).should_be(true);
},
'Should run registered disposal callbacks on descendants when a node is cleaned': function () {
var didRun = false;
var childNode = document.createElement("DIV");
var grandChildNode = document.createElement("DIV");
testNode.appendChild(childNode);
childNode.appendChild(grandChildNode);
ko.utils.domNodeDisposal.addDisposeCallback(grandChildNode, function() { didRun = true });
value_of(didRun).should_be(false);
ko.cleanNode(testNode);
value_of(didRun).should_be(true);
},
'Should run registered disposal callbacks and detach from DOM when a node is removed': function () {
var didRun = false;
var childNode = document.createElement("DIV");
testNode.appendChild(childNode);
ko.utils.domNodeDisposal.addDisposeCallback(childNode, function() { didRun = true });
value_of(didRun).should_be(false);
value_of(testNode.childNodes.length).should_be(1);
ko.removeNode(childNode);
value_of(didRun).should_be(true);
value_of(testNode.childNodes.length).should_be(0);
},
'Should be able to remove previously-registered disposal callbacks': function() {
var didRun = false;
var callback = function() { didRun = true };
ko.utils.domNodeDisposal.addDisposeCallback(testNode, callback);
value_of(didRun).should_be(false);
ko.utils.domNodeDisposal.removeDisposeCallback(testNode, callback);
ko.cleanNode(testNode);
value_of(didRun).should_be(false); // Didn't run only because we removed it
},
'Should be able to attach disposal callback to a node that has been cloned': function() {
// This represents bug https://github.com/SteveSanderson/knockout/issues/324
// IE < 9 copies expando properties when cloning nodes, so if the node already has some DOM data associated with it,
// the DOM data key will be copied too. This causes a problem for disposal, because if the original node gets disposed,
// the shared DOM data is disposed, and then it becomes an error to try to set new DOM data on the clone.
// The solution is to make the DOM-data-setting logic able to recover from the scenario by detecting that the original
// DOM data is gone, and therefore recreating a new DOM data store for the clone.
// Create an element with DOM data
var originalNode = document.createElement("DIV");
ko.utils.domNodeDisposal.addDisposeCallback(originalNode, function() { });
// Clone it, then dispose it. Then check it's still safe to associate DOM data with the clone.
var cloneNode = originalNode.cloneNode(true);
ko.cleanNode(originalNode);
ko.utils.domNodeDisposal.addDisposeCallback(cloneNode, function() { });
}
});