Reputation: 23
$xy('#simpan').click(function() {
$xy('input[id="cekbok[]"]:checked').each(function() {
var data = (this.value);
var div = (this.value);
var str = window.location.href;
var res = str.replace("wp-admin/options-general.php?page=katalogu-options", "/wp-content/plugins/katalog/includes/img/loading.gif");
var loading = ('<img src="'+res+'">') ;
$xy.ajax({
type : 'POST',
url : '../wp-content/plugins/katalogunique/proses2.php',
data: {
id : (this.value)
},
success:function (data) {
$xy('#result'+div).empty();
$xy('#result'+div).append(data);
$xy('#tz'+div).remove();
}
});
});
});
My function sends checkbox values to proses2.php
in a loop but when I run this script it will run all ajax POST calls at once. I want run the ajax requests 1 by 1 or wait until finish, how can I solve this problem?
Upvotes: 2
Views: 715
Reputation: 276286
Here is a way that doesn't have recursion and uses a straightforward loop:
$xy('#simpan').click(function() {
var url = '../wp-content/plugins/katalogunique/proses2.php';
var d = $.Deferred().resolve(); // empty promise
$xy('input[id="cekbok[]"]:checked').each(function() {
var div = this.value;
d = d.then(function(data){
$xy('#result'+div).empty().append(data);
$xy('#tz'+div).remove();
return $xy.post(url, {id: div}); // this will make the next request wait
});
});
// can d.then here for knowing when all of the requests are done.
});
Note I can "clever this up" with a .reduce
shortening the number of lines from 6 to 4, but honestly I'd rather keep the looping construct OP is comfy with. This works because of promise chaining - basically when you return an action from a then
it will wait for it before executing the next then
you're chaining it to.
Let us examplify:
function log(msg){ // simple function to log stuff
document.body.innerHTML += msg + "<br />";
}
var delay = function(ms){ // this is an async request, simulating your AJAX
var d = $.Deferred();
setTimeout(d.resolve, ms); // resolving a deferred makes its then handlers execute
return d;
};
// $.Deferred.resolve() starts a new chain so handlers execute
var p = $.Deferred().resolve().then(function(){
log("1");
return delay(1000); // like in your code, we're waiting for it when we chain
}).then(function(){ // in the above code, this is the d = d.then part,
log("2"); // this only runs when the above delay completes
return delay(1000);
});
// and more like in the above example, we can chain it with a loop:
[3,4,5,6,7,8,9,10].forEach(function(i){
p = p.then(function(){
log(i);
return delay(1000);
});
});
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
Note that doing $xy('#result'+div)
is likely a bad idea since you're querying your view layer for something you put there - consider keeping the relevant divs as an array and keeping references instead :)
Upvotes: 5
Reputation: 382102
You could use a recursive function like this to do all calls one after the other:
$xy('#simpan').click(function() {
var str = window.location.href;
var i=0; things = $xy('input[id="cekbok[]"]:checked');
(function doOneTask(){
if (i++>=things.length) return;
var thing = things[i];
var data = thing.value;
var div = thing.value;
var res = str.replace("wp-admin/options-general.php?page=katalogu-options", "/wp-content/plugins/katalog/includes/img/loading.gif");
var loading = ('<img src="'+res+'">') ;
$xy.ajax({
type : 'POST',
url : '../wp-content/plugins/katalogunique/proses2.php',
data: {
id : thing.value
},
success:function (data) {
$xy('#result'+div).empty();
$xy('#result'+div).append(data);
$xy('#tz'+div).remove();
}
}).always(doOneTask);
})();
});
Note : if you want to stop at the first fail, instead of doing the next calls as I do, replace always
with done
.
Upvotes: 1
Reputation: 93551
One way is to use a recursive function, passing the list (minus the first entry) back each time.
e.g. the calling code is reduced to just this:
$xy('#simpan').click(function() {
LoadAjax($xy('input[id="cekbok[]"]:checked'));
});
and the LoadAjax is recursive like this:
function LoadAjax(elements) {
// Exit when done
if (!elements.length) return;
var $this = elements.first();
var data = ($this.value);
var div = ($this.value);
var str = window.location.href;
var res = str.replace("wp-admin/options-general.php?page=katalogu-options", "/wp-content/plugins/katalog/includes/img/loading.gif");
var loading = ('<img src="' + res + '">');
$xy.ajax({
type: 'POST',
url: '../wp-content/plugins/katalogunique/proses2.php',
data: {
id: ($this.value)
},
success: function (data) {
$xy('#result' + div).empty();
$xy('#result' + div).append(data);
$xy('#tz' + div).remove();
// Go recursive with the rest of the list
LoadAjax(elements.slice(1));
}
});
}
Upvotes: 0