1

我有以下模型:

App.Checklist = DS.Model.extend({
  name: DS.attr('string'),
  checkitems: DS.hasMany('App.Checkitem', { embedded: true }),

  remainingItemsCount: function() {
    var checkitemsToCount = this.get('checkitems');
    return checkitemsToCount.filterProperty('isDone', false).get('length');
  }.property()

});

我想显示一个清单列表,其中包含每个列表的当前检查项的计数。

如果我将以下内容放入模板中,我会得到正确的输出:

{{#each checklists}}
  {{this.name}}
  {{this.remainingItemsCount}}
{{/each}}

但是,如果将新的检查项添加到检查表中,则计数不会增加。

但是,如果我更改remainingItemsCountChecklist 模型中的计算属性以使其依赖于checkitems.@each.done,那么计数会随着新检查项的添加而增加。

问题是,一旦添加了这个依赖项,子检查项的集合就是错误的——它不断重复第一个检查项的总检查项数(即,如果有五个项目的 'isDone' 为假,四个'isDone' 为真,则列表计数将显示为 9,第一个检查项将重复 9 次)。

我究竟做错了什么?

更新:

事实证明,将依赖项添加到 remainingItemsCount 属性会导致 ember-data 对服务器进行新的调用。

如果没有依赖关系,则在页面加载时会发出以下 XHR 请求:

GET http://localhost:3000/checklists

使用依赖项,在页面加载时会发出以下 XHR 请求:

GET http://localhost:3000/checklists
GET http://localhost:3000/checkitems

最后一个请求带有以下参数,这些参数似乎是第一个检查项的表示,包装在“ids”哈希中:

{"ids"=>
  {"0"=>
    {"id"=>"182",
     "checklist_id"=>"4",
     "title"=>
      "Make sure list count automatically increments",
     "is_done"=>"false"}},
 "action"=>"index",
 "controller"=>"checkitems"}

我想知道这是否是因为checkitem模型是使用 belongsTo 属性定义的?

App.Checkitem = DS.Model.extend({
  title: DS.attr('string'),
  isDone: DS.attr('boolean'),
  checklist: DS.belongsTo('App.Checklist')
});

更新 2

我仍然不确定为什么,但很明显,将依赖项添加到属性中,如下所示......

remainingItemsCount: function() {
    var checkitemsToCount = this.get('checkitems');
    return checkitemsToCount.filterProperty('isDone', false).length;
}.property('checkitems.@each.isDone').cacheable()

...导致 ember-data 的内置 DS.RESTAdapter 调用 findMany。findMany 请求应该接受一个 id 数组,而是将一个包含一个完整检查项对象的数组传递给它,该数组嵌套在一个散列中,键为 0。

解决方案

最后,我将问题追溯到 ember-data 深处的以下观察者:

dataDidChange: Ember.observer(function() {
    var associations = get(this.constructor, 'associationsByName'),
        data = get(this, 'data'), store = get(this, 'store'),
        idToClientId = store.idToClientId,
        cachedValue;

    associations.forEach(function(name, association) {
      if (association.kind === 'hasMany') {
        cachedValue = this.cacheFor(name);

        if (cachedValue) {
          var ids = data.get(name) || [];
          var clientIds = Ember.ArrayUtils.map(ids, function(id) {
            return store.clientIdForId(association.type, id);
          });

          set(cachedValue, 'content', Ember.A(clientIds));
          cachedValue.fetch();
        }
      }
    }, this);
  }, 'data')

当观察者到达 linereturn store.clientIdForId(association.type, id)时,该数组ids是一个 checkitem 对象数组,而不是一个 id 整数数组。修复非常简单:return store.clientIdForId(association.type, id.id)返回一个 id 整数数组。

4

1 回答 1

2

我根据您的描述创建了一个 JSFiddle,但无法重现您的问题。我正在使用 Ember.js 0.9.6 和最新版本的 ember-data,请参阅http://jsfiddle.net/pangratz666/dGjyR/

车把

<script type="text/x-handlebars" data-template-name="checklist" >
    {{#each checklists}}
      {{this.name}}
      remaining: {{this.remainingItemsCount}}
      {{#each checkitems}}
        {{view Ember.Checkbox valueBinding="isDone"}}
      {{/each}}
      <a {{action "addCheckitem"}} class="clickable">add item</a>
      <hr/>
    {{/each}}

</script>​

JavaScript

App = Ember.Application.create({});

App.Checkitem = DS.Model.extend({
    isDone: DS.attr('boolean')
});

App.Checklist = DS.Model.extend({
    name: DS.attr('string'),

    checkitems: DS.hasMany('App.Checkitem', {
        embedded: true
    }),

    remainingItemsCount: function() {
        var checkitemsToCount = this.get('checkitems');
        return checkitemsToCount.filterProperty('isDone', false).get('length');
    }.property('checkitems.@each.isDone').cacheable()
});

App.store = DS.Store.create({
    revision: 4
});

App.checklistsController = Ember.ArrayProxy.create({
    content: App.store.find(App.Checklist)
});

Ember.View.create({
    templateName: 'checklist',
    checklistsBinding: 'App.checklistsController',
    addCheckitem: function(evt) {
        var checklist = evt.context;
        checklist.get('checkitems').addObject(App.Checkitem.createRecord({
            isDone: false
        }));
    }
}).append();


var checklist = App.Checklist.createRecord({
    name: 'firstChecklist'
});

App.Checklist.createRecord({
    name: 'secondChecklist'
});

checklist.get('checkitems').addObject(App.Checkitem.createRecord({
    isDone: false
}));
checklist.get('checkitems').addObject(App.Checkitem.createRecord({
    isDone: true
}));
checklist.get('checkitems').addObject(App.Checkitem.createRecord({
    isDone: true
}));​
于 2012-04-06T20:46:51.397 回答