forked from blittle/backbone-nested-models
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathbackbone-nested.js
238 lines (194 loc) · 8.3 KB
/
backbone-nested.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
/**
* Backbone Nested Models
* Author: Bret Little
* Version: 0.4
*
* Nested model support in Backbone.js
*
**/
(function(Backbone) {
var Model = Backbone.Model,
Collection = Backbone.Collection;
// This is a function, so that it can be overridden in special cases
Backbone.Model.prototype._attrConstructor = function(schema, val) {
return schema._constructor;
};
Backbone.Model.prototype.getParsedValue = function(attr, val, options) {
var relation = this.attributes[attr],
schema, _constructor,
id = this.idAttribute || "id",
modelToSet, modelsToAdd = [], modelsToRemove = [];
if(options.unset && relation) delete relation.parent;
if( val === null || typeof val === 'undefined' ) return val;
if(this.schema && _.has(this.schema, attr)) {
schema = this.schema[attr];
_constructor = this._attrConstructor(schema, val);
if(schema.type == 'related' && _.isFunction(_constructor)) {
// If the relation already exists, we don't want to replace it, rather
// update the data within it whether it is a collection or model
if(relation && relation instanceof Collection) {
// If the val that is being set is already a collection, use the models
// within the collection.
if(val instanceof Collection || val instanceof Array){
val = val.models || val;
modelsToAdd = _.clone(val);
relation.each(function(model, i) {
// If the model does not have an "id" skip logic to detect if it already
// exists and simply add it to the collection
if(typeof model[id] == 'undefined') return;
// If the incoming model also exists within the existing collection,
// call set on that model. If it doesn't exist in the incoming array,
// then add it to a list that will be removed.
var rModel = _.find(val, function(_model) {
return _model[id] === model[id];
});
if(rModel) {
model !== rModel &&
model.set(rModel.toJSON ? rModel.toJSON() : rModel);
// Remove the model from the incoming list because all remaining models
// will be added to the relation
modelsToAdd = _(modelsToAdd).reject(function(modelToAdd) {
return modelToAdd[id] === model[id];
});
} else {
modelsToRemove.push(model);
}
});
_.each(modelsToRemove, function(model) {
relation.remove(model);
});
relation.add(modelsToAdd);
} else {
// The incoming val that is being set is not an array or collection, then it represents
// a single model. Go through each of the models in the existing relation and remove
// all models that aren't the same as this one (by id). If it is the same, call set on that
// model.
relation.each(function(model) {
if(val[id] === model[id]) {
model.set(val);
} else {
relation.remove(model);
}
});
}
return relation;
}
else if(relation && relation instanceof Model) {
if(val instanceof _constructor) {
if(val.id != relation.id) {
return val;
} else {
val = val.attributes;
}
}
relation.set(val);
return relation;
}
else if(!_.isObject(val)) return val;
else if(val instanceof _constructor) {
return val;
}
else {
val = new _constructor(val, _.defaults({ _parent: this }, options));
val.parent = this;
}
}
else if(_.isFunction(_constructor)) {
val = new _constructor(val);
}
}
return val;
};
Backbone.Model.prototype.set = function(key, val, options) {
var attr, attrs, unset, changes, silent, changing, prev, current;
if (key == null) return this;
// Handle both `"key", value` and `{key: value}` -style arguments.
if (typeof key === 'object') {
attrs = key;
options = val;
} else {
(attrs = {})[key] = val;
}
options || (options = {});
// Run validation.
if (!this._validate(attrs, options)) return false;
// Extract attributes and options.
unset = options.unset;
silent = options.silent;
changes = [];
changing = this._changing;
this._changing = true;
if (!changing) {
this._previousAttributes = _.clone(this.attributes);
this.changed = {};
}
current = this.attributes, prev = this._previousAttributes;
// Check for changes of `id`.
if (this.idAttribute in attrs) this.id = attrs[this.idAttribute];
// For each `set` attribute, update or delete the current value.
for (attr in attrs) {
val = attrs[attr];
// Inject in the relational lookup
if(options.handle_relations !== false) {
val = this.getParsedValue(attr, val, options);
}
if (!_.isEqual(current[attr], val)) changes.push(attr);
if (!_.isEqual(prev[attr], val)) {
this.changed[attr] = val;
} else {
delete this.changed[attr];
}
unset ? delete current[attr] : current[attr] = val;
}
// Trigger all relevant attribute changes.
if (!silent) {
if (changes.length) this._pending = true;
for (var i = 0, l = changes.length; i < l; i++) {
this.trigger('change:' + changes[i], this, current[changes[i]], options);
}
}
if (changing) return this;
if (!silent) {
while (this._pending) {
this._pending = false;
this.trigger('change', this, options);
}
}
this._pending = false;
this._changing = false;
return this;
};
Backbone.Model.prototype.toJSON = function(options) {
var attrs = _.clone(this.attributes);
_.each(this.schema, function(rel, key) {
if (_.has(attrs, key)) {
attrs[key].toJSON && (attrs[key] = attrs[key].toJSON());
}
});
return attrs;
};
Backbone.Collection.prototype.resetRelations = function(options) {
_.each(this.models, function(model) {
_.each(model.schema, function(rel, key) {
var val = model.get(key);
if(val instanceof Backbone.Collection) {
val.trigger('reset', val, options);
}
});
});
};
Backbone.Collection.prototype.reset = function(models, options) {
options || (options = {});
for (var i = 0, l = this.models.length; i < l; i++) {
this._removeReference(this.models[i]);
}
options.previousModels = this.models;
this._reset();
this.add(models, _.extend({silent: true}, options));
if (!options.silent) {
this.trigger('reset', this, options);
this.resetRelations(options);
}
return this;
};
})(Backbone);