12

我是 knockout.js 的新手,我正在尝试绑定到以下代表用户的对象:

{
    "$id": "1",
    "$values": [
        {
            "$id": "2",
            "Locations": {
                "$id": "3",
                "$values": []
            },
            "Photos": {
                "$id": "4",
                "$values": []
            },
            "UserId": 1,
            "Name": "Test User"
        }
    ]
}

用户可以零个或多个位置,以及零个或多个照片。

视图模型:

function UsersViewModel() {
    var self = this;
    self.users = ko.observableArray();

    var baseUri = 'http://localhost:46241/api/users';

    $.getJSON(baseUri, function (data) {
        self.users = data;
    });
}

$(document).ready(function () {
    ko.applyBindings(new UsersViewModel());
})

HTML 包含以下绑定:

<ul id="update-users" data-bind="foreach: users"> 
     <li> 
            <div><div class="item">User ID</div>
                <input type="text" data-bind="value: $data.UserId" />
            </div>                 
            <div><div class="item">Name</div>
                <input type="text" data-bind="value: $data.Name" />
            </div>                  
     </li> 
</ul>

我做错了吗?或者用户对 Locations 和 Photos 的对象引用可能会破坏绑定?

4

1 回答 1

18

我做错了吗?

是的。您正在覆盖observableArrayJSON 回调中的您,从而将其销毁:

$.getJSON(baseUri, function (data) {
    self.users = data;
});

一般来说,淘汰赛 observables 是这样分配的:

$.getJSON(baseUri, function (data) {
    self.users(data.$values);
});

请注意,在您的情况下,data.$values似乎包含实际数组,而不是data本身。

这适用于您的情况,但它并没有充分发挥淘汰赛的潜力。

正是为这种情况创建了映射插件。它递归地遍历一个复杂对象并将其所有属性转换为可观察对象,并将所有包含的数组转换为可观察数组。这使您可以对对象的状态进行细粒度控制,从而使您可以监视每个更改。

$.getJSON(baseUri, function (data) {
    ko.mapping.fromJS(data.$values, {}, self.users);
});

它还允许部分更新:假设您从服务器检索用户列表,并且您的视图模型已经有一半。映射插件能够仅将那些缺失的视图添加到您的视图模型中,并对现有的进行适当的更改。如果您想了解如何操作,请阅读文档中的“高级用法”部分。


假设您从服务器获取一组用户对象并使用映射插件,那么您的绑定将如下所示:

<ul id="update-users" data-bind="foreach: users"> 
     <li> 
            <div><div class="item">User ID</div>
                <input type="text" data-bind="value: UserId" />
            </div>                 
            <div><div class="item">Name</div>
                <input type="text" data-bind="value: Name" />
            </div>
            <ul class="photos" data-bind="foreach: Photos.$values">
              <!-- ... -->
            <ul>
     </li> 
</ul>

一般来说,@Greg Smith 的评论是正确的。尝试丢失$对象中的键,它们可能会在某些时候与淘汰赛中的特殊变量发生冲突。例如,您可以安全地用下划线替换它们。

更一般地说,如果我是你,我会尝试在你的 JSON 中去掉整个$id/$values概念,它似乎没有任何用途:

[
    {
        "Locations": [],
        "Photos": [],
        "UserId": 1,
        "Name": "Test User"
    }
]
于 2012-11-18T10:34:12.643 回答