1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144
|
/*
* This file is part of the MediaWiki extension MultimediaViewer.
*
* MultimediaViewer is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 2 of the License, or
* (at your option) any later version.
*
* MultimediaViewer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with MultimediaViewer. If not, see <http://www.gnu.org/licenses/>.
*/
( function () {
var tqp;
/**
* A queue which holds a list of tasks (functions). The tasks will be executed in order,
* each starting when the previous one has finished (or failed).
*
* @class mw.mmv.model.TaskQueue
* @constructor
*/
function TaskQueue() {
/**
* The list of functions to execute.
*
* @protected
* @property {Array.<function()>}
*/
this.queue = [];
/**
* State of the task queue (running, finished etc)
*
* @protected
* @property {mw.mmv.model.TaskQueue.State}
*/
this.state = TaskQueue.State.NOT_STARTED;
/**
* A deferred which shows the state of the queue.
*
* @protected
* @property {jQuery.Deferred}
*/
this.deferred = $.Deferred();
}
tqp = TaskQueue.prototype;
/**
* Adds a task. The task should be a function which returns a promise. (Other return values are
* permitted, and will be taken to mean that the task has finished already.) The next task will
* start when the promise resolves (or rejects).
*
* Tasks can only be added before the queue is first executed.
*
* @param {function()} task
*/
tqp.push = function ( task ) {
if ( this.state !== TaskQueue.State.NOT_STARTED ) {
throw new Error( 'Task queue already started!' );
}
this.queue.push( task );
};
/**
* Execute the queue. The tasks will be performed in order. No more tasks can be added to the
* queue.
*
* @return {jQuery.Promise} a promise which will resolve when the queue execution is finished,
* or reject when it is cancelled.
*/
tqp.execute = function () {
if ( this.state === TaskQueue.State.NOT_STARTED ) {
this.state = TaskQueue.State.RUNNING;
this.runNextTask( 0, $.Deferred().resolve() );
}
return this.deferred;
};
/**
* Runs the next task once the current one has finished.
*
* @param {number} index
* @param {jQuery.Promise} currentTask
*/
tqp.runNextTask = function ( index, currentTask ) {
var taskQueue = this;
function handleThen() {
if ( !taskQueue.queue[ index ] ) {
taskQueue.state = TaskQueue.State.FINISHED;
taskQueue.queue = []; // just to be sure there are no memory leaks
taskQueue.deferred.resolve();
return;
}
taskQueue.runNextTask( index + 1, $.when( taskQueue.queue[ index ]() ) );
}
if ( this.state !== TaskQueue.State.RUNNING ) {
return;
}
currentTask.then( handleThen, handleThen );
};
/**
* Cancel the queue. No more tasks will be executed.
*/
tqp.cancel = function () {
this.state = TaskQueue.State.CANCELLED;
this.queue = []; // just to be sure there are no memory leaks
this.deferred.reject();
};
/**
* State of the task queue (running, finished etc)
*
* @enum {string} mw.mmv.model.TaskQueue.State
*/
TaskQueue.State = {
/** not executed yet, tasks can still be added */
NOT_STARTED: 'not_started',
/** some task is being executed */
RUNNING: 'running',
/** all tasks finished, queue can be discarded */
FINISHED: 'finished',
/** cancel() function has been called, queue can be discarded */
CANCELLED: 'cancelled'
};
mw.mmv.model.TaskQueue = TaskQueue;
}() );
|