Marionette.js CollectionView,只渲染特定的模型

时间:2013-09-04 01:51:00

标签: javascript jquery model-view-controller backbone.js marionette

我正在重构我的Backbone.js应用程序以使用Marionette.js,我试图绕着CollectionView包围。

假设我有几个ItemView个模型Cow

// Declare my models.
var Cow = Backbone.Model.extend({});
var Cows = Backbone.Collection.extend({
  model: Cow
});

// Make my views
var GrassPatch = Marionette.ItemView.extend({
  tagName:  'div',
  template: "<section class='grass'>{{name}}</section>",
})

var Pasture = Marionette.CollectionView.extend({});

// Instantiate the CollectionView,
var blissLand = new Pasture({
  itemView: GrassPatch;
});

// Now, add models to the collection.
Cows.add({
  name: 'Bessie',
  hasSpots: true
});

Cows.add({
  name: 'Frank',
  hasSpots: false
});

现在这就是诀窍。我只想要在我的牧场上有斑点的奶牛。在定义我的CollectionView(牧场)时,我是如何告诉它只关注那些hasSpots === true的模型?

理想情况下,我希望在所有事件中都有CollectionView过滤器,但最低限度的是,我如何仅根据模型属性呈现一些ItemView?

更新

我使用了David Sulc的例子,这是一个简单的解决方案。这是一个示例实现:

  this.collection = Backbone.filterCollection(this.collection, function(criterion){
    var len = String(criterion).length;
    var a = criterion.toLowerCase();
    return function(model){
      var b = String(model.get('name')).substr(0, len).toLowerCase();
      if (a === b) {
        return model;
      }
    };
  });

  this.collection.add({ name: 'foo' });
  this.collection.add({ name: 'foosball' });
  this.collection.add({ name: 'foo bar' });
  this.collection.add({ name: 'goats' });
  this.collection.add({ name: 'cows' });

  this.collection.filter('foo');

  // -> returns the first three models

4 个答案:

答案 0 :(得分:3)

Marionette在v2.4.1中为您处理此事。

请参阅CollectionView.filter方法。

以下是来自文档:

  var cv = new Marionette.CollectionView({
    childView: SomeChildView,
    emptyView: SomeEmptyView,
    collection: new Backbone.Collection([
      { value: 1 },
      { value: 2 },
      { value: 3 },
      { value: 4 }
    ]),

    // Only show views with even values
    filter: function (child, index, collection) {
      return child.get('value') % 2 === 0;
    }
  });

  // renders the views with values '2' and '4'
  cv.render();

  // change the filter
  cv.filter = function (child, index, collection) {
    return child.get('value') % 2 !== 0;
  };

  // renders the views with values '1' and '3'
  cv.render();

  // remove the filter
  // note that using `delete cv.filter` will cause the prototype's filter to be used
  // which may be undesirable
  cv.filter = null;

  // renders all views
  cv.render();

答案 1 :(得分:0)

@Will M建议过滤集合是适当的方法。

答案 2 :(得分:0)

正如其他人所建议的那样,实现此目的的最佳方法是过滤集合以仅包含要显示的模型,并将该fitlered集合传递给CollectionView进行渲染。

您可以在此处看到一个有效的示例:http://davidsulc.github.io/marionette-gentle-introduction/#contacts使用右上角的输入字段过滤联系人,仅显示包含该文字的模型(例如“li”)。

这是通过使用处理过滤的特殊集合类型来实现的:https://github.com/davidsulc/marionette-gentle-introduction/blob/master/assets/js/entities/common.js

它在这里得到实例化:https://github.com/davidsulc/marionette-gentle-introduction/blob/master/assets/js/apps/contacts/list/list_controller.js#L13

此代码来自我的book on Marionette

答案 3 :(得分:0)

有时,由于某些自定义逻辑,您无法过滤您的收藏,并且您希望这些模型位于集合中,但您不希望它们被渲染。为实现这一目标,您可以:

var Pasture = Marionette.CollectionView.extend({
     addChild: function(child, ChildView, index) {
               if(child.get('hasSpots')) {
                    return Marionette.CollectionView.prototype.addChild.call(this, child, ChildView, index);
               }
     }});

虽然我同意过滤集合是一种更好的方法。