1

我正在尝试创建显示初始数据的可编辑表单,当用户进入编辑模式时,会显示 Kendo ComboBoxes 以选择新值。

只有当用户启动编辑模式时,组合框的所有值才会按需加载(不需要为了查看当前状态而向客户端获取数据)。

换句话说 - 表单显示为只读输入。点击“编辑”按钮。这将调用服务来加载数据(不是在小提琴中,只是辅助方法)并显示按特定 ID 过滤的组合框,用于 foreach 循环中的每个项目。因此,每个 ComboBox 仅根据过滤器显示适当的值。

如何创建 GetBrokersInRole() 的工作版本并将其绑定到表单数据?

谢谢

标记:

<div data-bind="foreach: Roles">
    <label data-bind="text: RoleName"></label>
    <div data-bind="visible: !$root.IsEditMode()">
        <input data-bind="enable: $root.IsEditMode(), value: ValueText" />
    </div>
    <div data-bind="visible: $root.IsEditMode()">
        <!-- OK -->
        <input data-bind="kendoComboBox: { data: $root.BrokersInRoles, dataTextField: 'BrokerName', dataValueField: 'BrokerID', value: ValueID }" />
        <!-- NOT OK -->
        <input data-bind="kendoComboBox: { data: $root.GetBrokersInRole($data), dataTextField: 'BrokerName', dataValueField: 'BrokerID', value: ValueID }" />
    </div>
</div>
<p>
    <button data-bind="click: StartEdit">Set Edit Mode</button>
</p>
<label data-bind="text: Brokers().length"></label>

视图模型:

// Roles and initial values
var roles = [
    {"RoleID":1,"RoleName":"Role 1","ValueID":101,"ValueText":"Broker 1"},
    {"RoleID":2,"RoleName":"Role 2","ValueID":102,"ValueText":"Broker 2"},
    {"RoleID":3,"RoleName":"Role 3","ValueID":103,"ValueText":"Broker 3"},
    {"RoleID":4,"RoleName":"Role 4","ValueID":104,"ValueText":"Broker 4"}
];

// RolesID is collection of Roles where Broker is available
// so BrokerID=101 is available in ComboBox for RoleID 1 and 3
var brokers = [
    {"BrokerID":101,"BrokerName":"Broker 1","RolesID":"1;3"},
    {"BrokerID":102,"BrokerName":"Broker 2","RolesID":"1;2"},
    {"BrokerID":103,"BrokerName":"Broker 3","RolesID":"3"},
    {"BrokerID":104,"BrokerName":"Broker 4","RolesID":"4"}
];

// ViewModel
var ViewModel = function() {

    // Roles and initial values
    this.Roles = ko.observableArray(roles);
    // Initial value is empty - no comboboxes if not in edit mode (save data)
    this.BrokersInRoles = ko.observableArray([]);  
    // Is edit mode?
    this.IsEditMode = ko.observable(false);

    // Alternative method to AJAX load of Brokers collection
    // Until edit mode is active, BrokersInRoles collection is empty
    this.StartEdit = function() 
    { 
        var self = this;
        self.BrokersInRoles(brokers); // This will be AJAX load on demand
        self.IsEditMode(true); 
    };

    // Filter brokers based on role
    this.GetBrokersInRole = function (roleItem) {
        var self = this;

        var filtered = ko.utils.arrayFilter(self.BrokersInRoles(), function (broker) {
            return _.contains(broker.RolesID().split(';'), roleItem.RoleTypeID().toString());
        });

        // Return only brokers in selected role
        return ko.observableArray(filtered);
    };
};

ko.applyBindings(new ViewModel());

小提琴:http: //jsfiddle.net/RaptorCZ/drkb6mLk/

4

1 回答 1

0

我看到两件事可以帮助解决这个问题。首先,Knockout-Kendo 没有设置为跟踪对不可观察或计算的事物的依赖关系(比如返回值的函数)。所以,它没有被重新评估。

您可以在您的每一个上创建一个计算Roles来表示该数据,或者根据您的代码,您可以包装kendoComboBox如下:

    <!-- ko with: $root.GetBrokersInRole($data) -->
    <input data-bind="kendoComboBox: { data: $data, dataTextField: 'BrokerName', dataValueField: 'BrokerID', value: $parent.ValueID }" />
    <!-- /ko -->

GetBrokersInRole现在,当访问的依赖项被更新时,它将被正确地重新渲染。

此外,您的过滤逻辑似乎有点偏离(至少对于您小提琴中的代码)。您角色的属性是不可观察的,并且没有RoleTypeID. 也许是这样:

var filtered = ko.utils.arrayFilter(self.BrokersInRoles(), function (broker) {
       return _.contains(broker.RolesID.split(';'), roleItem.RoleID.toString());
});

更新小提琴:http: //jsfiddle.net/rniemeyer/kz5o5h5y/

于 2015-04-08T14:02:00.120 回答