从 jQuery Validate 1.11.1(甚至更旧)开始,接受的答案不起作用。另外,这个问题没有简单的答案,解决方案需要在 jQuery Validation 中添加自定义验证方法。
事实上,简单的答案可能就是:不要打电话valid()
如果您只想提交表单,请手动进行。只需让验证插件为您完成即可。在内部,它将等待所有异步请求完成,然后才允许提交表单。仅当您手动检查时才会出现此问题valid()
or element()
.
但是,您可能需要这样做的原因有很多。例如,我正在处理的页面需要在启用表单的其余部分之前使用远程验证器检查字段的有效性。我可以手动完成,而不是使用 jQuery 验证,但这是重复的工作。
那么,为什么要设置async: false
不行?如果将 async 设置为 false,则请求will同步进行,但是,该插件无法正确处理此问题。内置的remote
函数总是返回"pending"
这将导致valid()
返回的函数true
即使请求已经完成并收到错误的响应!它直到稍后才检查响应的值或显示错误。
解决方案制作valid()
and element()
使用同步回调时的同步行为是添加自定义验证方法。我自己尝试过这个,看起来效果很好。您只需从常规远程验证中复制源代码并修改它以处理同步ajax调用,并且默认情况下是同步的。
v1.11.1中的远程函数的源代码从jquery.validate.js的第1112行开始:
remote: function( value, element, param ) {
if ( this.optional(element) ) {
return "dependency-mismatch";
}
var previous = this.previousValue(element);
if (!this.settings.messages[element.name] ) {
this.settings.messages[element.name] = {};
}
previous.originalMessage = this.settings.messages[element.name].remote;
this.settings.messages[element.name].remote = previous.message;
param = typeof param === "string" && {url:param} || param;
if ( previous.old === value ) {
return previous.valid;
}
previous.old = value;
var validator = this;
this.startRequest(element);
var data = {};
data[element.name] = value;
$.ajax($.extend(true, {
url: param,
mode: "abort",
port: "validate" + element.name,
dataType: "json",
data: data,
success: function( response ) {
validator.settings.messages[element.name].remote = previous.originalMessage;
var valid = response === true || response === "true";
if ( valid ) {
var submitted = validator.formSubmitted;
validator.prepareElement(element);
validator.formSubmitted = submitted;
validator.successList.push(element);
delete validator.invalid[element.name];
validator.showErrors();
} else {
var errors = {};
var message = response || validator.defaultMessage( element, "remote" );
errors[element.name] = previous.message = $.isFunction(message) ? message(value) : message;
validator.invalid[element.name] = true;
validator.showErrors(errors);
}
previous.valid = valid;
validator.stopRequest(element, valid);
}
}, param));
return "pending";
}
请注意,即使 ajax 调用已完成,它也总是返回“pending”。
要解决此问题,请进行以下修改:
- 移动声明
valid
ajax 调用和 success 函数之外的变量以创建闭包,并为其分配默认值“pending”。
- 更改旧的声明
valid
变量进行赋值。
- 返回
valid
变量而不是常量“pending”。
这是插件的完整代码。只需将其保存为 js 文件,并在包含 jQuery 验证后将其包含在页面或模板中:
//Created for jQuery Validation 1.11.1
$.validator.addMethod("synchronousRemote", function (value, element, param) {
if (this.optional(element)) {
return "dependency-mismatch";
}
var previous = this.previousValue(element);
if (!this.settings.messages[element.name]) {
this.settings.messages[element.name] = {};
}
previous.originalMessage = this.settings.messages[element.name].remote;
this.settings.messages[element.name].remote = previous.message;
param = typeof param === "string" && { url: param } || param;
if (previous.old === value) {
return previous.valid;
}
previous.old = value;
var validator = this;
this.startRequest(element);
var data = {};
data[element.name] = value;
var valid = "pending";
$.ajax($.extend(true, {
url: param,
async: false,
mode: "abort",
port: "validate" + element.name,
dataType: "json",
data: data,
success: function (response) {
validator.settings.messages[element.name].remote = previous.originalMessage;
valid = response === true || response === "true";
if (valid) {
var submitted = validator.formSubmitted;
validator.prepareElement(element);
validator.formSubmitted = submitted;
validator.successList.push(element);
delete validator.invalid[element.name];
validator.showErrors();
} else {
var errors = {};
var message = response || validator.defaultMessage(element, "remote");
errors[element.name] = previous.message = $.isFunction(message) ? message(value) : message;
validator.invalid[element.name] = true;
validator.showErrors(errors);
}
previous.valid = valid;
validator.stopRequest(element, valid);
}
}, param));
return valid;
}, "Please fix this field.");
我已经用自己的形式对此进行了测试,效果很好。在启用表单的其余部分之前,我可以测试我的元素的有效性。但是,您可能想要设置onkeyup: false
以防止在每次按键时执行同步回调。我也喜欢用onfocusout: false
.
要使用它,只需将验证设置中的“remote”替换为“synchronousRemote”即可。例如:
$("#someForm").validate({
rules: {
someField: {
required: true,
synchronousRemote: {
url: "/SomePath/ValidateSomeField"
//notice that async: false need not be specified. It's the default.
}
}
},
messages: {
someField: {
required: "SomeField is required.",
synchronousRemote: "SomeField does not exist."
}
},
onkeyup: false,
onfocusout: false
});