- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
(已针对ember-data API Rev 11更新...)
TL; DR
使用DS.Adapter.findAssociation(...)
DS.Adapter.findHasMany(...)
按需加载hasMany
关联的正确方法是什么?尤其是,一旦加载了子记录,如何处理从服务器重新加载父记录会清空hasMany数组的事实?我不想(也许不能)在父级数组中包含子级记录的ID。还是有另一种我想念的方式?
附带说明一下,我对于应该在键链接的hasMany
/ belongsTo
定义中传递哪些选项感到困惑(如果我没有侧面加载的数据或ID数组,我应该使用映射吗?),所以如果您认为我认为问题可能出在我的关联定义中,很可能您是对的。
长版
我正在编写自己的DS.RESTAdapter
子类,以将ember-data绑定(bind)到ASP.NET WebAPI后端(使用实体框架)。到目前为止,一切都很好,但是我很难过一段时间才能使协会正常工作。
与this poster相似,我注意到ember-data的首页上曾经说过,如果您的模型中具有hasMany
关联,并且您对该属性进行get
,则存储将发出对子记录的请求。从页面引用:
If you were to request the profile, like this:
author.get('profile');
…the REST adapter would send a request to the URL /profiles?author_id=1.
findAssociation
方法,在版本11中,
findHasMany
方法,我猜这可能是用来实现此目的的方法,现在我正在尝试使用。
find(...)
从存储中检索单个记录,但是我也不知道如何使用关联也可以。 findAssociation
findHasMany
DS.Adapter.findAssociation
DS.Adapter.findHasMany
方法来检索
hasMany
关联的子记录。这主要起作用,但是需要一些体操。这是我广义的
findAssociation
findHasMany
方法:
findHasMany: function (store, record, relationship, ids) {
var adapter = this;
var root = this.rootForType(relationship.type);
var query = relationship.options.query(record);
var hits = store.findQuery(relationship.type, query);
hits.on('didLoad', function () {
// NOTE: This MUST happen in the callback, because findHasMany is in
// the execution path for record.get(relationship.key)!!! Otherwise causes
// infinite loop!!!
var arrMany = record.get(relationship.key);
if (hits.get('isLoaded')) {
arrMany.loadingRecordsCount = 1 + hits.get('length') + (typeof arrMany.loadingRecordsCount == "number" ? arrMany.loadingRecordsCount : 0);
hits.forEach(function (item, index, enumerable) {
arrMany.addToContent(item);
arrMany.loadedRecord();
});
arrMany.loadedRecord(); // weird, but this and the "1 +" above make sure isLoaded/didLoad fires even if there were zero results.
}
});
}
hasMany
定义设置了
query
选项值,该值是记录中的一个函数,该函数返回子项请求中查询字符串的参数哈希。由于这是针对ASP.NET WebAPI后端的,因此它可能是OData过滤器,例如:
App.ParentEntity = DS.Model.extend({
...
children: DS.hasMany('App.ChildEntity', {
query: function (record) {
return {
"$filter": "ChildForeignKey eq '" + record.get('id') + "'"
};
}
})
});
ManyArray
将项目添加到
addToContent(item)
中,以便父记录不会像已被编辑一样被标记为“脏”。另一个是,当我最初检索父记录的JSON时,必须为关联名称的键手动设置
true
的值(来自服务器的JSON根本没有键)。即:
var a = Ember.isArray(json) ? json : [json];
for (var i = 0; i < a.length; i++) {
type.eachAssociation(function (key) {
var meta = type.metaForProperty(key);
a[i][key] = true;
});
}
DS.Store.findMany
的实现并找到
findAssociation
findHasMany
的调用位置,则会发现:
findMany: function(type, ids, record, relationship){
...
if (!Ember.isArray(ids)) {
var adapter = this.adapterForType(type);
if (adapter && adapter.findHasMany) { adapter.findHasMany(this, record, relationship, ids); }
else { throw fmt("Adapter is either null or does not implement `findMany` method", this); }
return this.createManyArray(type, Ember.A());
}
hasAssociation
hasRelationship
中调用
findMany
的那一行来看,您将看到第二个参数传递了什么:
relationship = store.findMany(type, ids || [], this, meta);
findAssociation
findHasMany
的唯一方法是使JSON中的值是“真实的”东西,而不是“数组”-我使用
true
。我认为这可能是错误/未完成,或者是表明我走错了轨道-如果有人可以告诉我,那也很好。
http://myserver.net/api/child_entity/$filter=ChildForeignKey eq '42'
并起作用-子记录被加载,并且它们与父记录相关联(顺便说一句,尽管我没有明确地触摸它,但
belongsTo
的反向关系也得到了适当填充-我不知道在何处或如何发生)。
ManyArray
的
isLoaded
属性仍然是
true
!所以我什至没有观察到任何东西可以重载孩子。
App.store.find('App.ParentEntity', 42)
时,将从存储中加载记录,而无需请求服务器,当然也没有子记录。
最佳答案
基于最新的Ember数据(截至2013年1月25日)...这是我对hasMany延迟加载的解决方案。我修改了DS.hasMany
并在DS.Adapter
中添加了一个方法。
我在DS.hasMany
中更改了两行:
DS.hasMany = function(type, options) {
Ember.assert("The type passed to DS.hasMany must be defined", !!type);
return (function(type, options) {
options = options || {};
var meta = { type: type, isRelationship: true, options: options, kind: 'hasMany' };
return Ember.computed(function(key, value) {
var data = get(this, 'data').hasMany,
store = get(this, 'store'),
ids, relationship;
if (typeof type === 'string') {
type = get(this, type, false) || get(Ember.lookup, type);
}
meta.key = key;
ids = data[key];
relationship = store.findMany(type, ids, this, meta);
set(relationship, 'owner', this);
set(relationship, 'name', key);
return relationship;
}).property().meta(meta);
})(type, options);
};
key
添加到
meta
对象中...
meta.key = key;
findMany
调用中删除了空数组。
relationship = store.findMany(type, ids || [], this, meta);
relationship = store.findMany(type, ids, this, meta);
ids
作为
findMany
传递到
undefined
中。
didFindHasMany
添加了
DS.Adapter
钩子(Hook):
DS.Adapter.reopen({
/**
Loads the response to a request for records by findHasMany.
Your adapter should call this method from its `findHasMany`
method with the response from the backend.
@param {DS.Store} store
@param {subclass of DS.Model} type
@param {any} payload
@param {subclass of DS.Model} record the record of which the relationship is a member (parent record)
@param {String} key the property name of the relationship on the parent record
*/
didFindHasMany: function(store, type, payload, record, key) {
var loader = DS.loaderFor(store);
loader.populateArray = function(references) {
store.loadHasMany(record, key, references.map(function(reference) { return reference.id; }));
};
get(this, 'serializer').extractMany(loader, payload, type);
}
});
DS.Adapter
的
didFindQuery
钩子(Hook)之后使用我发现已经在
loadHasMany
上实现的
DS.Store
对此建模。然后,在我的自定义适配器中,我实现了
findHasMany
方法,该方法在其成功回调中使用以下代码:
Ember.run(this, function() {
adapter.didFindHasMany(store, type, response.data, record, key);
});
关于asp.net-web-api - ember-data:按需加载hasMany关联,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13699796/
我是一名优秀的程序员,十分优秀!