1

我们无法在 Ember.js 中将一个数组控制器的内容与另一个数组控制器的内容链接起来。我们已经搜索了 Stackoverflow,但似乎没有任何解决方案适合我们。

场景很简单。我们有一个名为“Campaign”的模型。该模型非常基本,看起来像这样......

竞选模式:

App.Campaign = DS.Model.extend({

    name: DS.attr('string'),
    description: DS.attr('string'),
    opts: DS.attr('string'),
    starred: DS.attr('boolean'),
    keywords: DS.hasMany('App.Keyword')

});

// Create dummy campaign array

App.Campaign.FIXTURES = new Array();

// Populate dummy campaign array

...

Campaign 控制器也是基本的。这是一个简单的 ArrayController,里面什么也没发生……

活动索引控制器:

App.CampaignsIndexController = Em.ArrayController.extend({

});

为了这个问题,我们已经将我们的应用程序模板剥离到了基本要素。如下所示,应用程序模板有一个侧边栏。侧边栏中有两个视图。一个视图用于网站导航,另一个视图用于“已加星标的广告系列”。此视图将包含整个活动集合的一个子集。

申请模板:

<!-- ================================================== -->
<!-- ======================= APP ====================== -->
<!-- ================================================== -->

<div id="app" class="">

    <!-- ================================================== -->
    <!-- ==================== SIDEBAR ===================== -->
    <!-- ================================================== -->

    <div id="sidebar">

        {{ view App.NavigationView }}

        {{ view App.StarredCampaignsView controller="App.StarredCampaignsController" }}

    </div>

    <!-- ================================================== -->
    <!-- ====================== PAGE ====================== -->
    <!-- ================================================== -->

    <div id="page" class="">

        {{ outlet page }}

    </div>

</div>

我们有一个用于 Starred Campaigns 的控制器。这就是事情开始变得混乱的地方。在我们的控制器中,我们将 ArrayController 的内容绑定到 CampaignsIndexController 的内容,其中包含所有 Campaign 模型的数组。

加星标的活动控制器:

App.StarredCampaignsController = Em.ArrayController.extend({

    content: [],

    contentBinding: Ember.Binding.oneWay('App.CampaignsIndexController.content')

});

对于 Starred Campaigns 的视图也很简单...

已加星标的广告系列视图:

App.StarredCampaignsView = Em.View.extend({

    templateName: 'templates/layout/starred-campaigns',

});

这就是 Starred Campaigns 的模板。它试图遍历 controller.content 中的项目并将它们传递给另一个视图。

已加星标的广告系列模板:

<h5 class="sidebar-title">Starred Campaigns</h5>

<ul class="starred-campaigns clearfix">

    {{# each item in controller.content }}

        {{ view App.StarredCampaignItemView itemBinding="item" }}

    {{/ each }}

</ul>

然而,尽管如此,StarredCampaignsController 中的内容仍然是空的,即使我们填充了由 App.CampaignsIndexController 表示的 App.Campaign 数组。看到我们的方法有什么问题吗?我觉得我们在这里遗漏了一些非常简单的东西。

4

1 回答 1

1

在 Ember 中,对于需要其他控制器的控制器,您需要使用该needs属性。这在 Ember 中总是在变化:在以前的版本中,它是路由器的责任,this.controllerFor甚至在更旧的版本中,这种方法更类似于您似乎试图采用的方法。然而,最终方法早已被弃用,控制器/模型/视图的实例不再App直接保留,只有它们的对象表示。

要直接使用needs,你需要告诉你的App.StarredCampaignsController控制器它需要App.CampaignsIndexController这样的东西:

App.StarredCampaignsController = Em.ArrayController.extend({
    content: [],
    needs: ['campaigns_index']
});

您现在可以访问已加App.CampaignsIndexController星标的广告系列视图中的控制器:{{controllers.campaigns_index}}

参见简单的 JSFiddle:http: //jsfiddle.net/AGJfB/

你的不工作的原因是因为你引用了控制器的对象,而不是它的实例:

Ember.Binding.oneWay('App.CampaignsIndexController.content')

虽然你有一个App.CampaignsIndexControlleron App,但这就是它的对象。this.controllerFor当您通过任何 Ember 方式( 、、、needs通过 URL 导航到它等...)访问它时,Ember 将为您创建实例。

在过去的日子里,我们可以使用:

Ember.Binding.oneWay('App.campaignsIndexController.content')

但是,这是一种糟糕的做事方式,因为您引用的是控制器的绝对路径。自从 Ember pre 1 以来,我从来没有使用过Ember.Binding.oneWay,但仍然会有你需要它的情况。但是,当您发现自己在使用它时,最好问问自己是否有必要,以及是否有更好的方法来做到这一点。

我希望我已经很好地解释了自己。请随时提出任何问题!

于 2013-02-13T02:08:45.433 回答