Backbone-DocumentModel
A plugin to create entire Document structures with nested Backbone.js Models & Collections with deep model
references and event bubbling
.
The Document is simply a reference to the project's goal of allowing MongoDB Document JSON representation to be dynamically composed/referenced/updated and saved using native Backbone.js components.
Another Backbone.js plugin...
After working with document objects we kept running into a situation where we wanted to pass Model/Collection objects to our nested Backbone.Views, however this proved troublesome to keep track of changes made within those Views.
user.set({
name: {
first: 'John',
last: 'Doe'
},
addresses: [
{ type: 'Shipping', city: 'Charlottesville', state: 'VA' },
{ type: 'Billing', city: 'Prescott', state: 'AZ' }
]
});
When making a new Backbone.View its common to pass in a Model or Collection, and it would be best practice to pass only the specific Model/Collection that the control needed.
var AddressModalView = Backbone.View.extend({
events: {
'click .save': 'onSave'
},
render: function () {
},
onSave: function () {
if (this.model) {
this.model.set('type', this.$el.find('.type').val());
this.model.set('city', this.$el.find('.city').val());
this.model.set('state', this.$el.find('.state').val());
} else {
this.collection.add({
type: this.$el.find('.type').val(),
city: this.$el.find('.city').val(),
state: this.$el.find('.state').val()
});
}
}
});
var UserView = Backbone.View.extend({
initialize: function () {
this.model.on('add:addresses', function () {
alert('address added!');
}, this);
this.model.on('remove:addresses', function () {
alert('address removed!');
}, this);
this.model.on('change:addresses.*', function () {
alert('address changed!');
}, this);
},
onAddAddress: function () {
var addressModalView = new AddressModalView({ collection: this.model.get('addresses') });
addressModalView.render();
addressModalView.show();
},
onEditAddress: function () {
var addressModalView = new AddressModalView({ model: this.model.get('addresses').at(0) });
addressModalView.render();
addressModalView.show();
},
onRemoveAddress: function () {
this.model.get('addresses').remove(this.model.get('addresses.0'));
}
});
Usage
-
Download the latest version here, and add backbone-documentmodel.js
to your HTML <head>
, after backbone.js
is included.
<script type="text/javascript" src="http://ajax.googleapis.com/ajax/libs/jquery/1.7.2/jquery.min.js"></script>
<script type="text/javascript" src="underscore.js"></script>
<script type="text/javascript" src="backbone.js"></script>
<script type="text/javascript" src="backbone-documentmodel.js"></script>
-
Change your models to extend from Backbone.DocumentModel
, e.g.
var Person = Backbone.Model.extend({ ... });
var Person = Backbone.DocumentModel.extend({ ... });
-
Change your collections to extend from Backbone.DocumentCollection
, e.g.
var People = Backbone.Collection.extend({ ... });
var People = Backbone.DocumentCollection.extend({ ... });
Nested Attributes & Document Composition
get()
and set()
will work as before, you can now reference deep model names and set()
will also dynamically compose Document data into nested Models & Collections:
1-1
set()
user.set({
name: {
first: 'John',
last: 'Doe',
middle: {
initial: 'Z'
}
}
});
user.get('name').get('middle').set('initial', 'Z');
user.set({
'name.first': 'John',
'name.last': 'Doe',
'name.middle.initial': 'Z'
});
user.get('name').set({ 'middle.initial': 'Z' });
get()
user.get('name.first');
user.get('name.middle.initial');
user.get('name').get('middle.initial');
user.get('name').get('first');
user.get('name').get('middle').get('initial');
1-N
set()
user.set({
addresses: [
{ city: 'Charlottesville', state: 'VA' },
{ city: 'Prescott', state: 'AZ' }
]
});
user.get('addresses').at(0).set('state', 'VA');
user.get('addresses').at(1).set({ state: 'AZ', city: 'Prescott' });
user.set('addresses.0.state': 'VA');
user.set({ 'addresses.1.state': 'AZ', 'addresses.1.city': 'Prescott' });
get()
user.get('addresses.0.state')
user.get('addresses.1.city')
user.get('addresses').at(0).get('state')
user.get('addresses').at(1).get('city')
JSON
toJSON
will decompose the Document Model/Collection into a JSON object, ready for transport.
serverInput = {
name: {
first: 'John',
last: 'Doe',
middle: {
initial: 'Z'
}
},
addresses: [
{ city: 'Charlottesville', state: 'VA' },
{ city: 'Prescott', state: 'AZ' }
],
items: [123, 456]
};
user.set(serverInput);
modelJSON = user.toJSON();
modelJSON = {
name: {
first: 'John',
last: 'Doe',
middle: {
initial: 'Z'
}
},
addresses: [
{ city: 'Charlottesville', state: 'VA' },
{ city: 'Prescott', state: 'AZ' }
],
items: [123, 456]
};
Events
"change"
"change"
events can be bound to nested attributes in the same way, and changing nested attributes will fire up the chain:
user.get('name').get('middle').on('change:initial', function () { ... });
user.get('name').on('change:middle.initial', function () { ... });
user.on('change:name.middle.initial', function () { ... });
user.on('change:addresses.city', function () { ... });
user.on('change:addresses.*', function () { ... });
user.on('change:*', function () { ... });
"add" and "remove"
Additionally, nested arrays fire "add"
and "remove"
events:
user.on('add:addresses', function () { ... });
user.on('add:*', function () { ... });
user.on('remove:addresses', function () { ... });
user.on('remove:*', function () { ... });
Changelog
0.6.4
- Fixes issue #7 nested array's toJSON() now returns the value.
0.6.3
- Fixes issue #6 fully reference Object.prototype.toString since certain browser's do not resolve method.
0.6.2
- Fixes issue #5 address wrapped primitives.
0.6.1
- Fixes issue #4 properly handles nested arrays.
0.6.0
- Added minimally altered backbone model/collection unit tests.
- Added readme.md specific tests.
- Updated backbone-documentmodel.js to pass unit tests.
0.5.x