How to queue functions within a JS class until an essential ajax request is done
I'm building a javascript object which mainly should send an authentication ajax request before开发者_高级运维 processing any other function on it.
Example:
hello={
say:function(txt){
alert(txt);
},
auth:function(){
... ajax ...
}
}
hello.say("hello world!");
The alert should not be triggered until the authentication ajax request is successfully fetched. It's like queuing any functions to the object until the initialization is done.
P.S. the auth should be automatically fired when the page is fully loaded.
- EDIT
Attempt to use SLaks method:
functionQueue = [];
function exec(func) {
    if (functionQueue){
        functionQueue.push(func);
    }else{ 
        func();
    }
}
hello={
say:function(txt){
alert(txt);
},
auth:function(){
ajax.post("http://www.hello.com",function(resp){
        var queue = functionQueue;
        functionQueue = false;    //Prevent re-entrancy
        for (var i = 0; i < queue.length; i++){
            queue[i]();
        }
});
}
}
function start(){
hello.auth();
}
window.addEventListener("load", start, false);
hello.say("hello world!");
Would absolutely appreciate your help.
Thanks!
Make an array of functions to execute, push each function to the array if the request isn't finished yet, then loop through the array executing functions when the AJAX request finishes.
For example:
var functionQueue = [];
function exec(func) {
    if (functionQueue)
        functionQueue.push(func);
    else    //AJAX request already finished
        func();
}
$.ajax({
    ...
    success: function(...) {
        ...
        var queue = functionQueue;
        functionQueue = false;    //Prevent re-entrancy
        for (var i = 0; i < queue.length; i++)
            queue[i]();
    }
});
EDIT: Demo
2nd EDIT:  To use exec inside an object, it can be useful to preserve this:
var functionQueue = [];
function exec(func, context) {
    if (functionQueue){
        functionQueue.push(function() { func.call(context); });
    } else {
        func.call(context);
    }
}
var hello = {
    say: function(txt){
        exec(function() {
            alert(txt);   //this still works
        }, this);
    },
    auth: function(){
        $.get("/echo/json/", function(resp){
            var queue = functionQueue;
            functionQueue = false;    //Prevent re-entrancy
            for (var i = 0; i < queue.length; i++){
                queue[i]();
            }
        });
    }
};
hello.say("hello world!");
hello.auth();
You might want to put the queue inside your object:
var hello = {
    functionQueue: [],
    exec: function(func) {
        if (this.functionQueue){
            this.functionQueue.push(func);
        } else {
            func.call(this);
        }
    },
    say: function(txt){
        this.exec(function() {
            alert(txt);   //this still works
        });
    },
    auth: function(){
        var me = this;
        $.get("/echo/json/", function(resp){
            alert('AJAX finished!');
            var queue = me.functionQueue;
            functionQueue = false;    //Prevent re-entrancy
            for (var i = 0; i < queue.length; i++){
                queue[i].call(me);
            }
        });
    }
};
hello.say("hello world!");
hello.auth();
 
         加载中,请稍侯......
 加载中,请稍侯......
      
精彩评论