4

我正在使用 Knockout.js 创建一个客人列表应用程序,到目前为止一切进展顺利。但是,我有一个最佳实践问题。我的应用程序有几种不同类型的对象:客人和标签。客人可以有多个标签,标签可以有多个客人。在应用程序的不同点,我需要分别显示两个数组。例如,我有一个“客人”视图,可以看到所有客人及其关联的标签,我还有一个“标签”视图,可以看到所有标签及其关联的客人。目前,我为客人添加标签的代码如下所示:

var tag = function(opts) {
  this.guests = ko.observableArray()

  // Other tag code here...
}

var guest = function(opts) {
  this.tags = ko.observableArray()
  // Other guest code here...

  var self = this

  this.addTag = function(tag) {
    self.tags.push(tag)
    tag.guests.push(self)
  }
}

我知道除了独立更新每个 observableArray 之外,在 Knockout 中必须有更好的方法来处理这种多对多关系。这也导致了一种递归应用程序结构,其中客人有一个标签属性/数组,其中包含一个标签,它有一个客人属性/数组,其中包含一个客人,它有一个标签属性......你明白了。

现在,ViewModel 结构是这样的:

- Parent Object
  - Guests ObservableArray
    - Guest Object
      - Tag Object as property of Guest
  - Tags ObservableArray
    - Tag Object
      - Guest Object as property of Tag

所以我想我的问题是双重的:1)有没有更好的方法来构造我的 ViewModel 以避免递归数组?和 2) 如何更好地使用 Knockout.js 以 DRY 方式更新我的 ViewModel,而不是单独更新标签数组和来宾数组?谢谢!

4

2 回答 2

6

可能还有其他方法可以做到这一点,但是这种方法的重复性非常小,而且不会牺牲适当的建模。服务器以这种格式生成数据应该没有问题。

这是在一个(粗略风格的)小提琴中。请注意,单击标签或访客将导致其下方的选择更新(默认选择第一个)。

基本上,通过 id 存储关系,并使用computed数组来表示关联。这是一个基本的视图模型:

var ViewModel = function(guests, tags) {
    var self = this;
    self.guests = ko.observableArray(
        ko.utils.arrayMap(guests, function(i){ return new Guest(i); }
    ));
    self.tags= ko.observableArray(
        ko.utils.arrayMap(tags, function(i){ return new Tag(i); }
    ));

    self.selectedGuest = ko.observable(self.guests()[0]);
    self.selectedTag = ko.observable(self.tags()[0]);

    self.guestTags = ko.computed(function() {
        return ko.utils.arrayFilter(self.tags(), function(t) {
            return self.selectedGuest().tags().indexOf(t.id()) > -1;
        });        
    });

    self.tagGuests = ko.computed(function() {
        return ko.utils.arrayFilter(self.guests (), function(g) {
            return self.selectedTag().guests().indexOf(g.id()) > -1;
        });        
    });
};

更新

所以我做了一个新的小提琴来演示一种不同的映射,但是这段代码可以很容易地与上面的视图模型共存;它仅用于演示。它不是处理选择,而是提供一般查找,以便任何代码都可以使用它。下面是来自标签的 HTML(来宾是对称的),以及guestMap添加到视图模型中的功能。

您会注意到名称input现在是 s,因此您可以更改名称并观察所有绑定保持最新状态。让我知道你的想法:

<div>Tags
    <ul data-bind="foreach: tags">
        <li>
            <input data-bind="value: name, valueUpdate: 'afterkeydown'" />
            </br><span>Tags</span>
            <ul data-bind="foreach: guests">
                <li><span data-bind="text: $parents[1].guestMap($data).name()"></span></li>
            </ul>
        </li>
    </ul>
</div>

self.guestMap = function(id) {
        return ko.utils.arrayFirst(self.guests(), function(g) {
            return id == g.id();
        });
    };   
于 2012-07-09T23:45:25.473 回答
0

我有同样的问题。显示多对多相关的东西。我必须进行“网格”样式显示并在其中具有更新机制。

我最终复制了我在后端数据库中的结构。两个项目表,中间有连接表。从三个数组中提取数据并不断更新“加入”数组。测试数据和我的测试在下面。

var items = [{name: "T1",id: 1}, {name: "T2",id: 2}, {name: "T3",id: 3}, {name: "T4",id: 4}];
var cats = [{catname: 'C1', catid: 1}, {catname: 'C2',catid: 2}, {catname: 'C3',catid: 3}, {catname: 'C4',catid: 4}];
var catsItems = [{catid:1,itemid:2},{catid:2,itemid:1},{catid:4,itemid:2},{catid:3,itemid:4},{catid:3,itemid:1},{catid:1,itemid:3},{catid:2,itemid:4}];

加入表小提琴

我不确定这种方法在处理大量数据时的效率如何,但做了我需要它做的事情。

于 2013-04-05T17:24:43.580 回答