使用可观察数组进行敲除验证

Knockout validation with the observable arrays

我有一个用户输入不同用户详细信息的场景。我为每个用户的不同字段设置了文本框。如果用户必须至少填写一个字段而不是每一行的所有字段,我需要让用户保存详细信息。任何行都不应完全为空。这是 fiddle 我正在尝试

保存时,我只需要要求用户在每个 row.Instead 询问所有必填字段时只输入一个值。 http://jsfiddle.net/KHFn8/7161/

var NameModel = function(names) {
  var self = this;
  self.names = ko.observableArray(names);
  self.addName = function() {
    self.names.push(new xyz());
  };
  self.removeName = function(name) {
    self.names.remove(name);
  };
  if (names != null)
    self.names(names);
};
var xyz = function() {
  var self = this;
  self.FirstName = ko.observable().extend({
    required: true
  });
  self.MiddleName = ko.observable().extend({
    required: true
  });
  self.LastName = ko.observable().extend({
    required: true
  });
  self.Gender = ko.observable().extend({
    required: true
  });
  self.NameSuffix = ko.observable().extend({
    required: true
  });
};
ko.validation.init({ 
    grouping: { deep: true, observable: true, live: true }, 
    messagesOnModified: false
});
ViewModel = function() {    
    this.nameModel = new NameModel();
    this.save = function() {
    if (
    this.errors().length == 0)
           alert('Everything is valid, we can save!');
       else if (
       !this.nameModel.isValid())
           alert('You must have at least one valid item ');
     }
     };
ko.applyBindings(new ViewModel());


<div data-bind="with: nameModel" class="panel-body">
  <table class="table table-hover table-responsive">
    <thead>
      <tr>
        <th>FirstName</th>
        <th>MiddleName</th>
        <th>LastName</th>
        <th>Gender</th>
        <th>NameSuffix</th>
      </tr>
    </thead>
    <tbody data-bind="foreach: names">
      <tr>
        <td>
          <input class="form-control" data-bind="value: FirstName" />
        </td>
        <td>
          <input class="form-control" data-bind="value: MiddleName" />
        </td>
        <td>
          <input class="form-control" data-bind="value: LastName" />
        </td>
        <td>
          <input class="form-control" data-bind="value: Gender" />
        </td>
        <td>
          <input class="form-control" data-bind="value: NameSuffix" />
        </td>
        <td style="vertical-align: inherit;"><a href="#" data-bind="click:$parent.removeName.bind($data)">Delete</a></td>
      </tr>
    </tbody>
  </table>
  <button class="btn btn-success btn-sm" data-bind="click: addName.bind($data)">Add Name</button>
</div>
<button id="saveButton" type="button" class="btn btn-primary btn-company pull-left" data-bind="click: save">
    Save
    <span class="glyphicon glyphicon-download-alt"> </span>
</button>

这里有一些用户贡献的 KO 验证规则:

https://github.com/Knockout-Contrib/Knockout-Validation/wiki/User-Contributed-Rules

这包括 'requires One Of' 的规则,它将执行您想要的操作。但是,我发现该页面上的最新版本有点错误,所以我使用了一个项目的版本,并将其应用到您的 fiddle。结果在这里:

http://jsfiddle.net/4f8s93k4/1/

插件在 JS 的顶部,看起来像:

ko.validation.rules['requiresOneOf'] = {
    getValue: function (o) {
        return (typeof o === 'function' ? o() : o);
    },
    validator: function (val, fields) {
        var self = this;
        var result = false;
        ko.utils.arrayForEach(fields, function (field) {
            var val = self.getValue(field);
            if (val) {
                result = true;
            }
        });

        return result;
    },
    message: 'You must fill in at least one of these fields'
};

我没有将它应用于您的每一个可观察对象,而是使用单个验证消息来应用它,只是为了稍微减少代码:

self.requiresOneValidation  = ko.observable().extend({
    requiresOneOf: [self.FirstName, self.MiddleName, self.LastName, self.Gender, self.NameSuffix]
})

这会出现在您的 HTML 中,如下所示:

    <tr>
      <td colspan="6">
        <span data-bind="validationMessage: requiresOneValidation"></span>
      </td>
    </tr>