Ember路由器:异步模型(承诺?)

时间:2013-02-11 14:42:19

标签: ember.js ember-router

[这是关于新的1.0.0-pre.4 +路由器。]

我想从Ember Route的model方法返回需要异步回调加载的记录,例如因为它要求我们加载多个(嵌套)模型。最好的方法是什么?


以下是假设博客应用中的示例代码,用于说明问题:

App.Router.map ->
  @resource 'filteredArticles', path: '/:filter'

App.FilteredArticlesRoute = Ember.Route.extend
  model: (params) ->
    blog = App.Blog.find(1) # get the user's Blog singleton
    property = switch params.filter
      when 'published' then 'publishedArticles'
      when 'draft' then 'drafts'
      when 'all' then 'articles'
    # Return the list of articles from the `blog` record.
    # But `blog` hasn't necessarily finished loading :(
    blog.get(property)

2 个答案:

答案 0 :(得分:4)

我正在重写Travis CI到最新的ember版本,我遇到了同样的问题 - 我们通过slug(例如emberjs/ember.js)获取存储库,这不是主键。我的解决方案涉及使用Ember.ProxyObject

当有人进入/emberjs/ember.js之类的路径时,参数将如下所示:

{ owner: 'emberjs', name: 'ember.js` }

因此slug将等于emberjs/ember.js

有了这些信息,我创建了一个简单的Ember对象,它只保留slugisLoaded属性:

content = Ember.Object.create slug: slug, isLoaded: false

然后我用这个对象创建一个代理作为内容:

proxy = Ember.ObjectProxy.create(content:content)

现在我可以使用slug从服务器加载记录并返回代理作为模型。当我从服务器获取记录时,我只是将代理内容设置为实际记录。

完整的解决方案在这里:

deserialize: (params) ->
  slug = "#{params.owner}/#{params.name}"
  content = Ember.Object.create slug: slug, isLoaded: false
  proxy = Ember.ObjectProxy.create(content: content)

  repos = Travis.Repo.bySlug(slug)

  observer = ->
    if repos.get 'isLoaded'
      repos.removeObserver 'isLoaded', observer
      proxy.set 'content', repos.objectAt(0)

  if repos.length
    proxy.set('content', repos[0])
  else
    repos.addObserver 'isLoaded', observer

  proxy

您还可以查看代码的其余部分on github

答案 1 :(得分:0)

如何在模型本身中添加观察者,在模型的isLoaded状态,然后调用blog.get(property)

blogReady: function() {
  if(this.get('isLoaded') {
     // switch logic
  }
}.observes('isLoaded')