|
1 /** |
|
2 * Backbone-relational.js 0.6.0 |
|
3 * (c) 2011 Paul Uithol |
|
4 * |
|
5 * Backbone-relational may be freely distributed under the MIT license; see the accompanying LICENSE.txt. |
|
6 * For details and documentation: https://github.com/PaulUithol/Backbone-relational. |
|
7 * Depends on Backbone (and thus on Underscore as well): https://github.com/documentcloud/backbone. |
|
8 */ |
|
9 ( function( undefined ) { |
|
10 "use strict"; |
|
11 |
|
12 /** |
|
13 * CommonJS shim |
|
14 **/ |
|
15 var _, Backbone, exports; |
|
16 if ( typeof window === 'undefined' ) { |
|
17 _ = require( 'underscore' ); |
|
18 Backbone = require( 'backbone' ); |
|
19 exports = module.exports = Backbone; |
|
20 } |
|
21 else { |
|
22 _ = window._; |
|
23 Backbone = window.Backbone; |
|
24 exports = window; |
|
25 } |
|
26 |
|
27 Backbone.Relational = { |
|
28 showWarnings: true |
|
29 }; |
|
30 |
|
31 /** |
|
32 * Semaphore mixin; can be used as both binary and counting. |
|
33 **/ |
|
34 Backbone.Semaphore = { |
|
35 _permitsAvailable: null, |
|
36 _permitsUsed: 0, |
|
37 |
|
38 acquire: function() { |
|
39 if ( this._permitsAvailable && this._permitsUsed >= this._permitsAvailable ) { |
|
40 throw new Error( 'Max permits acquired' ); |
|
41 } |
|
42 else { |
|
43 this._permitsUsed++; |
|
44 } |
|
45 }, |
|
46 |
|
47 release: function() { |
|
48 if ( this._permitsUsed === 0 ) { |
|
49 throw new Error( 'All permits released' ); |
|
50 } |
|
51 else { |
|
52 this._permitsUsed--; |
|
53 } |
|
54 }, |
|
55 |
|
56 isLocked: function() { |
|
57 return this._permitsUsed > 0; |
|
58 }, |
|
59 |
|
60 setAvailablePermits: function( amount ) { |
|
61 if ( this._permitsUsed > amount ) { |
|
62 throw new Error( 'Available permits cannot be less than used permits' ); |
|
63 } |
|
64 this._permitsAvailable = amount; |
|
65 } |
|
66 }; |
|
67 |
|
68 /** |
|
69 * A BlockingQueue that accumulates items while blocked (via 'block'), |
|
70 * and processes them when unblocked (via 'unblock'). |
|
71 * Process can also be called manually (via 'process'). |
|
72 */ |
|
73 Backbone.BlockingQueue = function() { |
|
74 this._queue = []; |
|
75 }; |
|
76 _.extend( Backbone.BlockingQueue.prototype, Backbone.Semaphore, { |
|
77 _queue: null, |
|
78 |
|
79 add: function( func ) { |
|
80 if ( this.isBlocked() ) { |
|
81 this._queue.push( func ); |
|
82 } |
|
83 else { |
|
84 func(); |
|
85 } |
|
86 }, |
|
87 |
|
88 process: function() { |
|
89 while ( this._queue && this._queue.length ) { |
|
90 this._queue.shift()(); |
|
91 } |
|
92 }, |
|
93 |
|
94 block: function() { |
|
95 this.acquire(); |
|
96 }, |
|
97 |
|
98 unblock: function() { |
|
99 this.release(); |
|
100 if ( !this.isBlocked() ) { |
|
101 this.process(); |
|
102 } |
|
103 }, |
|
104 |
|
105 isBlocked: function() { |
|
106 return this.isLocked(); |
|
107 } |
|
108 }); |
|
109 /** |
|
110 * Global event queue. Accumulates external events ('add:<key>', 'remove:<key>' and 'update:<key>') |
|
111 * until the top-level object is fully initialized (see 'Backbone.RelationalModel'). |
|
112 */ |
|
113 Backbone.Relational.eventQueue = new Backbone.BlockingQueue(); |
|
114 |
|
115 /** |
|
116 * Backbone.Store keeps track of all created (and destruction of) Backbone.RelationalModel. |
|
117 * Handles lookup for relations. |
|
118 */ |
|
119 Backbone.Store = function() { |
|
120 this._collections = []; |
|
121 this._reverseRelations = []; |
|
122 this._subModels = []; |
|
123 this._modelScopes = [ exports ]; |
|
124 }; |
|
125 _.extend( Backbone.Store.prototype, Backbone.Events, { |
|
126 addModelScope: function( scope ) { |
|
127 this._modelScopes.push( scope ); |
|
128 }, |
|
129 |
|
130 /** |
|
131 * Add a set of subModelTypes to the store, that can be used to resolve the '_superModel' |
|
132 * for a model later in 'setupSuperModel'. |
|
133 * |
|
134 * @param {Backbone.RelationalModel} subModelTypes |
|
135 * @param {Backbone.RelationalModel} superModelType |
|
136 */ |
|
137 addSubModels: function( subModelTypes, superModelType ) { |
|
138 this._subModels.push({ |
|
139 'superModelType': superModelType, |
|
140 'subModels': subModelTypes |
|
141 }); |
|
142 }, |
|
143 |
|
144 /** |
|
145 * Check if the given modelType is registered as another model's subModel. If so, add it to the super model's |
|
146 * '_subModels', and set the modelType's '_superModel', '_subModelTypeName', and '_subModelTypeAttribute'. |
|
147 * |
|
148 * @param {Backbone.RelationalModel} modelType |
|
149 */ |
|
150 setupSuperModel: function( modelType ) { |
|
151 _.find( this._subModels, function( subModelDef ) { |
|
152 return _.find( subModelDef.subModels, function( subModelTypeName, typeValue ) { |
|
153 var subModelType = this.getObjectByName( subModelTypeName ); |
|
154 |
|
155 if ( modelType === subModelType ) { |
|
156 // Set 'modelType' as a child of the found superModel |
|
157 subModelDef.superModelType._subModels[ typeValue ] = modelType; |
|
158 |
|
159 // Set '_superModel', '_subModelTypeValue', and '_subModelTypeAttribute' on 'modelType'. |
|
160 modelType._superModel = subModelDef.superModelType; |
|
161 modelType._subModelTypeValue = typeValue; |
|
162 modelType._subModelTypeAttribute = subModelDef.superModelType.prototype.subModelTypeAttribute; |
|
163 return true; |
|
164 } |
|
165 }, this ); |
|
166 }, this ); |
|
167 }, |
|
168 |
|
169 /** |
|
170 * Add a reverse relation. Is added to the 'relations' property on model's prototype, and to |
|
171 * existing instances of 'model' in the store as well. |
|
172 * @param {Object} relation |
|
173 * @param {Backbone.RelationalModel} relation.model |
|
174 * @param {String} relation.type |
|
175 * @param {String} relation.key |
|
176 * @param {String|Object} relation.relatedModel |
|
177 */ |
|
178 addReverseRelation: function( relation ) { |
|
179 var exists = _.any( this._reverseRelations, function( rel ) { |
|
180 return _.all( relation, function( val, key ) { |
|
181 return val === rel[ key ]; |
|
182 }); |
|
183 }); |
|
184 |
|
185 if ( !exists && relation.model && relation.type ) { |
|
186 this._reverseRelations.push( relation ); |
|
187 |
|
188 var addRelation = function( model, relation ) { |
|
189 if ( !model.prototype.relations ) { |
|
190 model.prototype.relations = []; |
|
191 } |
|
192 model.prototype.relations.push( relation ); |
|
193 |
|
194 _.each( model._subModels, function( subModel ) { |
|
195 addRelation( subModel, relation ); |
|
196 }, this ); |
|
197 }; |
|
198 |
|
199 addRelation( relation.model, relation ); |
|
200 |
|
201 this.retroFitRelation( relation ); |
|
202 } |
|
203 }, |
|
204 |
|
205 /** |
|
206 * Add a 'relation' to all existing instances of 'relation.model' in the store |
|
207 * @param {Object} relation |
|
208 */ |
|
209 retroFitRelation: function( relation ) { |
|
210 var coll = this.getCollection( relation.model ); |
|
211 coll.each( function( model ) { |
|
212 if ( !( model instanceof relation.model ) ) { |
|
213 return; |
|
214 } |
|
215 |
|
216 new relation.type( model, relation ); |
|
217 }, this); |
|
218 }, |
|
219 |
|
220 /** |
|
221 * Find the Store's collection for a certain type of model. |
|
222 * @param {Backbone.RelationalModel} model |
|
223 * @return {Backbone.Collection} A collection if found (or applicable for 'model'), or null |
|
224 */ |
|
225 getCollection: function( model ) { |
|
226 if ( model instanceof Backbone.RelationalModel ) { |
|
227 model = model.constructor; |
|
228 } |
|
229 |
|
230 var rootModel = model; |
|
231 while ( rootModel._superModel ) { |
|
232 rootModel = rootModel._superModel; |
|
233 } |
|
234 |
|
235 var coll = _.detect( this._collections, function( c ) { |
|
236 return c.model === rootModel; |
|
237 }); |
|
238 |
|
239 if ( !coll ) { |
|
240 coll = this._createCollection( rootModel ); |
|
241 } |
|
242 |
|
243 return coll; |
|
244 }, |
|
245 |
|
246 /** |
|
247 * Find a type on the global object by name. Splits name on dots. |
|
248 * @param {String} name |
|
249 * @return {Object} |
|
250 */ |
|
251 getObjectByName: function( name ) { |
|
252 var parts = name.split( '.' ), |
|
253 type = null; |
|
254 |
|
255 _.find( this._modelScopes, function( scope ) { |
|
256 type = _.reduce( parts, function( memo, val ) { |
|
257 return memo[ val ]; |
|
258 }, scope ); |
|
259 |
|
260 if ( type && type !== scope ) { |
|
261 return true; |
|
262 } |
|
263 }, this ); |
|
264 |
|
265 return type; |
|
266 }, |
|
267 |
|
268 _createCollection: function( type ) { |
|
269 var coll; |
|
270 |
|
271 // If 'type' is an instance, take its constructor |
|
272 if ( type instanceof Backbone.RelationalModel ) { |
|
273 type = type.constructor; |
|
274 } |
|
275 |
|
276 // Type should inherit from Backbone.RelationalModel. |
|
277 if ( type.prototype instanceof Backbone.RelationalModel ) { |
|
278 coll = new Backbone.Collection(); |
|
279 coll.model = type; |
|
280 |
|
281 this._collections.push( coll ); |
|
282 } |
|
283 |
|
284 return coll; |
|
285 }, |
|
286 |
|
287 /** |
|
288 * Find the attribute that is to be used as the `id` on a given object |
|
289 * @param type |
|
290 * @param {String|Number|Object|Backbone.RelationalModel} item |
|
291 * @return {String|Number} |
|
292 */ |
|
293 resolveIdForItem: function( type, item ) { |
|
294 var id = _.isString( item ) || _.isNumber( item ) ? item : null; |
|
295 |
|
296 if ( id === null ) { |
|
297 if ( item instanceof Backbone.RelationalModel ) { |
|
298 id = item.id; |
|
299 } |
|
300 else if ( _.isObject( item ) ) { |
|
301 id = item[ type.prototype.idAttribute ]; |
|
302 } |
|
303 } |
|
304 |
|
305 // Make all falsy values `null` (except for 0, which could be an id.. see '/issues/179') |
|
306 if ( !id && id !== 0 ) { |
|
307 id = null; |
|
308 } |
|
309 |
|
310 return id; |
|
311 }, |
|
312 |
|
313 /** |
|
314 * |
|
315 * @param type |
|
316 * @param {String|Number|Object|Backbone.RelationalModel} item |
|
317 */ |
|
318 find: function( type, item ) { |
|
319 var id = this.resolveIdForItem( type, item ); |
|
320 var coll = this.getCollection( type ); |
|
321 |
|
322 // Because the found object could be of any of the type's superModel |
|
323 // types, only return it if it's actually of the type asked for. |
|
324 if ( coll ) { |
|
325 var obj = coll.get( id ); |
|
326 |
|
327 if ( obj instanceof type ) { |
|
328 return obj; |
|
329 } |
|
330 } |
|
331 |
|
332 return null; |
|
333 }, |
|
334 |
|
335 /** |
|
336 * Add a 'model' to it's appropriate collection. Retain the original contents of 'model.collection'. |
|
337 * @param {Backbone.RelationalModel} model |
|
338 */ |
|
339 register: function( model ) { |
|
340 var coll = this.getCollection( model ); |
|
341 |
|
342 if ( coll ) { |
|
343 if ( coll.get( model ) ) { |
|
344 throw new Error( "Cannot instantiate more than one Backbone.RelationalModel with the same id per type!" ); |
|
345 } |
|
346 |
|
347 var modelColl = model.collection; |
|
348 coll.add( model ); |
|
349 model.bind( 'destroy', this.unregister, this ); |
|
350 model.collection = modelColl; |
|
351 } |
|
352 }, |
|
353 |
|
354 /** |
|
355 * Explicitly update a model's id in it's store collection |
|
356 * @param {Backbone.RelationalModel} model |
|
357 */ |
|
358 update: function( model ) { |
|
359 var coll = this.getCollection( model ); |
|
360 coll._onModelEvent( 'change:' + model.idAttribute, model, coll ); |
|
361 }, |
|
362 |
|
363 /** |
|
364 * Remove a 'model' from the store. |
|
365 * @param {Backbone.RelationalModel} model |
|
366 */ |
|
367 unregister: function( model ) { |
|
368 model.unbind( 'destroy', this.unregister ); |
|
369 var coll = this.getCollection( model ); |
|
370 coll && coll.remove( model ); |
|
371 } |
|
372 }); |
|
373 Backbone.Relational.store = new Backbone.Store(); |
|
374 |
|
375 /** |
|
376 * The main Relation class, from which 'HasOne' and 'HasMany' inherit. Internally, 'relational:<key>' events |
|
377 * are used to regulate addition and removal of models from relations. |
|
378 * |
|
379 * @param {Backbone.RelationalModel} instance |
|
380 * @param {Object} options |
|
381 * @param {string} options.key |
|
382 * @param {Backbone.RelationalModel.constructor} options.relatedModel |
|
383 * @param {Boolean|String} [options.includeInJSON=true] Serialize the given attribute for related model(s)' in toJSON, or just their ids. |
|
384 * @param {Boolean} [options.createModels=true] Create objects from the contents of keys if the object is not found in Backbone.store. |
|
385 * @param {Object} [options.reverseRelation] Specify a bi-directional relation. If provided, Relation will reciprocate |
|
386 * the relation to the 'relatedModel'. Required and optional properties match 'options', except that it also needs |
|
387 * {Backbone.Relation|String} type ('HasOne' or 'HasMany'). |
|
388 */ |
|
389 Backbone.Relation = function( instance, options ) { |
|
390 this.instance = instance; |
|
391 // Make sure 'options' is sane, and fill with defaults from subclasses and this object's prototype |
|
392 options = _.isObject( options ) ? options : {}; |
|
393 this.reverseRelation = _.defaults( options.reverseRelation || {}, this.options.reverseRelation ); |
|
394 this.reverseRelation.type = !_.isString( this.reverseRelation.type ) ? this.reverseRelation.type : |
|
395 Backbone[ this.reverseRelation.type ] || Backbone.Relational.store.getObjectByName( this.reverseRelation.type ); |
|
396 this.model = options.model || this.instance.constructor; |
|
397 this.options = _.defaults( options, this.options, Backbone.Relation.prototype.options ); |
|
398 |
|
399 this.key = this.options.key; |
|
400 this.keySource = this.options.keySource || this.key; |
|
401 this.keyDestination = this.options.keyDestination || this.keySource || this.key; |
|
402 |
|
403 // 'exports' should be the global object where 'relatedModel' can be found on if given as a string. |
|
404 this.relatedModel = this.options.relatedModel; |
|
405 if ( _.isString( this.relatedModel ) ) { |
|
406 this.relatedModel = Backbone.Relational.store.getObjectByName( this.relatedModel ); |
|
407 } |
|
408 |
|
409 if ( !this.checkPreconditions() ) { |
|
410 return false; |
|
411 } |
|
412 |
|
413 if ( instance ) { |
|
414 this.keyContents = this.instance.get( this.keySource ); |
|
415 |
|
416 // Explicitly clear 'keySource', to prevent a leaky abstraction if 'keySource' differs from 'key'. |
|
417 if ( this.key !== this.keySource ) { |
|
418 this.instance.unset( this.keySource, { silent: true } ); |
|
419 } |
|
420 |
|
421 // Add this Relation to instance._relations |
|
422 this.instance._relations.push( this ); |
|
423 } |
|
424 |
|
425 // Add the reverse relation on 'relatedModel' to the store's reverseRelations |
|
426 if ( !this.options.isAutoRelation && this.reverseRelation.type && this.reverseRelation.key ) { |
|
427 Backbone.Relational.store.addReverseRelation( _.defaults( { |
|
428 isAutoRelation: true, |
|
429 model: this.relatedModel, |
|
430 relatedModel: this.model, |
|
431 reverseRelation: this.options // current relation is the 'reverseRelation' for it's own reverseRelation |
|
432 }, |
|
433 this.reverseRelation // Take further properties from this.reverseRelation (type, key, etc.) |
|
434 ) ); |
|
435 } |
|
436 |
|
437 _.bindAll( this, '_modelRemovedFromCollection', '_relatedModelAdded', '_relatedModelRemoved' ); |
|
438 |
|
439 if ( instance ) { |
|
440 this.initialize(); |
|
441 |
|
442 // When a model in the store is destroyed, check if it is 'this.instance'. |
|
443 Backbone.Relational.store.getCollection( this.instance ) |
|
444 .bind( 'relational:remove', this._modelRemovedFromCollection ); |
|
445 |
|
446 // When 'relatedModel' are created or destroyed, check if it affects this relation. |
|
447 Backbone.Relational.store.getCollection( this.relatedModel ) |
|
448 .bind( 'relational:add', this._relatedModelAdded ) |
|
449 .bind( 'relational:remove', this._relatedModelRemoved ); |
|
450 } |
|
451 }; |
|
452 // Fix inheritance :\ |
|
453 Backbone.Relation.extend = Backbone.Model.extend; |
|
454 // Set up all inheritable **Backbone.Relation** properties and methods. |
|
455 _.extend( Backbone.Relation.prototype, Backbone.Events, Backbone.Semaphore, { |
|
456 options: { |
|
457 createModels: true, |
|
458 includeInJSON: true, |
|
459 isAutoRelation: false |
|
460 }, |
|
461 |
|
462 instance: null, |
|
463 key: null, |
|
464 keyContents: null, |
|
465 relatedModel: null, |
|
466 reverseRelation: null, |
|
467 related: null, |
|
468 |
|
469 _relatedModelAdded: function( model, coll, options ) { |
|
470 // Allow 'model' to set up it's relations, before calling 'tryAddRelated' |
|
471 // (which can result in a call to 'addRelated' on a relation of 'model') |
|
472 var dit = this; |
|
473 model.queue( function() { |
|
474 dit.tryAddRelated( model, options ); |
|
475 }); |
|
476 }, |
|
477 |
|
478 _relatedModelRemoved: function( model, coll, options ) { |
|
479 this.removeRelated( model, options ); |
|
480 }, |
|
481 |
|
482 _modelRemovedFromCollection: function( model ) { |
|
483 if ( model === this.instance ) { |
|
484 this.destroy(); |
|
485 } |
|
486 }, |
|
487 |
|
488 /** |
|
489 * Check several pre-conditions. |
|
490 * @return {Boolean} True if pre-conditions are satisfied, false if they're not. |
|
491 */ |
|
492 checkPreconditions: function() { |
|
493 var i = this.instance, |
|
494 k = this.key, |
|
495 m = this.model, |
|
496 rm = this.relatedModel, |
|
497 warn = Backbone.Relational.showWarnings && typeof console !== 'undefined'; |
|
498 |
|
499 if ( !m || !k || !rm ) { |
|
500 warn && console.warn( 'Relation=%o; no model, key or relatedModel (%o, %o, %o)', this, m, k, rm ); |
|
501 return false; |
|
502 } |
|
503 // Check if the type in 'model' inherits from Backbone.RelationalModel |
|
504 if ( !( m.prototype instanceof Backbone.RelationalModel ) ) { |
|
505 warn && console.warn( 'Relation=%o; model does not inherit from Backbone.RelationalModel (%o)', this, i ); |
|
506 return false; |
|
507 } |
|
508 // Check if the type in 'relatedModel' inherits from Backbone.RelationalModel |
|
509 if ( !( rm.prototype instanceof Backbone.RelationalModel ) ) { |
|
510 warn && console.warn( 'Relation=%o; relatedModel does not inherit from Backbone.RelationalModel (%o)', this, rm ); |
|
511 return false; |
|
512 } |
|
513 // Check if this is not a HasMany, and the reverse relation is HasMany as well |
|
514 if ( this instanceof Backbone.HasMany && this.reverseRelation.type === Backbone.HasMany ) { |
|
515 warn && console.warn( 'Relation=%o; relation is a HasMany, and the reverseRelation is HasMany as well.', this ); |
|
516 return false; |
|
517 } |
|
518 |
|
519 // Check if we're not attempting to create a duplicate relationship |
|
520 if ( i && i._relations.length ) { |
|
521 var exists = _.any( i._relations, function( rel ) { |
|
522 var hasReverseRelation = this.reverseRelation.key && rel.reverseRelation.key; |
|
523 return rel.relatedModel === rm && rel.key === k && |
|
524 ( !hasReverseRelation || this.reverseRelation.key === rel.reverseRelation.key ); |
|
525 }, this ); |
|
526 |
|
527 if ( exists ) { |
|
528 warn && console.warn( 'Relation=%o between instance=%o.%s and relatedModel=%o.%s already exists', |
|
529 this, i, k, rm, this.reverseRelation.key ); |
|
530 return false; |
|
531 } |
|
532 } |
|
533 |
|
534 return true; |
|
535 }, |
|
536 |
|
537 /** |
|
538 * Set the related model(s) for this relation |
|
539 * @param {Backbone.Mode|Backbone.Collection} related |
|
540 * @param {Object} [options] |
|
541 */ |
|
542 setRelated: function( related, options ) { |
|
543 this.related = related; |
|
544 |
|
545 this.instance.acquire(); |
|
546 this.instance.set( this.key, related, _.defaults( options || {}, { silent: true } ) ); |
|
547 this.instance.release(); |
|
548 }, |
|
549 |
|
550 /** |
|
551 * Determine if a relation (on a different RelationalModel) is the reverse |
|
552 * relation of the current one. |
|
553 * @param {Backbone.Relation} relation |
|
554 * @return {Boolean} |
|
555 */ |
|
556 _isReverseRelation: function( relation ) { |
|
557 if ( relation.instance instanceof this.relatedModel && this.reverseRelation.key === relation.key && |
|
558 this.key === relation.reverseRelation.key ) { |
|
559 return true; |
|
560 } |
|
561 return false; |
|
562 }, |
|
563 |
|
564 /** |
|
565 * Get the reverse relations (pointing back to 'this.key' on 'this.instance') for the currently related model(s). |
|
566 * @param {Backbone.RelationalModel} [model] Get the reverse relations for a specific model. |
|
567 * If not specified, 'this.related' is used. |
|
568 * @return {Backbone.Relation[]} |
|
569 */ |
|
570 getReverseRelations: function( model ) { |
|
571 var reverseRelations = []; |
|
572 // Iterate over 'model', 'this.related.models' (if this.related is a Backbone.Collection), or wrap 'this.related' in an array. |
|
573 var models = !_.isUndefined( model ) ? [ model ] : this.related && ( this.related.models || [ this.related ] ); |
|
574 _.each( models , function( related ) { |
|
575 _.each( related.getRelations(), function( relation ) { |
|
576 if ( this._isReverseRelation( relation ) ) { |
|
577 reverseRelations.push( relation ); |
|
578 } |
|
579 }, this ); |
|
580 }, this ); |
|
581 |
|
582 return reverseRelations; |
|
583 }, |
|
584 |
|
585 /** |
|
586 * Rename options.silent to options.silentChange, so events propagate properly. |
|
587 * (for example in HasMany, from 'addRelated'->'handleAddition') |
|
588 * @param {Object} [options] |
|
589 * @return {Object} |
|
590 */ |
|
591 sanitizeOptions: function( options ) { |
|
592 options = options ? _.clone( options ) : {}; |
|
593 if ( options.silent ) { |
|
594 options.silentChange = true; |
|
595 delete options.silent; |
|
596 } |
|
597 return options; |
|
598 }, |
|
599 |
|
600 /** |
|
601 * Rename options.silentChange to options.silent, so events are silenced as intended in Backbone's |
|
602 * original functions. |
|
603 * @param {Object} [options] |
|
604 * @return {Object} |
|
605 */ |
|
606 unsanitizeOptions: function( options ) { |
|
607 options = options ? _.clone( options ) : {}; |
|
608 if ( options.silentChange ) { |
|
609 options.silent = true; |
|
610 delete options.silentChange; |
|
611 } |
|
612 return options; |
|
613 }, |
|
614 |
|
615 // Cleanup. Get reverse relation, call removeRelated on each. |
|
616 destroy: function() { |
|
617 Backbone.Relational.store.getCollection( this.instance ) |
|
618 .unbind( 'relational:remove', this._modelRemovedFromCollection ); |
|
619 |
|
620 Backbone.Relational.store.getCollection( this.relatedModel ) |
|
621 .unbind( 'relational:add', this._relatedModelAdded ) |
|
622 .unbind( 'relational:remove', this._relatedModelRemoved ); |
|
623 |
|
624 _.each( this.getReverseRelations(), function( relation ) { |
|
625 relation.removeRelated( this.instance ); |
|
626 }, this ); |
|
627 } |
|
628 }); |
|
629 |
|
630 Backbone.HasOne = Backbone.Relation.extend({ |
|
631 options: { |
|
632 reverseRelation: { type: 'HasMany' } |
|
633 }, |
|
634 |
|
635 initialize: function() { |
|
636 _.bindAll( this, 'onChange' ); |
|
637 |
|
638 this.instance.bind( 'relational:change:' + this.key, this.onChange ); |
|
639 |
|
640 var model = this.findRelated( { silent: true } ); |
|
641 this.setRelated( model ); |
|
642 |
|
643 // Notify new 'related' object of the new relation. |
|
644 _.each( this.getReverseRelations(), function( relation ) { |
|
645 relation.addRelated( this.instance ); |
|
646 }, this ); |
|
647 }, |
|
648 |
|
649 findRelated: function( options ) { |
|
650 var item = this.keyContents; |
|
651 var model = null; |
|
652 |
|
653 if ( item instanceof this.relatedModel ) { |
|
654 model = item; |
|
655 } |
|
656 else if ( item || item === 0 ) { // since 0 can be a valid `id` as well |
|
657 model = this.relatedModel.findOrCreate( item, { create: this.options.createModels } ); |
|
658 } |
|
659 |
|
660 return model; |
|
661 }, |
|
662 |
|
663 /** |
|
664 * If the key is changed, notify old & new reverse relations and initialize the new relation |
|
665 */ |
|
666 onChange: function( model, attr, options ) { |
|
667 // Don't accept recursive calls to onChange (like onChange->findRelated->findOrCreate->initializeRelations->addRelated->onChange) |
|
668 if ( this.isLocked() ) { |
|
669 return; |
|
670 } |
|
671 this.acquire(); |
|
672 options = this.sanitizeOptions( options ); |
|
673 |
|
674 // 'options._related' is set by 'addRelated'/'removeRelated'. If it is set, the change |
|
675 // is the result of a call from a relation. If it's not, the change is the result of |
|
676 // a 'set' call on this.instance. |
|
677 var changed = _.isUndefined( options._related ); |
|
678 var oldRelated = changed ? this.related : options._related; |
|
679 |
|
680 if ( changed ) { |
|
681 this.keyContents = attr; |
|
682 |
|
683 // Set new 'related' |
|
684 if ( attr instanceof this.relatedModel ) { |
|
685 this.related = attr; |
|
686 } |
|
687 else if ( attr ) { |
|
688 var related = this.findRelated( options ); |
|
689 this.setRelated( related ); |
|
690 } |
|
691 else { |
|
692 this.setRelated( null ); |
|
693 } |
|
694 } |
|
695 |
|
696 // Notify old 'related' object of the terminated relation |
|
697 if ( oldRelated && this.related !== oldRelated ) { |
|
698 _.each( this.getReverseRelations( oldRelated ), function( relation ) { |
|
699 relation.removeRelated( this.instance, options ); |
|
700 }, this ); |
|
701 } |
|
702 |
|
703 // Notify new 'related' object of the new relation. Note we do re-apply even if this.related is oldRelated; |
|
704 // that can be necessary for bi-directional relations if 'this.instance' was created after 'this.related'. |
|
705 // In that case, 'this.instance' will already know 'this.related', but the reverse might not exist yet. |
|
706 _.each( this.getReverseRelations(), function( relation ) { |
|
707 relation.addRelated( this.instance, options ); |
|
708 }, this); |
|
709 |
|
710 // Fire the 'update:<key>' event if 'related' was updated |
|
711 if ( !options.silentChange && this.related !== oldRelated ) { |
|
712 var dit = this; |
|
713 Backbone.Relational.eventQueue.add( function() { |
|
714 dit.instance.trigger( 'update:' + dit.key, dit.instance, dit.related, options ); |
|
715 }); |
|
716 } |
|
717 this.release(); |
|
718 }, |
|
719 |
|
720 /** |
|
721 * If a new 'this.relatedModel' appears in the 'store', try to match it to the last set 'keyContents' |
|
722 */ |
|
723 tryAddRelated: function( model, options ) { |
|
724 if ( this.related ) { |
|
725 return; |
|
726 } |
|
727 options = this.sanitizeOptions( options ); |
|
728 |
|
729 var item = this.keyContents; |
|
730 if ( item || item === 0 ) { // since 0 can be a valid `id` as well |
|
731 var id = Backbone.Relational.store.resolveIdForItem( this.relatedModel, item ); |
|
732 if ( !_.isNull( id ) && model.id === id ) { |
|
733 this.addRelated( model, options ); |
|
734 } |
|
735 } |
|
736 }, |
|
737 |
|
738 addRelated: function( model, options ) { |
|
739 if ( model !== this.related ) { |
|
740 var oldRelated = this.related || null; |
|
741 this.setRelated( model ); |
|
742 this.onChange( this.instance, model, { _related: oldRelated } ); |
|
743 } |
|
744 }, |
|
745 |
|
746 removeRelated: function( model, options ) { |
|
747 if ( !this.related ) { |
|
748 return; |
|
749 } |
|
750 |
|
751 if ( model === this.related ) { |
|
752 var oldRelated = this.related || null; |
|
753 this.setRelated( null ); |
|
754 this.onChange( this.instance, model, { _related: oldRelated } ); |
|
755 } |
|
756 } |
|
757 }); |
|
758 |
|
759 Backbone.HasMany = Backbone.Relation.extend({ |
|
760 collectionType: null, |
|
761 |
|
762 options: { |
|
763 reverseRelation: { type: 'HasOne' }, |
|
764 collectionType: Backbone.Collection, |
|
765 collectionKey: true, |
|
766 collectionOptions: {} |
|
767 }, |
|
768 |
|
769 initialize: function() { |
|
770 _.bindAll( this, 'onChange', 'handleAddition', 'handleRemoval', 'handleReset' ); |
|
771 this.instance.bind( 'relational:change:' + this.key, this.onChange ); |
|
772 |
|
773 // Handle a custom 'collectionType' |
|
774 this.collectionType = this.options.collectionType; |
|
775 if ( _.isString( this.collectionType ) ) { |
|
776 this.collectionType = Backbone.Relational.store.getObjectByName( this.collectionType ); |
|
777 } |
|
778 if ( !this.collectionType.prototype instanceof Backbone.Collection ){ |
|
779 throw new Error( 'collectionType must inherit from Backbone.Collection' ); |
|
780 } |
|
781 |
|
782 // Handle cases where a model/relation is created with a collection passed straight into 'attributes' |
|
783 if ( this.keyContents instanceof Backbone.Collection ) { |
|
784 this.setRelated( this._prepareCollection( this.keyContents ) ); |
|
785 } |
|
786 else { |
|
787 this.setRelated( this._prepareCollection() ); |
|
788 } |
|
789 |
|
790 this.findRelated( { silent: true } ); |
|
791 }, |
|
792 |
|
793 _getCollectionOptions: function() { |
|
794 return _.isFunction( this.options.collectionOptions ) ? |
|
795 this.options.collectionOptions( this.instance ) : |
|
796 this.options.collectionOptions; |
|
797 }, |
|
798 |
|
799 /** |
|
800 * Bind events and setup collectionKeys for a collection that is to be used as the backing store for a HasMany. |
|
801 * If no 'collection' is supplied, a new collection will be created of the specified 'collectionType' option. |
|
802 * @param {Backbone.Collection} [collection] |
|
803 */ |
|
804 _prepareCollection: function( collection ) { |
|
805 if ( this.related ) { |
|
806 this.related |
|
807 .unbind( 'relational:add', this.handleAddition ) |
|
808 .unbind( 'relational:remove', this.handleRemoval ) |
|
809 .unbind( 'relational:reset', this.handleReset ) |
|
810 } |
|
811 |
|
812 if ( !collection || !( collection instanceof Backbone.Collection ) ) { |
|
813 collection = new this.collectionType( [], this._getCollectionOptions() ); |
|
814 } |
|
815 |
|
816 collection.model = this.relatedModel; |
|
817 |
|
818 if ( this.options.collectionKey ) { |
|
819 var key = this.options.collectionKey === true ? this.options.reverseRelation.key : this.options.collectionKey; |
|
820 |
|
821 if ( collection[ key ] && collection[ key ] !== this.instance ) { |
|
822 if ( Backbone.Relational.showWarnings && typeof console !== 'undefined' ) { |
|
823 console.warn( 'Relation=%o; collectionKey=%s already exists on collection=%o', this, key, this.options.collectionKey ); |
|
824 } |
|
825 } |
|
826 else if ( key ) { |
|
827 collection[ key ] = this.instance; |
|
828 } |
|
829 } |
|
830 |
|
831 collection |
|
832 .bind( 'relational:add', this.handleAddition ) |
|
833 .bind( 'relational:remove', this.handleRemoval ) |
|
834 .bind( 'relational:reset', this.handleReset ); |
|
835 |
|
836 return collection; |
|
837 }, |
|
838 |
|
839 findRelated: function( options ) { |
|
840 if ( this.keyContents ) { |
|
841 var models = []; |
|
842 |
|
843 if ( this.keyContents instanceof Backbone.Collection ) { |
|
844 models = this.keyContents.models; |
|
845 } |
|
846 else { |
|
847 // Handle cases the an API/user supplies just an Object/id instead of an Array |
|
848 this.keyContents = _.isArray( this.keyContents ) ? this.keyContents : [ this.keyContents ]; |
|
849 |
|
850 // Try to find instances of the appropriate 'relatedModel' in the store |
|
851 _.each( this.keyContents, function( item ) { |
|
852 var model = null; |
|
853 if ( item instanceof this.relatedModel ) { |
|
854 model = item; |
|
855 } |
|
856 else if ( item || item === 0 ) { // since 0 can be a valid `id` as well |
|
857 model = this.relatedModel.findOrCreate( item, { create: this.options.createModels } ); |
|
858 } |
|
859 |
|
860 if ( model && !this.related.getByCid( model ) && !this.related.get( model ) ) { |
|
861 models.push( model ); |
|
862 } |
|
863 }, this ); |
|
864 } |
|
865 |
|
866 // Add all found 'models' in on go, so 'add' will only be called once (and thus 'sort', etc.) |
|
867 if ( models.length ) { |
|
868 options = this.unsanitizeOptions( options ); |
|
869 this.related.add( models, options ); |
|
870 } |
|
871 } |
|
872 }, |
|
873 |
|
874 /** |
|
875 * If the key is changed, notify old & new reverse relations and initialize the new relation |
|
876 */ |
|
877 onChange: function( model, attr, options ) { |
|
878 options = this.sanitizeOptions( options ); |
|
879 this.keyContents = attr; |
|
880 |
|
881 // Notify old 'related' object of the terminated relation |
|
882 _.each( this.getReverseRelations(), function( relation ) { |
|
883 relation.removeRelated( this.instance, options ); |
|
884 }, this ); |
|
885 |
|
886 // Replace 'this.related' by 'attr' if it is a Backbone.Collection |
|
887 if ( attr instanceof Backbone.Collection ) { |
|
888 this._prepareCollection( attr ); |
|
889 this.related = attr; |
|
890 } |
|
891 // Otherwise, 'attr' should be an array of related object ids. |
|
892 // Re-use the current 'this.related' if it is a Backbone.Collection, and remove any current entries. |
|
893 // Otherwise, create a new collection. |
|
894 else { |
|
895 var coll; |
|
896 |
|
897 if ( this.related instanceof Backbone.Collection ) { |
|
898 coll = this.related; |
|
899 coll.remove( coll.models ); |
|
900 } |
|
901 else { |
|
902 coll = this._prepareCollection(); |
|
903 } |
|
904 |
|
905 this.setRelated( coll ); |
|
906 this.findRelated( options ); |
|
907 } |
|
908 |
|
909 // Notify new 'related' object of the new relation |
|
910 _.each( this.getReverseRelations(), function( relation ) { |
|
911 relation.addRelated( this.instance, options ); |
|
912 }, this ); |
|
913 |
|
914 var dit = this; |
|
915 Backbone.Relational.eventQueue.add( function() { |
|
916 !options.silentChange && dit.instance.trigger( 'update:' + dit.key, dit.instance, dit.related, options ); |
|
917 }); |
|
918 }, |
|
919 |
|
920 tryAddRelated: function( model, options ) { |
|
921 options = this.sanitizeOptions( options ); |
|
922 if ( !this.related.getByCid( model ) && !this.related.get( model ) ) { |
|
923 // Check if this new model was specified in 'this.keyContents' |
|
924 var item = _.any( this.keyContents, function( item ) { |
|
925 var id = Backbone.Relational.store.resolveIdForItem( this.relatedModel, item ); |
|
926 return !_.isNull( id ) && id === model.id; |
|
927 }, this ); |
|
928 |
|
929 if ( item ) { |
|
930 this.related.add( model, options ); |
|
931 } |
|
932 } |
|
933 }, |
|
934 |
|
935 /** |
|
936 * When a model is added to a 'HasMany', trigger 'add' on 'this.instance' and notify reverse relations. |
|
937 * (should be 'HasOne', must set 'this.instance' as their related). |
|
938 */ |
|
939 handleAddition: function( model, coll, options ) { |
|
940 //console.debug('handleAddition called; args=%o', arguments); |
|
941 // Make sure the model is in fact a valid model before continuing. |
|
942 // (it can be invalid as a result of failing validation in Backbone.Collection._prepareModel) |
|
943 if ( !( model instanceof Backbone.Model ) ) { |
|
944 return; |
|
945 } |
|
946 |
|
947 options = this.sanitizeOptions( options ); |
|
948 |
|
949 _.each( this.getReverseRelations( model ), function( relation ) { |
|
950 relation.addRelated( this.instance, options ); |
|
951 }, this ); |
|
952 |
|
953 // Only trigger 'add' once the newly added model is initialized (so, has it's relations set up) |
|
954 var dit = this; |
|
955 Backbone.Relational.eventQueue.add( function() { |
|
956 !options.silentChange && dit.instance.trigger( 'add:' + dit.key, model, dit.related, options ); |
|
957 }); |
|
958 }, |
|
959 |
|
960 /** |
|
961 * When a model is removed from a 'HasMany', trigger 'remove' on 'this.instance' and notify reverse relations. |
|
962 * (should be 'HasOne', which should be nullified) |
|
963 */ |
|
964 handleRemoval: function( model, coll, options ) { |
|
965 //console.debug('handleRemoval called; args=%o', arguments); |
|
966 if ( !( model instanceof Backbone.Model ) ) { |
|
967 return; |
|
968 } |
|
969 |
|
970 options = this.sanitizeOptions( options ); |
|
971 |
|
972 _.each( this.getReverseRelations( model ), function( relation ) { |
|
973 relation.removeRelated( this.instance, options ); |
|
974 }, this ); |
|
975 |
|
976 var dit = this; |
|
977 Backbone.Relational.eventQueue.add( function() { |
|
978 !options.silentChange && dit.instance.trigger( 'remove:' + dit.key, model, dit.related, options ); |
|
979 }); |
|
980 }, |
|
981 |
|
982 handleReset: function( coll, options ) { |
|
983 options = this.sanitizeOptions( options ); |
|
984 |
|
985 var dit = this; |
|
986 Backbone.Relational.eventQueue.add( function() { |
|
987 !options.silentChange && dit.instance.trigger( 'reset:' + dit.key, dit.related, options ); |
|
988 }); |
|
989 }, |
|
990 |
|
991 addRelated: function( model, options ) { |
|
992 var dit = this; |
|
993 options = this.unsanitizeOptions( options ); |
|
994 model.queue( function() { // Queued to avoid errors for adding 'model' to the 'this.related' set twice |
|
995 if ( dit.related && !dit.related.getByCid( model ) && !dit.related.get( model ) ) { |
|
996 dit.related.add( model, options ); |
|
997 } |
|
998 }); |
|
999 }, |
|
1000 |
|
1001 removeRelated: function( model, options ) { |
|
1002 options = this.unsanitizeOptions( options ); |
|
1003 if ( this.related.getByCid( model ) || this.related.get( model ) ) { |
|
1004 this.related.remove( model, options ); |
|
1005 } |
|
1006 } |
|
1007 }); |
|
1008 |
|
1009 /** |
|
1010 * A type of Backbone.Model that also maintains relations to other models and collections. |
|
1011 * New events when compared to the original: |
|
1012 * - 'add:<key>' (model, related collection, options) |
|
1013 * - 'remove:<key>' (model, related collection, options) |
|
1014 * - 'update:<key>' (model, related model or collection, options) |
|
1015 */ |
|
1016 Backbone.RelationalModel = Backbone.Model.extend({ |
|
1017 relations: null, // Relation descriptions on the prototype |
|
1018 _relations: null, // Relation instances |
|
1019 _isInitialized: false, |
|
1020 _deferProcessing: false, |
|
1021 _queue: null, |
|
1022 |
|
1023 subModelTypeAttribute: 'type', |
|
1024 subModelTypes: null, |
|
1025 |
|
1026 constructor: function( attributes, options ) { |
|
1027 // Nasty hack, for cases like 'model.get( <HasMany key> ).add( item )'. |
|
1028 // Defer 'processQueue', so that when 'Relation.createModels' is used we: |
|
1029 // a) Survive 'Backbone.Collection.add'; this takes care we won't error on "can't add model to a set twice" |
|
1030 // (by creating a model from properties, having the model add itself to the collection via one of |
|
1031 // it's relations, then trying to add it to the collection). |
|
1032 // b) Trigger 'HasMany' collection events only after the model is really fully set up. |
|
1033 // Example that triggers both a and b: "p.get('jobs').add( { company: c, person: p } )". |
|
1034 var dit = this; |
|
1035 if ( options && options.collection ) { |
|
1036 this._deferProcessing = true; |
|
1037 |
|
1038 var processQueue = function( model ) { |
|
1039 if ( model === dit ) { |
|
1040 dit._deferProcessing = false; |
|
1041 dit.processQueue(); |
|
1042 options.collection.unbind( 'relational:add', processQueue ); |
|
1043 } |
|
1044 }; |
|
1045 options.collection.bind( 'relational:add', processQueue ); |
|
1046 |
|
1047 // So we do process the queue eventually, regardless of whether this model really gets added to 'options.collection'. |
|
1048 _.defer( function() { |
|
1049 processQueue( dit ); |
|
1050 }); |
|
1051 } |
|
1052 |
|
1053 this._queue = new Backbone.BlockingQueue(); |
|
1054 this._queue.block(); |
|
1055 Backbone.Relational.eventQueue.block(); |
|
1056 |
|
1057 Backbone.Model.apply( this, arguments ); |
|
1058 |
|
1059 // Try to run the global queue holding external events |
|
1060 Backbone.Relational.eventQueue.unblock(); |
|
1061 }, |
|
1062 |
|
1063 /** |
|
1064 * Override 'trigger' to queue 'change' and 'change:*' events |
|
1065 */ |
|
1066 trigger: function( eventName ) { |
|
1067 if ( eventName.length > 5 && 'change' === eventName.substr( 0, 6 ) ) { |
|
1068 var dit = this, args = arguments; |
|
1069 Backbone.Relational.eventQueue.add( function() { |
|
1070 Backbone.Model.prototype.trigger.apply( dit, args ); |
|
1071 }); |
|
1072 } |
|
1073 else { |
|
1074 Backbone.Model.prototype.trigger.apply( this, arguments ); |
|
1075 } |
|
1076 |
|
1077 return this; |
|
1078 }, |
|
1079 |
|
1080 /** |
|
1081 * Initialize Relations present in this.relations; determine the type (HasOne/HasMany), then creates a new instance. |
|
1082 * Invoked in the first call so 'set' (which is made from the Backbone.Model constructor). |
|
1083 */ |
|
1084 initializeRelations: function() { |
|
1085 this.acquire(); // Setting up relations often also involve calls to 'set', and we only want to enter this function once |
|
1086 this._relations = []; |
|
1087 |
|
1088 _.each( this.relations, function( rel ) { |
|
1089 var type = !_.isString( rel.type ) ? rel.type : Backbone[ rel.type ] || Backbone.Relational.store.getObjectByName( rel.type ); |
|
1090 if ( type && type.prototype instanceof Backbone.Relation ) { |
|
1091 new type( this, rel ); // Also pushes the new Relation into _relations |
|
1092 } |
|
1093 else { |
|
1094 Backbone.Relational.showWarnings && typeof console !== 'undefined' && console.warn( 'Relation=%o; missing or invalid type!', rel ); |
|
1095 } |
|
1096 }, this ); |
|
1097 |
|
1098 this._isInitialized = true; |
|
1099 this.release(); |
|
1100 this.processQueue(); |
|
1101 }, |
|
1102 |
|
1103 /** |
|
1104 * When new values are set, notify this model's relations (also if options.silent is set). |
|
1105 * (Relation.setRelated locks this model before calling 'set' on it to prevent loops) |
|
1106 */ |
|
1107 updateRelations: function( options ) { |
|
1108 if ( this._isInitialized && !this.isLocked() ) { |
|
1109 _.each( this._relations, function( rel ) { |
|
1110 // Update from data in `rel.keySource` if set, or `rel.key` otherwise |
|
1111 var val = this.attributes[ rel.keySource ] || this.attributes[ rel.key ]; |
|
1112 if ( rel.related !== val ) { |
|
1113 this.trigger( 'relational:change:' + rel.key, this, val, options || {} ); |
|
1114 } |
|
1115 }, this ); |
|
1116 } |
|
1117 }, |
|
1118 |
|
1119 /** |
|
1120 * Either add to the queue (if we're not initialized yet), or execute right away. |
|
1121 */ |
|
1122 queue: function( func ) { |
|
1123 this._queue.add( func ); |
|
1124 }, |
|
1125 |
|
1126 /** |
|
1127 * Process _queue |
|
1128 */ |
|
1129 processQueue: function() { |
|
1130 if ( this._isInitialized && !this._deferProcessing && this._queue.isBlocked() ) { |
|
1131 this._queue.unblock(); |
|
1132 } |
|
1133 }, |
|
1134 |
|
1135 /** |
|
1136 * Get a specific relation. |
|
1137 * @param key {string} The relation key to look for. |
|
1138 * @return {Backbone.Relation} An instance of 'Backbone.Relation', if a relation was found for 'key', or null. |
|
1139 */ |
|
1140 getRelation: function( key ) { |
|
1141 return _.detect( this._relations, function( rel ) { |
|
1142 if ( rel.key === key ) { |
|
1143 return true; |
|
1144 } |
|
1145 }, this ); |
|
1146 }, |
|
1147 |
|
1148 /** |
|
1149 * Get all of the created relations. |
|
1150 * @return {Backbone.Relation[]} |
|
1151 */ |
|
1152 getRelations: function() { |
|
1153 return this._relations; |
|
1154 }, |
|
1155 |
|
1156 /** |
|
1157 * Retrieve related objects. |
|
1158 * @param key {string} The relation key to fetch models for. |
|
1159 * @param [options] {Object} Options for 'Backbone.Model.fetch' and 'Backbone.sync'. |
|
1160 * @param [update=false] {boolean} Whether to force a fetch from the server (updating existing models). |
|
1161 * @return {jQuery.when[]} An array of request objects |
|
1162 */ |
|
1163 fetchRelated: function( key, options, update ) { |
|
1164 options || ( options = {} ); |
|
1165 var setUrl, |
|
1166 requests = [], |
|
1167 rel = this.getRelation( key ), |
|
1168 keyContents = rel && rel.keyContents, |
|
1169 toFetch = keyContents && _.select( _.isArray( keyContents ) ? keyContents : [ keyContents ], function( item ) { |
|
1170 var id = Backbone.Relational.store.resolveIdForItem( rel.relatedModel, item ); |
|
1171 return !_.isNull( id ) && ( update || !Backbone.Relational.store.find( rel.relatedModel, id ) ); |
|
1172 }, this ); |
|
1173 |
|
1174 if ( toFetch && toFetch.length ) { |
|
1175 // Create a model for each entry in 'keyContents' that is to be fetched |
|
1176 var models = _.map( toFetch, function( item ) { |
|
1177 var model; |
|
1178 |
|
1179 if ( _.isObject( item ) ) { |
|
1180 model = rel.relatedModel.build( item ); |
|
1181 } |
|
1182 else { |
|
1183 var attrs = {}; |
|
1184 attrs[ rel.relatedModel.prototype.idAttribute ] = item; |
|
1185 model = rel.relatedModel.build( attrs ); |
|
1186 } |
|
1187 |
|
1188 return model; |
|
1189 }, this ); |
|
1190 |
|
1191 // Try if the 'collection' can provide a url to fetch a set of models in one request. |
|
1192 if ( rel.related instanceof Backbone.Collection && _.isFunction( rel.related.url ) ) { |
|
1193 setUrl = rel.related.url( models ); |
|
1194 } |
|
1195 |
|
1196 // An assumption is that when 'Backbone.Collection.url' is a function, it can handle building of set urls. |
|
1197 // To make sure it can, test if the url we got by supplying a list of models to fetch is different from |
|
1198 // the one supplied for the default fetch action (without args to 'url'). |
|
1199 if ( setUrl && setUrl !== rel.related.url() ) { |
|
1200 var opts = _.defaults( |
|
1201 { |
|
1202 error: function() { |
|
1203 var args = arguments; |
|
1204 _.each( models, function( model ) { |
|
1205 model.trigger( 'destroy', model, model.collection, options ); |
|
1206 options.error && options.error.apply( model, args ); |
|
1207 }); |
|
1208 }, |
|
1209 url: setUrl |
|
1210 }, |
|
1211 options, |
|
1212 { add: true } |
|
1213 ); |
|
1214 |
|
1215 requests = [ rel.related.fetch( opts ) ]; |
|
1216 } |
|
1217 else { |
|
1218 requests = _.map( models, function( model ) { |
|
1219 var opts = _.defaults( |
|
1220 { |
|
1221 error: function() { |
|
1222 model.trigger( 'destroy', model, model.collection, options ); |
|
1223 options.error && options.error.apply( model, arguments ); |
|
1224 } |
|
1225 }, |
|
1226 options |
|
1227 ); |
|
1228 return model.fetch( opts ); |
|
1229 }, this ); |
|
1230 } |
|
1231 } |
|
1232 |
|
1233 return requests; |
|
1234 }, |
|
1235 |
|
1236 set: function( key, value, options ) { |
|
1237 Backbone.Relational.eventQueue.block(); |
|
1238 |
|
1239 // Duplicate backbone's behavior to allow separate key/value parameters, instead of a single 'attributes' object |
|
1240 var attributes; |
|
1241 if ( _.isObject( key ) || key == null ) { |
|
1242 attributes = key; |
|
1243 options = value; |
|
1244 } |
|
1245 else { |
|
1246 attributes = {}; |
|
1247 attributes[ key ] = value; |
|
1248 } |
|
1249 |
|
1250 var result = Backbone.Model.prototype.set.apply( this, arguments ); |
|
1251 |
|
1252 // Ideal place to set up relations :) |
|
1253 if ( !this._isInitialized && !this.isLocked() ) { |
|
1254 this.constructor.initializeModelHierarchy(); |
|
1255 |
|
1256 Backbone.Relational.store.register( this ); |
|
1257 |
|
1258 this.initializeRelations(); |
|
1259 } |
|
1260 // Update the 'idAttribute' in Backbone.store if; we don't want it to miss an 'id' update due to {silent:true} |
|
1261 else if ( attributes && this.idAttribute in attributes ) { |
|
1262 Backbone.Relational.store.update( this ); |
|
1263 } |
|
1264 |
|
1265 if ( attributes ) { |
|
1266 this.updateRelations( options ); |
|
1267 } |
|
1268 |
|
1269 // Try to run the global queue holding external events |
|
1270 Backbone.Relational.eventQueue.unblock(); |
|
1271 |
|
1272 return result; |
|
1273 }, |
|
1274 |
|
1275 unset: function( attribute, options ) { |
|
1276 Backbone.Relational.eventQueue.block(); |
|
1277 |
|
1278 var result = Backbone.Model.prototype.unset.apply( this, arguments ); |
|
1279 this.updateRelations( options ); |
|
1280 |
|
1281 // Try to run the global queue holding external events |
|
1282 Backbone.Relational.eventQueue.unblock(); |
|
1283 |
|
1284 return result; |
|
1285 }, |
|
1286 |
|
1287 clear: function( options ) { |
|
1288 Backbone.Relational.eventQueue.block(); |
|
1289 |
|
1290 var result = Backbone.Model.prototype.clear.apply( this, arguments ); |
|
1291 this.updateRelations( options ); |
|
1292 |
|
1293 // Try to run the global queue holding external events |
|
1294 Backbone.Relational.eventQueue.unblock(); |
|
1295 |
|
1296 return result; |
|
1297 }, |
|
1298 |
|
1299 /** |
|
1300 * Override 'change', so the change will only execute after 'set' has finised (relations are updated), |
|
1301 * and 'previousAttributes' will be available when the event is fired. |
|
1302 */ |
|
1303 change: function( options ) { |
|
1304 var dit = this, args = arguments; |
|
1305 Backbone.Relational.eventQueue.add( function() { |
|
1306 Backbone.Model.prototype.change.apply( dit, args ); |
|
1307 }); |
|
1308 }, |
|
1309 |
|
1310 clone: function() { |
|
1311 var attributes = _.clone( this.attributes ); |
|
1312 if ( !_.isUndefined( attributes[ this.idAttribute ] ) ) { |
|
1313 attributes[ this.idAttribute ] = null; |
|
1314 } |
|
1315 |
|
1316 _.each( this.getRelations(), function( rel ) { |
|
1317 delete attributes[ rel.key ]; |
|
1318 }); |
|
1319 |
|
1320 return new this.constructor( attributes ); |
|
1321 }, |
|
1322 |
|
1323 /** |
|
1324 * Convert relations to JSON, omits them when required |
|
1325 */ |
|
1326 toJSON: function() { |
|
1327 // If this Model has already been fully serialized in this branch once, return to avoid loops |
|
1328 if ( this.isLocked() ) { |
|
1329 return this.id; |
|
1330 } |
|
1331 |
|
1332 this.acquire(); |
|
1333 var json = Backbone.Model.prototype.toJSON.call( this ); |
|
1334 |
|
1335 if ( this.constructor._superModel && !( this.constructor._subModelTypeAttribute in json ) ) { |
|
1336 json[ this.constructor._subModelTypeAttribute ] = this.constructor._subModelTypeValue; |
|
1337 } |
|
1338 |
|
1339 _.each( this._relations, function( rel ) { |
|
1340 var value = json[ rel.key ]; |
|
1341 |
|
1342 if ( rel.options.includeInJSON === true) { |
|
1343 if ( value && _.isFunction( value.toJSON ) ) { |
|
1344 json[ rel.keyDestination ] = value.toJSON(); |
|
1345 } |
|
1346 else { |
|
1347 json[ rel.keyDestination ] = null; |
|
1348 } |
|
1349 } |
|
1350 else if ( _.isString( rel.options.includeInJSON ) ) { |
|
1351 if ( value instanceof Backbone.Collection ) { |
|
1352 json[ rel.keyDestination ] = value.pluck( rel.options.includeInJSON ); |
|
1353 } |
|
1354 else if ( value instanceof Backbone.Model ) { |
|
1355 json[ rel.keyDestination ] = value.get( rel.options.includeInJSON ); |
|
1356 } |
|
1357 else { |
|
1358 json[ rel.keyDestination ] = null; |
|
1359 } |
|
1360 } |
|
1361 else if ( _.isArray( rel.options.includeInJSON ) ) { |
|
1362 if ( value instanceof Backbone.Collection ) { |
|
1363 var valueSub = []; |
|
1364 value.each( function( model ) { |
|
1365 var curJson = {}; |
|
1366 _.each( rel.options.includeInJSON, function( key ) { |
|
1367 curJson[ key ] = model.get( key ); |
|
1368 }); |
|
1369 valueSub.push( curJson ); |
|
1370 }); |
|
1371 json[ rel.keyDestination ] = valueSub; |
|
1372 } |
|
1373 else if ( value instanceof Backbone.Model ) { |
|
1374 var valueSub = {}; |
|
1375 _.each( rel.options.includeInJSON, function( key ) { |
|
1376 valueSub[ key ] = value.get( key ); |
|
1377 }); |
|
1378 json[ rel.keyDestination ] = valueSub; |
|
1379 } |
|
1380 else { |
|
1381 json[ rel.keyDestination ] = null; |
|
1382 } |
|
1383 } |
|
1384 else { |
|
1385 delete json[ rel.key ]; |
|
1386 } |
|
1387 |
|
1388 if ( rel.keyDestination !== rel.key ) { |
|
1389 delete json[ rel.key ]; |
|
1390 } |
|
1391 }); |
|
1392 |
|
1393 this.release(); |
|
1394 return json; |
|
1395 } |
|
1396 }, |
|
1397 { |
|
1398 setup: function( superModel ) { |
|
1399 // We don't want to share a relations array with a parent, as this will cause problems with |
|
1400 // reverse relations. |
|
1401 this.prototype.relations = ( this.prototype.relations || [] ).slice( 0 ); |
|
1402 |
|
1403 this._subModels = {}; |
|
1404 this._superModel = null; |
|
1405 |
|
1406 // If this model has 'subModelTypes' itself, remember them in the store |
|
1407 if ( this.prototype.hasOwnProperty( 'subModelTypes' ) ) { |
|
1408 Backbone.Relational.store.addSubModels( this.prototype.subModelTypes, this ); |
|
1409 } |
|
1410 // The 'subModelTypes' property should not be inherited, so reset it. |
|
1411 else { |
|
1412 this.prototype.subModelTypes = null; |
|
1413 } |
|
1414 |
|
1415 // Initialize all reverseRelations that belong to this new model. |
|
1416 _.each( this.prototype.relations, function( rel ) { |
|
1417 if ( !rel.model ) { |
|
1418 rel.model = this; |
|
1419 } |
|
1420 |
|
1421 if ( rel.reverseRelation && rel.model === this ) { |
|
1422 var preInitialize = true; |
|
1423 if ( _.isString( rel.relatedModel ) ) { |
|
1424 /** |
|
1425 * The related model might not be defined for two reasons |
|
1426 * 1. it never gets defined, e.g. a typo |
|
1427 * 2. it is related to itself |
|
1428 * In neither of these cases do we need to pre-initialize reverse relations. |
|
1429 */ |
|
1430 var relatedModel = Backbone.Relational.store.getObjectByName( rel.relatedModel ); |
|
1431 preInitialize = relatedModel && ( relatedModel.prototype instanceof Backbone.RelationalModel ); |
|
1432 } |
|
1433 |
|
1434 var type = !_.isString( rel.type ) ? rel.type : Backbone[ rel.type ] || Backbone.Relational.store.getObjectByName( rel.type ); |
|
1435 if ( preInitialize && type && type.prototype instanceof Backbone.Relation ) { |
|
1436 new type( null, rel ); |
|
1437 } |
|
1438 } |
|
1439 }, this ); |
|
1440 |
|
1441 return this; |
|
1442 }, |
|
1443 |
|
1444 /** |
|
1445 * Create a 'Backbone.Model' instance based on 'attributes'. |
|
1446 * @param {Object} attributes |
|
1447 * @param {Object} [options] |
|
1448 * @return {Backbone.Model} |
|
1449 */ |
|
1450 build: function( attributes, options ) { |
|
1451 var model = this; |
|
1452 |
|
1453 // 'build' is a possible entrypoint; it's possible no model hierarchy has been determined yet. |
|
1454 this.initializeModelHierarchy(); |
|
1455 |
|
1456 // Determine what type of (sub)model should be built if applicable. |
|
1457 // Lookup the proper subModelType in 'this._subModels'. |
|
1458 if ( this._subModels && this.prototype.subModelTypeAttribute in attributes ) { |
|
1459 var subModelTypeAttribute = attributes[ this.prototype.subModelTypeAttribute ]; |
|
1460 var subModelType = this._subModels[ subModelTypeAttribute ]; |
|
1461 if ( subModelType ) { |
|
1462 model = subModelType; |
|
1463 } |
|
1464 } |
|
1465 |
|
1466 return new model( attributes, options ); |
|
1467 }, |
|
1468 |
|
1469 initializeModelHierarchy: function() { |
|
1470 // If we're here for the first time, try to determine if this modelType has a 'superModel'. |
|
1471 if ( _.isUndefined( this._superModel ) || _.isNull( this._superModel ) ) { |
|
1472 Backbone.Relational.store.setupSuperModel( this ); |
|
1473 |
|
1474 // If a superModel has been found, copy relations from the _superModel if they haven't been |
|
1475 // inherited automatically (due to a redefinition of 'relations'). |
|
1476 // Otherwise, make sure we don't get here again for this type by making '_superModel' false so we fail |
|
1477 // the isUndefined/isNull check next time. |
|
1478 if ( this._superModel ) { |
|
1479 // |
|
1480 if ( this._superModel.prototype.relations ) { |
|
1481 var supermodelRelationsExist = _.any( this.prototype.relations, function( rel ) { |
|
1482 return rel.model && rel.model !== this; |
|
1483 }, this ); |
|
1484 |
|
1485 if ( !supermodelRelationsExist ) { |
|
1486 this.prototype.relations = this._superModel.prototype.relations.concat( this.prototype.relations ); |
|
1487 } |
|
1488 } |
|
1489 } |
|
1490 else { |
|
1491 this._superModel = false; |
|
1492 } |
|
1493 } |
|
1494 |
|
1495 // If we came here through 'build' for a model that has 'subModelTypes', and not all of them have been resolved yet, try to resolve each. |
|
1496 if ( this.prototype.subModelTypes && _.keys( this.prototype.subModelTypes ).length !== _.keys( this._subModels ).length ) { |
|
1497 _.each( this.prototype.subModelTypes, function( subModelTypeName ) { |
|
1498 var subModelType = Backbone.Relational.store.getObjectByName( subModelTypeName ); |
|
1499 subModelType && subModelType.initializeModelHierarchy(); |
|
1500 }); |
|
1501 } |
|
1502 }, |
|
1503 |
|
1504 /** |
|
1505 * Find an instance of `this` type in 'Backbone.Relational.store'. |
|
1506 * - If `attributes` is a string or a number, `findOrCreate` will just query the `store` and return a model if found. |
|
1507 * - If `attributes` is an object, the model will be updated with `attributes` if found. |
|
1508 * Otherwise, a new model is created with `attributes` (unless `options.create` is explicitly set to `false`). |
|
1509 * @param {Object|String|Number} attributes Either a model's id, or the attributes used to create or update a model. |
|
1510 * @param {Object} [options] |
|
1511 * @param {Boolean} [options.create=true] |
|
1512 * @return {Backbone.RelationalModel} |
|
1513 */ |
|
1514 findOrCreate: function( attributes, options ) { |
|
1515 // Try to find an instance of 'this' model type in the store |
|
1516 var model = Backbone.Relational.store.find( this, attributes ); |
|
1517 |
|
1518 // If we found an instance, update it with the data in 'item'; if not, create an instance |
|
1519 // (unless 'options.create' is false). |
|
1520 if ( _.isObject( attributes ) ) { |
|
1521 if ( model ) { |
|
1522 model.set( model.parse ? model.parse( attributes ) : attributes, options ); |
|
1523 } |
|
1524 else if ( !options || ( options && options.create !== false ) ) { |
|
1525 model = this.build( attributes, options ); |
|
1526 } |
|
1527 } |
|
1528 |
|
1529 return model; |
|
1530 } |
|
1531 }); |
|
1532 _.extend( Backbone.RelationalModel.prototype, Backbone.Semaphore ); |
|
1533 |
|
1534 /** |
|
1535 * Override Backbone.Collection._prepareModel, so objects will be built using the correct type |
|
1536 * if the collection.model has subModels. |
|
1537 */ |
|
1538 Backbone.Collection.prototype.__prepareModel = Backbone.Collection.prototype._prepareModel; |
|
1539 Backbone.Collection.prototype._prepareModel = function ( model, options ) { |
|
1540 options || (options = {}); |
|
1541 if ( !( model instanceof Backbone.Model ) ) { |
|
1542 var attrs = model; |
|
1543 options.collection = this; |
|
1544 |
|
1545 if ( typeof this.model.findOrCreate !== 'undefined' ) { |
|
1546 model = this.model.findOrCreate( attrs, options ); |
|
1547 } |
|
1548 else { |
|
1549 model = new this.model( attrs, options ); |
|
1550 } |
|
1551 |
|
1552 if ( !model._validate( model.attributes, options ) ) { |
|
1553 model = false; |
|
1554 } |
|
1555 } |
|
1556 else if ( !model.collection ) { |
|
1557 model.collection = this; |
|
1558 } |
|
1559 |
|
1560 return model; |
|
1561 } |
|
1562 |
|
1563 /** |
|
1564 * Override Backbone.Collection.add, so objects fetched from the server multiple times will |
|
1565 * update the existing Model. Also, trigger 'relational:add'. |
|
1566 */ |
|
1567 var add = Backbone.Collection.prototype.__add = Backbone.Collection.prototype.add; |
|
1568 Backbone.Collection.prototype.add = function( models, options ) { |
|
1569 options || (options = {}); |
|
1570 if ( !_.isArray( models ) ) { |
|
1571 models = [ models ]; |
|
1572 } |
|
1573 |
|
1574 var modelsToAdd = []; |
|
1575 |
|
1576 //console.debug( 'calling add on coll=%o; model=%o, options=%o', this, models, options ); |
|
1577 _.each( models, function( model ) { |
|
1578 if ( !( model instanceof Backbone.Model ) ) { |
|
1579 // `_prepareModel` attempts to find `model` in Backbone.store through `findOrCreate`, |
|
1580 // and sets the new properties on it if is found. Otherwise, a new model is instantiated. |
|
1581 model = Backbone.Collection.prototype._prepareModel.call( this, model, options ); |
|
1582 } |
|
1583 |
|
1584 if ( model instanceof Backbone.Model && !this.get( model ) && !this.getByCid( model ) ) { |
|
1585 modelsToAdd.push( model ); |
|
1586 } |
|
1587 }, this ); |
|
1588 |
|
1589 |
|
1590 // Add 'models' in a single batch, so the original add will only be called once (and thus 'sort', etc). |
|
1591 if ( modelsToAdd.length ) { |
|
1592 add.call( this, modelsToAdd, options ); |
|
1593 |
|
1594 _.each( modelsToAdd, function( model ) { |
|
1595 this.trigger( 'relational:add', model, this, options ); |
|
1596 }, this ); |
|
1597 } |
|
1598 |
|
1599 return this; |
|
1600 }; |
|
1601 |
|
1602 /** |
|
1603 * Override 'Backbone.Collection.remove' to trigger 'relational:remove'. |
|
1604 */ |
|
1605 var remove = Backbone.Collection.prototype.__remove = Backbone.Collection.prototype.remove; |
|
1606 Backbone.Collection.prototype.remove = function( models, options ) { |
|
1607 options || (options = {}); |
|
1608 if ( !_.isArray( models ) ) { |
|
1609 models = [ models ]; |
|
1610 } |
|
1611 else { |
|
1612 models = models.slice( 0 ); |
|
1613 } |
|
1614 |
|
1615 //console.debug('calling remove on coll=%o; models=%o, options=%o', this, models, options ); |
|
1616 _.each( models, function( model ) { |
|
1617 model = this.getByCid( model ) || this.get( model ); |
|
1618 |
|
1619 if ( model instanceof Backbone.Model ) { |
|
1620 remove.call( this, model, options ); |
|
1621 this.trigger('relational:remove', model, this, options); |
|
1622 } |
|
1623 }, this ); |
|
1624 |
|
1625 return this; |
|
1626 }; |
|
1627 |
|
1628 /** |
|
1629 * Override 'Backbone.Collection.reset' to trigger 'relational:reset'. |
|
1630 */ |
|
1631 var reset = Backbone.Collection.prototype.__reset = Backbone.Collection.prototype.reset; |
|
1632 Backbone.Collection.prototype.reset = function( models, options ) { |
|
1633 reset.call( this, models, options ); |
|
1634 this.trigger( 'relational:reset', this, options ); |
|
1635 |
|
1636 return this; |
|
1637 }; |
|
1638 |
|
1639 /** |
|
1640 * Override 'Backbone.Collection.sort' to trigger 'relational:reset'. |
|
1641 */ |
|
1642 var sort = Backbone.Collection.prototype.__sort = Backbone.Collection.prototype.sort; |
|
1643 Backbone.Collection.prototype.sort = function( options ) { |
|
1644 sort.call( this, options ); |
|
1645 this.trigger( 'relational:reset', this, options ); |
|
1646 |
|
1647 return this; |
|
1648 }; |
|
1649 |
|
1650 /** |
|
1651 * Override 'Backbone.Collection.trigger' so 'add', 'remove' and 'reset' events are queued until relations |
|
1652 * are ready. |
|
1653 */ |
|
1654 var trigger = Backbone.Collection.prototype.__trigger = Backbone.Collection.prototype.trigger; |
|
1655 Backbone.Collection.prototype.trigger = function( eventName ) { |
|
1656 if ( eventName === 'add' || eventName === 'remove' || eventName === 'reset' ) { |
|
1657 var dit = this, args = arguments; |
|
1658 |
|
1659 if (eventName === 'add') { |
|
1660 args = _.toArray(args); |
|
1661 // the fourth argument in case of a regular add is the option object. |
|
1662 // we need to clone it, as it could be modified while we wait on the eventQueue to be unblocked |
|
1663 if (_.isObject(args[3])) { |
|
1664 args[3] = _.clone(args[3]); |
|
1665 } |
|
1666 } |
|
1667 |
|
1668 Backbone.Relational.eventQueue.add( function() { |
|
1669 trigger.apply( dit, args ); |
|
1670 }); |
|
1671 } |
|
1672 else { |
|
1673 trigger.apply( this, arguments ); |
|
1674 } |
|
1675 |
|
1676 return this; |
|
1677 }; |
|
1678 |
|
1679 // Override .extend() to automatically call .setup() |
|
1680 Backbone.RelationalModel.extend = function( protoProps, classProps ) { |
|
1681 var child = Backbone.Model.extend.apply( this, arguments ); |
|
1682 |
|
1683 child.setup( this ); |
|
1684 |
|
1685 return child; |
|
1686 }; |
|
1687 })(); |