I want to differentiate node items based on item_id with Backbone.xmpp. For example in the todo.app for each "todo" item I want to be able to assign different notes (or several users publish notes on each "todo"). I thought of assigning those notes to a todo based on the todo id.
Can I use Backbone relational with Backbone.xmpp ?
Any help or guidelines are appreciated.
Edit2: What options do I have to store nested models in leaf nodes on the xmpp server?
todos and notes are separate items published on the leaf node. Will it be efficient to assign comments to todos? The differetiation will be based on item ids (todoid: todo_1, noteid: todo_1_note_1).
todos are items and notes are an array of objects inside a todo item (JSON objects)? But with this solution I won't get notifications when notes are published since it will be an update of the todo item. Further, all notes will be stored in one single item - and this could get very long.
Originally I had the idea to map todos on leaf nodes (as leaf node name or "title" attribute) and notes on items, but BB.xmpp would not support this so far right?
Therefore, I tend to the first solution where todos and notes are differentiated by item id.
How is it possible in Backbone.xmpp to achieve that?
Edit1: The code is for the original todo.app with localstorage.
$(function(){
// ------------------- Todo Model ------------------
var Todo = Backbone.RelationalModel.extend({
relations: [{
type: Backbone.HasMany,
key: "children",
relatedModel: "Todo",
collectionType: "TodoList",
reverseRelation: {
key: "parent",
includeInJSON: "id"
}
}],
initialize: function() {
console.log("MODEL: initialize()");
if (!this.get("order") && this.get ("parent")) {
this.set( {order: this.get("parent").nextChildIndex() });
}
},
defaults: function() {
console.log("MODEL: defaults()");
return {
done: false,
content: "default content" };
},
nextChildIndex: function() {
var children = this.get( 'children' );
return children && children.length || 0;
},
clear: function() {
this.destroy();
}
});
// ------------------- Todo Collection ------------------
var TodoList = Backbone.Collection.extend({
model: Todo,
// Save all of the todo items under the `"todos"` namespace.
localStorage: new Store("todos-backbone"),
done: function() {
return this.filter(function(todo){ return todo.get('done'); });
},
});
var Todos = new TodoList;
// ------------------- Todo View ------------------
var TodoView = Backbone.View.extend({
tagName: "li",
template: _.template($('#item-template').html()),
events: {
"keypress input.add-child": "addChild",
"click .check" : "toggleDone",
"dblclick label.todo-content" : "edit",
"click span.todo-destroy" : "clear",
"keypress .todo-input" : "updateOnEnter",
"blur .todo-input" : "close"
},
initialize: function() {
console.log("TODOVIEW: initialize()");
this.model.bind('change', this.render);
this.model.bind('destroy', this.remove);
this.model.bind("update:children", this.renderChild);
this.model.bind("add:children", this.renderChild);
this.el = $( this.el );
this.childViews = {};
},
render: function() {
console.log("TODOVIEW: render()");
$(this.el).html(this.template(this.model.toJSON()));
this.setText();
this.input = this.$('.todo-input');
this.el.append("<ul>", {"class": "children"}).append("<input>", { type: "text", "class": "add-child" });
_.each(this.get("children"), function(child) {
this.renderChild(child);
}, this);
return this;
},
addChild: function(text) {
console.log("TODOVIEW: addChild()");
if (e.keyCode == 13){
var text = this.el.find("input.add-child").text();
var child = new Todo( { parent: this.model, text: text});
}
},
renderChild: function(model){
console.log("TODOVIEW: renderChild()");
var childView = new TodoView({ model: model});
this.childViews[model.cid] = childView;
this.el.find("ul.children").append(childView.render());
},
// Remove the item, destroy the model.
clear: function() {
console.log("TODOVIEW: clear()");
this.model.set({parent: null});
this.model.destroy();
//this.model.clear();
}
});
// ------------------ The Application ------------------------
var AppView = Backbone.View.extend({
el: $("#todoapp"),
statsTemplate: _.template($('#stats-template').html()),
events: {
"keypress #new-todo": "createOnEnter",
"keyup #new-todo": "showTooltip",
"click .todo-clear a": "clearCompleted",
"click .mark-all-done": "toggleAllComplete"
},
initialize: function() {
console.log("APPVIEW: initialize()");
_.bindAll(this, 'addOne', 'addAll', 'render', 'toggleAllComplete');
this.input = this.$("#new-todo");
Todos.bind('add', this.addOne);
Todos.bind('reset', this.addAll);
Todos.bind('all', this.render);
Todos.fetch();
},
render: function() {
},
addOne: function(todo) {
var view = new TodoView({model: todo});
this.$("#todo-list").append(view.render().el);
},
addAll: function() {
Todos.each(this.addOne);
},
// Generate the attributes for a new Todo item.
newAttributes: function() {
return {
content: this.input.val(),
order: Todos.nextOrder(),
done: false
};
},
createOnEnter: function(e) {
console.log("APPVIEW: createOnEnter()");
if (e.keyCode != 13) return;
Todos.create( this.newAttributes());
this.input.val('');
},
});
var App = new AppView;
});