-
Notifications
You must be signed in to change notification settings - Fork 0
/
knockout-es5-proper.js
542 lines (462 loc) · 21.5 KB
/
knockout-es5-proper.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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
/*!
* Knockout ES5 plugin - https://github.com/SteveSanderson/knockout-es5
* Copyright (c) Steve Sanderson
* MIT license
*/
(function(global, undefined) {
'use strict';
var ko;
// Model tracking
// --------------
//
// This is the central feature of Knockout-ES5. We augment model objects by converting properties
// into ES5 getter/setter pairs that read/write an underlying Knockout observable. This means you can
// use plain JavaScript syntax to read/write the property while still getting the full benefits of
// Knockout's automatic dependency detection and notification triggering.
//
// For comparison, here's Knockout ES3-compatible syntax:
//
// var firstNameLength = myModel.user().firstName().length; // Read
// myModel.user().firstName('Bert'); // Write
//
// ... versus Knockout-ES5 syntax:
//
// var firstNameLength = myModel.user.firstName.length; // Read
// myModel.user.firstName = 'Bert'; // Write
// `ko.track(model)` converts each property on the given model object into a getter/setter pair that
// wraps a Knockout observable. Optionally specify an array of property names to wrap; otherwise we
// wrap all properties. If any of the properties are already observables, we replace them with
// ES5 getter/setter pairs that wrap your original observable instances. In the case of readonly
// ko.computed properties, we simply do not define a setter (so attempted writes will be ignored,
// which is how ES5 readonly properties normally behave).
//
// By design, this does *not* recursively walk child object properties, because making literally
// everything everywhere independently observable is usually unhelpful. When you do want to track
// child object properties independently, define your own class for those child objects and put
// a separate ko.track call into its constructor --- this gives you far more control.
/**
* @param {object} obj
* @param {object|array.<string>} propertyNamesOrSettings
* @param {boolean} propertyNamesOrSettings.deep Use deep track.
* @param {array.<string>} propertyNamesOrSettings.fields Array of property names to wrap.
* todo: @param {array.<string>} propertyNamesOrSettings.exclude Array of exclude property names to wrap.
* todo: @param {function(string, *):boolean} propertyNamesOrSettings.filter Function to filter property
* names to wrap. A function that takes ... params
* @return {object}
*/
function track(obj, propertyNamesOrSettings) {
if (!obj || typeof obj !== 'object') {
throw new Error('When calling ko.track, you must pass an object as the first parameter.');
}
var propertyNames;
if ( isPlainObject(propertyNamesOrSettings) ) {
// defaults
propertyNamesOrSettings.deep = propertyNamesOrSettings.deep || false;
propertyNamesOrSettings.fields = propertyNamesOrSettings.fields || Object.getOwnPropertyNames(obj);
propertyNamesOrSettings.lazy = propertyNamesOrSettings.lazy || false;
wrap(obj, propertyNamesOrSettings.fields, propertyNamesOrSettings);
} else {
propertyNames = propertyNamesOrSettings || Object.getOwnPropertyNames(obj);
wrap(obj, propertyNames, {});
}
return obj;
}
// fix for ie
var rFunctionName = /^function\s*([^\s(]+)/;
function getFunctionName( ctor ){
if (ctor.name) {
return ctor.name;
}
return (ctor.toString().trim().match( rFunctionName ) || [])[1];
}
function canTrack(obj) {
return obj && typeof obj === 'object' && getFunctionName(obj.constructor) === 'Object';
}
function createPropertyDescriptor(originalValue, prop, map) {
var isObservable = ko.isObservable(originalValue);
var isArray = !isObservable && Array.isArray(originalValue);
var observable = isObservable ? originalValue
: isArray ? ko.observableArray(originalValue)
: ko.observable(originalValue);
map[prop] = function () { return observable; };
// add check in case the object is already an observable array
if (isArray || (isObservable && 'push' in observable)) {
notifyWhenPresentOrFutureArrayValuesMutate(ko, observable);
}
return {
configurable: true,
enumerable: true,
get: observable,
set: ko.isWriteableObservable(observable) ? observable : undefined
};
}
function createLazyPropertyDescriptor(originalValue, prop, map) {
if (ko.isObservable(originalValue)) {
// no need to be lazy if we already have an observable
return createPropertyDescriptor(originalValue, prop, map);
}
var observable;
function getOrCreateObservable(value, writing) {
if (observable) {
return writing ? observable(value) : observable;
}
if (Array.isArray(value)) {
observable = ko.observableArray(value);
notifyWhenPresentOrFutureArrayValuesMutate(ko, observable);
return observable;
}
return (observable = ko.observable(value));
}
map[prop] = function () { return getOrCreateObservable(originalValue); };
return {
configurable: true,
enumerable: true,
get: function () { return getOrCreateObservable(originalValue)(); },
set: function (value) { getOrCreateObservable(value, true); }
};
}
function wrap(obj, props, options) {
if (!props.length) {
return;
}
var allObservablesForObject = getAllObservablesForObject(obj, true);
var descriptors = {};
props.forEach(function (prop) {
// Skip properties that are already tracked
if (prop in allObservablesForObject) {
return;
}
// Skip properties where descriptor can't be redefined
if (Object.getOwnPropertyDescriptor(obj, prop).configurable === false){
return;
}
var originalValue = obj[prop];
descriptors[prop] = (options.lazy ? createLazyPropertyDescriptor : createPropertyDescriptor)
(originalValue, prop, allObservablesForObject);
if (options.deep && canTrack(originalValue)) {
wrap(originalValue, Object.keys(originalValue), options);
}
});
Object.defineProperties(obj, descriptors);
}
function isPlainObject( obj ){
return !!obj && typeof obj === 'object' && obj.constructor === Object;
}
// Lazily created by `getAllObservablesForObject` below. Has to be created lazily because the
// WeakMap factory isn't available until the module has finished loading (may be async).
var objectToObservableMap;
// Gets or creates the hidden internal key-value collection of observables corresponding to
// properties on the model object.
function getAllObservablesForObject(obj, createIfNotDefined) {
if (!objectToObservableMap) {
objectToObservableMap = weakMapFactory();
}
var result = objectToObservableMap.get(obj);
if (!result && createIfNotDefined) {
result = {};
objectToObservableMap.set(obj, result);
}
return result;
}
// Removes the internal references to observables mapped to the specified properties
// or the entire object reference if no properties are passed in. This allows the
// observables to be replaced and tracked again.
function untrack(obj, propertyNames) {
if (!objectToObservableMap) {
return;
}
if (arguments.length === 1) {
objectToObservableMap['delete'](obj);
} else {
var allObservablesForObject = getAllObservablesForObject(obj, false);
if (allObservablesForObject) {
propertyNames.forEach(function(propertyName) {
delete allObservablesForObject[propertyName];
});
}
}
}
// Computed properties
// -------------------
//
// The preceding code is already sufficient to upgrade ko.computed model properties to ES5
// getter/setter pairs (or in the case of readonly ko.computed properties, just a getter).
// These then behave like a regular property with a getter function, except they are smarter:
// your evaluator is only invoked when one of its dependencies changes. The result is cached
// and used for all evaluations until the next time a dependency changes).
//
// However, instead of forcing developers to declare a ko.computed property explicitly, it's
// nice to offer a utility function that declares a computed getter directly.
// Implements `ko.defineProperty`
function defineComputedProperty(obj, propertyName, evaluatorOrOptions) {
var ko = this,
computedOptions = { owner: obj, deferEvaluation: true };
if (typeof evaluatorOrOptions === 'function') {
computedOptions.read = evaluatorOrOptions;
} else {
if ('value' in evaluatorOrOptions) {
throw new Error('For ko.defineProperty, you must not specify a "value" for the property. ' +
'You must provide a "get" function.');
}
if (typeof evaluatorOrOptions.get !== 'function') {
throw new Error('For ko.defineProperty, the third parameter must be either an evaluator function, ' +
'or an options object containing a function called "get".');
}
computedOptions.read = evaluatorOrOptions.get;
computedOptions.write = evaluatorOrOptions.set;
}
obj[propertyName] = ko.computed(computedOptions);
track.call(ko, obj, [propertyName]);
return obj;
}
// Array handling
// --------------
//
// Arrays are special, because unlike other property types, they have standard mutator functions
// (`push`/`pop`/`splice`/etc.) and it's desirable to trigger a change notification whenever one of
// those mutator functions is invoked.
//
// Traditionally, Knockout handles this by putting special versions of `push`/`pop`/etc. on observable
// arrays that mutate the underlying array and then trigger a notification. That approach doesn't
// work for Knockout-ES5 because properties now return the underlying arrays, so the mutator runs
// in the context of the underlying array, not any particular observable:
//
// // Operates on the underlying array value
// myModel.someCollection.push('New value');
//
// To solve this, Knockout-ES5 detects array values, and modifies them as follows:
// 1. Associates a hidden subscribable with each array instance that it encounters
// 2. Intercepts standard mutators (`push`/`pop`/etc.) and makes them trigger the subscribable
// Then, for model properties whose values are arrays, the property's underlying observable
// subscribes to the array subscribable, so it can trigger a change notification after mutation.
// Given an observable that underlies a model property, watch for any array value that might
// be assigned as the property value, and hook into its change events
function notifyWhenPresentOrFutureArrayValuesMutate(ko, observable) {
var watchingArraySubscription = null;
ko.computed(function () {
// Unsubscribe to any earlier array instance
if (watchingArraySubscription) {
watchingArraySubscription.dispose();
watchingArraySubscription = null;
}
// Subscribe to the new array instance
var newArrayInstance = observable();
if (newArrayInstance instanceof Array) {
watchingArraySubscription = startWatchingArrayInstance(ko, observable, newArrayInstance);
}
});
}
// Listens for array mutations, and when they happen, cause the observable to fire notifications.
// This is used to make model properties of type array fire notifications when the array changes.
// Returns a subscribable that can later be disposed.
function startWatchingArrayInstance(ko, observable, arrayInstance) {
var subscribable = getSubscribableForArray(ko, arrayInstance);
return subscribable.subscribe(observable);
}
// Lazily created by `getSubscribableForArray` below. Has to be created lazily because the
// WeakMap factory isn't available until the module has finished loading (may be async).
var arraySubscribablesMap;
// Gets or creates a subscribable that fires after each array mutation
function getSubscribableForArray(ko, arrayInstance) {
if (!arraySubscribablesMap) {
arraySubscribablesMap = weakMapFactory();
}
var subscribable = arraySubscribablesMap.get(arrayInstance);
if (!subscribable) {
subscribable = new ko.subscribable();
arraySubscribablesMap.set(arrayInstance, subscribable);
var notificationPauseSignal = {};
wrapStandardArrayMutators(arrayInstance, subscribable, notificationPauseSignal);
addKnockoutArrayMutators(ko, arrayInstance, subscribable, notificationPauseSignal);
}
return subscribable;
}
// After each array mutation, fires a notification on the given subscribable
function wrapStandardArrayMutators(arrayInstance, subscribable, notificationPauseSignal) {
['pop', 'push', 'reverse', 'shift', 'sort', 'splice', 'unshift'].forEach(function(fnName) {
var origMutator = arrayInstance[fnName];
arrayInstance[fnName] = function() {
var result = origMutator.apply(this, arguments);
if (notificationPauseSignal.pause !== true) {
subscribable.notifySubscribers(this);
}
return result;
};
});
}
// Adds Knockout's additional array mutation functions to the array
function addKnockoutArrayMutators(ko, arrayInstance, subscribable, notificationPauseSignal) {
['remove', 'removeAll', 'destroy', 'destroyAll', 'replace'].forEach(function(fnName) {
// Make it a non-enumerable property for consistency with standard Array functions
Object.defineProperty(arrayInstance, fnName, {
enumerable: false,
value: function() {
var result;
// These additional array mutators are built using the underlying push/pop/etc.
// mutators, which are wrapped to trigger notifications. But we don't want to
// trigger multiple notifications, so pause the push/pop/etc. wrappers and
// delivery only one notification at the end of the process.
notificationPauseSignal.pause = true;
try {
// Creates a temporary observableArray that can perform the operation.
result = ko.observableArray.fn[fnName].apply(ko.observableArray(arrayInstance), arguments);
}
finally {
notificationPauseSignal.pause = false;
}
subscribable.notifySubscribers(arrayInstance);
return result;
}
});
});
}
// Static utility functions
// ------------------------
//
// Since Knockout-ES5 sets up properties that return values, not observables, you can't
// trivially subscribe to the underlying observables (e.g., `someProperty.subscribe(...)`),
// or tell them that object values have mutated, etc. To handle this, we set up some
// extra utility functions that can return or work with the underlying observables.
// Returns the underlying observable associated with a model property (or `null` if the
// model or property doesn't exist, or isn't associated with an observable). This means
// you can subscribe to the property, e.g.:
//
// ko.getObservable(model, 'propertyName')
// .subscribe(function(newValue) { ... });
function getObservable(obj, propertyName) {
if (!obj || typeof obj !== 'object') {
return null;
}
var allObservablesForObject = getAllObservablesForObject(obj, false);
if (allObservablesForObject && propertyName in allObservablesForObject) {
return allObservablesForObject[propertyName]();
}
return null;
}
// Returns a boolean indicating whether the property on the object has an underlying
// observables. This does the check in a way not to create an observable if the
// object was created with lazily created observables
function isTracked(obj, propertyName) {
if (!obj || typeof obj !== 'object') {
return false;
}
var allObservablesForObject = getAllObservablesForObject(obj, false);
return !!allObservablesForObject && propertyName in allObservablesForObject;
}
// Causes a property's associated observable to fire a change notification. Useful when
// the property value is a complex object and you've modified a child property.
function valueHasMutated(obj, propertyName) {
var observable = getObservable(obj, propertyName);
if (observable) {
observable.valueHasMutated();
}
}
// Module initialisation
// ---------------------
//
// When this script is first evaluated, it works out what kind of module loading scenario
// it is in (Node.js or a browser `<script>` tag), stashes a reference to its dependencies
// (currently that's just the WeakMap shim), and then finally attaches itself to whichever
// instance of Knockout.js it can find.
// A function that returns a new ES6-compatible WeakMap instance (using ES5 shim if needed).
// Instantiated by prepareExports, accounting for which module loader is being used.
var weakMapFactory;
// Extends a Knockout instance with Knockout-ES5 functionality
function attachToKo(ko) {
ko.track = track;
ko.untrack = untrack;
ko.getObservable = getObservable;
ko.valueHasMutated = valueHasMutated;
ko.defineProperty = defineComputedProperty;
// todo: test it, maybe added it to ko. directly
ko.es5 = {
getAllObservablesForObject: getAllObservablesForObject,
notifyWhenPresentOrFutureArrayValuesMutate: notifyWhenPresentOrFutureArrayValuesMutate,
isTracked: isTracked
};
// Custom Binding Provider
// -------------------
//
// To ensure that when using this plugin any custom bindings are provided with the observable
// rather than only the value of the property, a custom binding provider supplies bindings with
// actual observable values. The built in bindings use Knockout's internal `_ko_property_writers`
// feature to be able to write back to the property, but custom bindings may not be able to use
// that, especially if they use an options object.
function CustomBindingProvider(providerToWrap) {
this.bindingCache = {};
this._providerToWrap = providerToWrap;
this._nativeBindingProvider = new ko.bindingProvider();
}
CustomBindingProvider.prototype.nodeHasBindings = function() {
return this._providerToWrap.nodeHasBindings.apply(this._providerToWrap, arguments);
};
CustomBindingProvider.prototype.getBindingAccessors = function(node, bindingContext) {
var bindingsString = this._nativeBindingProvider.getBindingsString(node, bindingContext);
return bindingsString ? this.parseBindingsString(bindingsString, bindingContext, node, {'valueAccessors':true}) : null;
};
CustomBindingProvider.prototype.parseBindingsString = function(bindingsString, bindingContext, node, options) {
try {
var bindingFunction = createBindingsStringEvaluatorViaCache(bindingsString, this.bindingCache, options);
return bindingFunction(bindingContext, node);
} catch (ex) {
ex.message = 'Unable to parse bindings.\nBindings value: ' + bindingsString + '\nMessage: ' + ex.message;
throw ex;
}
};
function preProcessBindings(bindingsStringOrKeyValueArray, bindingOptions) {
bindingOptions = bindingOptions || {};
function processKeyValue(key, val) {
// Handle arrays if value starts with bracket
if(val.match(/^\[/)){
// This is required or will throw errors
resultStrings.push(key + ':ko.observableArray(' + val + ')');
}else{
resultStrings.push(key + ':ko.getObservable($data,"' + val + '")||' + val);
}
}
var resultStrings = [],
keyValueArray = typeof bindingsStringOrKeyValueArray === 'string' ?
ko.expressionRewriting.parseObjectLiteral(bindingsStringOrKeyValueArray) : bindingsStringOrKeyValueArray;
keyValueArray.forEach(function(keyValue) {
processKeyValue(keyValue.key || keyValue.unknown, keyValue.value);
});
return ko.expressionRewriting.preProcessBindings(resultStrings.join(','), bindingOptions);
}
function createBindingsStringEvaluatorViaCache(bindingsString, cache, options) {
var cacheKey = bindingsString + (options && options.valueAccessors || '');
return cache[cacheKey] || (cache[cacheKey] = createBindingsStringEvaluator(bindingsString, options));
}
function createBindingsStringEvaluator(bindingsString, options) {
var rewrittenBindings = preProcessBindings(bindingsString, options),
functionBody = 'with($context){with($data||{}){return{' + rewrittenBindings + '}}}';
/* jshint -W054 */
return new Function('$context', '$element', functionBody);
}
ko.es5BindingProvider = CustomBindingProvider;
ko.bindingProvider.instance = new CustomBindingProvider(ko.bindingProvider.instance);
}
// Determines which module loading scenario we're in, grabs dependencies, and attaches to KO
function prepareExports() {
if (typeof exports === 'object' && typeof module === 'object') {
// Node.js case - load KO and WeakMap modules synchronously
ko = require('knockout');
var WM = require('../lib/weakmap');
attachToKo(ko);
weakMapFactory = function() { return new WM(); };
module.exports = ko;
} else if (typeof define === 'function' && define.amd) {
define(['knockout'], function(koModule) {
ko = koModule;
attachToKo(koModule);
weakMapFactory = function() { return new global.WeakMap(); };
return koModule;
});
} else if ('ko' in global) {
// Non-module case - attach to the global instance, and assume a global WeakMap constructor
ko = global.ko;
attachToKo(global.ko);
weakMapFactory = function() { return new global.WeakMap(); };
}
}
prepareExports();
})(this);