javascript - Implementing a composite pattern using MVCBackbone.js - Stack Overflow

My webapp has a posite structure i.e. each Category collection can contain a mixture of individual Item

My webapp has a posite structure i.e. each Category collection can contain a mixture of individual Items and other Categories as its rows/nodes/children (not sure of the correct terminology here). In actual fact, it's a little bit simpler than that as each collection is represented by a model, Category, so essentially each Category collection has both Item models and Category models as its children.

In general is this an advisable way to implement this structure using MVC? More specifically, in Backbone.js is it possible for a collection to have a model factory (taking the json and calculating which model to generate based on the json's structure) instead of a static model property?

My webapp has a posite structure i.e. each Category collection can contain a mixture of individual Items and other Categories as its rows/nodes/children (not sure of the correct terminology here). In actual fact, it's a little bit simpler than that as each collection is represented by a model, Category, so essentially each Category collection has both Item models and Category models as its children.

In general is this an advisable way to implement this structure using MVC? More specifically, in Backbone.js is it possible for a collection to have a model factory (taking the json and calculating which model to generate based on the json's structure) instead of a static model property?

Share asked Jan 16, 2012 at 22:54 wheresrhyswheresrhys 23.6k21 gold badges96 silver badges165 bronze badges 1
  • In short, I'd say, "I don't see why not", but I'm curious what others might say. – JayC Commented Jan 19, 2012 at 19:17
Add a ment  | 

4 Answers 4

Reset to default 5 +50

I'm assuming you're receiving a Category/Items list in JSON that looks something like this...

{
    'id': 1,
    'name': 'My 1st Category',
    'children': [
        {
            'id': 2,
            'name': 'My 2nd Category',
            'children': []
        },
        {
            'id': 1,
            'name': 'An Item',
            'price': 109.99
        }
    ]
}

Backbone.js doesn't have anything out of the box that supports multiple models in a collection, but it also doesn't have any restrictions on the types of models you put in a collection.

Specifying the model type in the collection definition only does one thing, it lets Backbone know what model type to create if you pass raw JSON to the collection instead of a Backbone.Model object. If you add an Item model to a collection that already contains a few Category Models, Backbone will have no problem popping it into the models list; it doesn't do any type checking.

So with that in mind, you can use just about everything the collection offers except pass it raw JSON; you'll need to handle that yourself. So your choices are to either build up your models beforehand, making them into Backbone.Model objects, or create something that'll do the parsing for you.

For the second option, the parser, I'd suggest passing a special variable to the collection that contains your raw JSON, then handling that in your initialize function. Here's an example:

var CategoryCollection = Backbone.Collection.extend({
    initialize: function(m, models) {
        _.each(models, function(model) {
            var modelObject = null;
            if (model.price !== undefined) {
                modelObject = new Item(model);
            } else {
                modelObject = new Category(model);
            }

            this.add(modelObject);
        }, this);
    }
});

So it is a little hacky, but you determine the type of model based on if it has a specific field (price in my example), create the model object, then add that to the collection.

You'd then call it this way:

var myCollection = new CategoryCollection([], myJSON);

Notice you have to pass an empty array as the first argument since that's how'd you normally pass a set of models to the collection.

Later on when using the collection, you can determine if you're dealing with an Item or Category using a simple instanceof check:

_.each(myCollection.models, function(model) {
    if (model instanceof Item) {
        console.log("It's an Item! Price: ", model.get("price"));
    } else {
        console.log("It's a Category!");
    }
});

Yes, you can. I have done it before. I think this links could help you: http://documentcloud.github./backbone/#Collection-model

Here is one of the main scripts I used for my project: https://gist.github./b65893e0c2e3c46d3dc1

It's fairly simply achieved by overwriting the built in public methods parse and toJSON used internally by backbone when retrieving and saving model's data.

Firstly when you retrive the Model from the database you should overwrite the model's parse method to create models representing given items from your example.

Then on save the toJSON method is used to serialize the data back to what the server can understand - there you'd just call the toJSON method on each item's Model to serialize it to a format that backend will recognize. If you look at the code for Backbone.sync you will see that the model always gets serialized using the toJSON if no custom data is passed.

Let me know if you needed more detailed info though I believe you should be able to pick it up from here!

Yes, you can use a Factory Method pattern for model creation in backbone's Collections. Assuming the data structure suggested in @kpeel's answer, you could define

// Factory method
var CategoryOrItemFactory = function(data, options) {
    if (data.children) {
        return new Category(data, options);
    } else {
        return new Item(data, options);
    }
};

// Model definitions
var Item = Backbone.Model.extend();

var Category = Backbone.Model.extend({
    initialize: function() {
        this.children = new Children(this.get("children"));
    }
});

var Children = Backbone.Collection.extend({
    model: CategoryOrItemFactory
});

You would then create your root item (Category) and have the plete data structure created:

// create the root item
var rootItem = new Category(rawData);

Access a Category's children its children property, e.g.

  rootItem.children.get(2).get("name");

Here is a jsFiddle with the above code to play with.

发布者:admin,转转请注明出处:http://www.yc00.com/questions/1743690247a4490841.html

相关推荐

发表回复

评论列表(0条)

  • 暂无评论

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信