gpt4 book ai didi

jquery验证不等待远程验证返回true,认为表单有效

转载 作者:行者123 更新时间:2023-12-03 21:41:53 26 4
gpt4 key购买 nike

$("#new_component_form").validate({
errorClass: 'input-error',
rules : {
"comp_data[account_name]" : {
required: true,
remote: {
url: "/validate",
data: {
provider: 'twitter'
}
}
}
},
onsubmit: true,
onfocusout: false,
onkeyup: false,
onclick: false
});



$("#new_component_form").submit(function(){
console.log($(this).valid());

即使该值无效,也会输出 true。我看到验证最终失败并显示错误消息,但表单仍然提交。

最佳答案

从 jQuery Validate 1.11.1(甚至更旧)开始,接受的答案不起作用。另外,这个问题没有简单的答案,解决方案需要在 jQuery Validation 中添加自定义验证方法。

实际上,简单的答案可能就是:如果您只想提交表单,请不要手动调用valid()。只需让验证插件为您完成即可。在内部,它将等待所有异步请求完成,然后才允许提交表单。仅当您手动检查 valid()element() 时,才会出现此问题。

但是,您可能需要这样做的原因有很多。例如,我正在处理的页面需要在启用表单的其余部分之前使用远程验证器检查字段的有效性。我可以手动完成此操作,而不是使用 jQuery 验证,但这是重复工作。

那么,为什么设置async: false不起作用?如果将 async 设置为 false,请求将同步发出,但是插件无法正确处理此问题。内部remote函数始终返回“pending”,这将导致valid()函数返回true即使请求已经完成并收到错误响应!直到稍后它才会检查响应的值或显示错误。

在使用同步回调时使 valid()element() 行为同步的解决方案是添加自定义验证方法。我自己尝试过这个,看起来效果很好。您只需从常规远程验证中复制源代码并修改它以处理同步aj​​ax调用,并且默认情况下是同步的。

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”。

要解决此问题,请进行以下修改:

  1. valid 变量的声明移到 ajax 调用和 success 函数之外以创建闭包,并为其指定默认值“pending”。
  2. valid 变量的旧声明更改为赋值。
  3. 返回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
});

关于jquery验证不等待远程验证返回true,认为表单有效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7247250/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com