Skip to content

Commit

Permalink
Use WeakMap is available for domData storage.
Browse files Browse the repository at this point in the history
  • Loading branch information
mbest committed Jul 21, 2017
1 parent b58e2b6 commit 199fdb2
Showing 1 changed file with 43 additions and 26 deletions.
69 changes: 43 additions & 26 deletions src/utils.domData.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,41 +4,58 @@ ko.utils.domData = new (function () {
var dataStoreKeyExpandoPropertyName = "__ko__" + (new Date).getTime();
var dataStore = {};

function getAll(node, createIfNotFound) {
var dataStoreKey = node[dataStoreKeyExpandoPropertyName];
var hasExistingDataStore = dataStoreKey && (dataStoreKey !== "null") && dataStore[dataStoreKey];
if (!hasExistingDataStore) {
if (!createIfNotFound)
return undefined;
dataStoreKey = node[dataStoreKeyExpandoPropertyName] = "ko" + uniqueId++;
dataStore[dataStoreKey] = {};
}
return dataStore[dataStoreKey];
}

return {
get: function (node, key) {
var allDataForNode = getAll(node, false);
return allDataForNode === undefined ? undefined : allDataForNode[key];
},
set: function (node, key, value) {
if (value === undefined) {
// Make sure we don't actually create a new domData key if we are actually deleting a value
if (getAll(node, false) === undefined)
return;
var getDataForNode, clear;
if (window['WeakMap']) {
getDataForNode = function (node, createIfNotFound) {
var ownerDoc = node.ownerDocument,
dataStore = ownerDoc[dataStoreKeyExpandoPropertyName] || (ownerDoc[dataStoreKeyExpandoPropertyName] = new ownerDoc.defaultView['WeakMap']());
if (dataStore['has'](node)) {
return dataStore.get(node);
}
var allDataForNode = getAll(node, true);
allDataForNode[key] = value;
},
clear: function (node) {
if (createIfNotFound) {
var dataForNode = {};
dataStore.set(node, dataForNode);
return dataForNode;
}
};
clear = function (node) {
var dataStore = node.ownerDocument[dataStoreKeyExpandoPropertyName];
return dataStore ? dataStore['delete'](node) : false;
};
} else {
getDataForNode = function (node, createIfNotFound) {
var dataStoreKey = node[dataStoreKeyExpandoPropertyName];
var hasExistingDataStore = dataStoreKey && (dataStoreKey !== "null") && dataStore[dataStoreKey];
if (!hasExistingDataStore) {
if (!createIfNotFound)
return undefined;
dataStoreKey = node[dataStoreKeyExpandoPropertyName] = "ko" + uniqueId++;
dataStore[dataStoreKey] = {};
}
return dataStore[dataStoreKey];
};
clear = function (node) {
var dataStoreKey = node[dataStoreKeyExpandoPropertyName];
if (dataStoreKey) {
delete dataStore[dataStoreKey];
node[dataStoreKeyExpandoPropertyName] = null;
return true; // Exposing "did clean" flag purely so specs can infer whether things have been cleaned up as intended
}
return false;
};
}

return {
get: function (node, key) {
var dataForNode = getDataForNode(node, false);
return dataForNode && dataForNode[key];
},
set: function (node, key, value) {
// Make sure we don't actually create a new domData key if we are actually deleting a value
var dataForNode = getDataForNode(node, value !== undefined /* createIfNotFound */);
dataForNode && (dataForNode[key] = value);
},
clear: clear,

nextKey: function () {
return (uniqueId++) + dataStoreKeyExpandoPropertyName;
Expand Down

0 comments on commit 199fdb2

Please sign in to comment.