我想为jQuery.post()提供一些额外的功能。我希望能够检查来自服务器的响应并调用不同的回调。
例如:
$("#frmFoo").postForm("ajax")
.start(function () { showSpinner(); })
.success(function () { alert ("Foo saved"); })
.error(function() { alert ("Foo could not be saved at this time"); })
.validationError(function() { alert("Please fix foo and try again"); })
.complete(function() { hideSpinner(); });
所以我希望能够将一致的JSON对象发送回页面并基于该JSON对象调用success或validationError为validationError提供附加信息。
更新 这是一个陡峭的学习曲线,我几乎100%确定我做错了什么。以下是我对我想要的功能的尝试。
(function ($) {
$.fn.postForm = function (url) {
var options = options || {};
var formData = this.serialize();
var validationCb = jQuery.Callbacks("once memory");
var successCb = jQuery.Callbacks("once memory");
var errorCb = jQuery.Callbacks("once memory");
var completeCb = jQuery.Callbacks("once memory");
var s = jQuery.ajaxSetup({ }, options);
var callbackContext = s.context || s;
foo = {
success: function () {
successCb.add(arguments[0]);
return this;
},
complete: function() {
completeCb.add(arguments[0]);
return this;
},
validationError : function() {
validationCb.add(arguments[0]);
return this;
},
error: function() {
errorCb.add(arguments[0]);
return this;
}
};
$.post(url, formData)
.success(function(data, textStatus, jqXhr) {
if (!data.Success) {
validationCb.fireWith(callbackContext, [data.Message, data.KeyValuePairs]);
} else {
successCb.fireWith(callbackContext, [data, textStatus, jqXhr]);
}
})
.complete(function (jqXhr, textStatus) {
completeCb.fireWith(callbackContext, [jqXhr, textStatus]);
})
.error(function(jqXhr, textStatus, errorThrown) {
errorCb.fireWith(callbackContext, [jqXhr, textStatus, errorThrown]);
});
return foo;
};
})(jQuery);
那么我做得有多糟糕?我不熟悉javascript和jQuery,我确信我打破了一些必然会咬我的规则。
答案 0 :(得分:1)
执行您提出的建议并不是很实际,因为它有效地更改了promise
界面。
请注意,如果您只是将$.post
的结果返回给调用方,那么您想要的大部分内容已经存在:
showSpinner();
$.post({...})
.done(...) // aka .success (deprecated)
.fail(...) // aka .error (deprecated)
.always(stopSpinner) // aka .complete (deprecated)
所以,你必须首先显示微调器,但这几乎没有任何困难,并且无法捕获验证错误,因此需要成为.done
函数的一部分。
答案 1 :(得分:0)
如果我理解你想要达到的目的,你应该使用$.ajax()。
$.ajax({
url: your_url,
type: 'POST'
beforeSend: function(){ showSpinner(); },
error: function() { alert ("Foo could not be saved at this time"); }
success: function (data) {
if (data.validation == true) { //example of validation, depends on how you send data from backend
alert ("Foo saved");
hideSpinner();
} else {
alert("Please fix " + data.foo + " and try again"); //again depends on how you send data
}
}
答案 2 :(得分:0)
您正在寻找$.Deferred(或$.Callbacks,如果您想要真正自定义的内容),或更常见的是promise pattern。一个简单的实现可能如下所示:
(function ($) {
$.fn.postForm = function (url, options) {
var options = options || {},
s = jQuery.ajaxSetup({ }, options),
callbackContext = s.context || s;
var formData = this.serialize();
var post = $.post(url, formData);
var deferred = post.pipe(function(data, textStatus, jqXhr) {
if (!data.Success) {
// HTTP OK, soft error
return $.Deferred().rejectWith(callbackContext, [data.Message, data.KeyValuePairs]);
} else {
// success
return $.Deferred().resolveWith(callbackContext, [data, textStatus, jqXhr]);
}
}, function(jqXhr, textStatus, errorThrown) {
// HTTP error
return $.Deferred().rejectWith(callbackContext, [errorThrown, {}]);
});
deferred.success = deferred.done;
deferred.error = deferred.fail;
deferred.complete = deferred.always;
return deferred;
};
})(jQuery);
这不区分error
/ validationError
,但摆脱了大部分的biolerplate代码。