-
-
Notifications
You must be signed in to change notification settings - Fork 4.2k
/
Copy pathwatch_key.js
114 lines (95 loc) · 3.5 KB
/
watch_key.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
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
import isEnabled from 'ember-metal/features';
import {
meta as metaFor
} from 'ember-metal/meta';
import {
MANDATORY_SETTER_FUNCTION,
DEFAULT_GETTER_FUNCTION
} from 'ember-metal/properties';
let handleMandatorySetter, lookupDescriptor;
export function watchKey(obj, keyName, meta) {
// can't watch length on Array - it is special...
if (keyName === 'length' && Array.isArray(obj)) { return; }
var m = meta || metaFor(obj);
// activate watching first time
if (!m.peekWatching(keyName)) {
m.writeWatching(keyName, 1);
var possibleDesc = obj[keyName];
var desc = (possibleDesc !== null && typeof possibleDesc === 'object' && possibleDesc.isDescriptor) ? possibleDesc : undefined;
if (desc && desc.willWatch) { desc.willWatch(obj, keyName); }
if ('function' === typeof obj.willWatchProperty) {
obj.willWatchProperty(keyName);
}
if (isEnabled('mandatory-setter')) {
handleMandatorySetter(m, obj, keyName);
}
} else {
m.writeWatching(keyName, (m.peekWatching(keyName) || 0) + 1);
}
}
if (isEnabled('mandatory-setter')) {
lookupDescriptor = function lookupDescriptor(obj, keyName) {
let current = obj;
while (current) {
let descriptor = Object.getOwnPropertyDescriptor(current, keyName);
if (descriptor) {
return descriptor;
}
current = Object.getPrototypeOf(current);
}
return null;
};
handleMandatorySetter = function handleMandatorySetter(m, obj, keyName) {
let descriptor = lookupDescriptor(obj, keyName);
var configurable = descriptor ? descriptor.configurable : true;
var isWritable = descriptor ? descriptor.writable : true;
var hasValue = descriptor ? 'value' in descriptor : true;
var possibleDesc = descriptor && descriptor.value;
var isDescriptor = possibleDesc !== null && typeof possibleDesc === 'object' && possibleDesc.isDescriptor;
if (isDescriptor) { return; }
// this x in Y deopts, so keeping it in this function is better;
if (configurable && isWritable && hasValue && keyName in obj) {
m.writeValues(keyName, obj[keyName]);
Object.defineProperty(obj, keyName, {
configurable: true,
enumerable: Object.prototype.propertyIsEnumerable.call(obj, keyName),
set: MANDATORY_SETTER_FUNCTION(keyName),
get: DEFAULT_GETTER_FUNCTION(keyName)
});
}
};
}
export function unwatchKey(obj, keyName, meta) {
var m = meta || metaFor(obj);
let count = m.peekWatching(keyName);
if (count === 1) {
m.writeWatching(keyName, 0);
var possibleDesc = obj[keyName];
var desc = (possibleDesc !== null && typeof possibleDesc === 'object' && possibleDesc.isDescriptor) ? possibleDesc : undefined;
if (desc && desc.didUnwatch) { desc.didUnwatch(obj, keyName); }
if ('function' === typeof obj.didUnwatchProperty) {
obj.didUnwatchProperty(keyName);
}
if (isEnabled('mandatory-setter')) {
if (!desc && keyName in obj) {
Object.defineProperty(obj, keyName, {
configurable: true,
enumerable: Object.prototype.propertyIsEnumerable.call(obj, keyName),
set(val) {
// redefine to set as enumerable
Object.defineProperty(obj, keyName, {
configurable: true,
writable: true,
enumerable: true,
value: val
});
m.deleteFromValues(keyName);
},
get: DEFAULT_GETTER_FUNCTION(keyName)
});
}
}
} else if (count > 1) {
m.writeWatching(keyName, count - 1);
}
}