-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathreadystate.js
152 lines (131 loc) · 3.62 KB
/
readystate.js
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
145
146
147
148
149
150
151
152
'use strict';
/**
* Generate a new prototype method which will the given function once the
* desired state has been reached. The returned function accepts 2 arguments:
*
* - fn: The assigned function which needs to be called.
* - context: Context/this value of the function we need to execute.
*
* @param {String} state The state we need to operate upon.
* @returns {Function}
* @api private
*/
function generate(state) {
return function proxy(fn, context) {
var rs = this;
if (rs.is(state)) {
setTimeout(function () {
fn.call(context, rs.readyState);
}, 0);
} else {
if (!rs._events[state]) rs._events[state] = [];
rs._events[state].push({ fn: fn, context: context });
}
return rs;
};
}
/**
* RS (readyState) instance.
*
* @constructor
* @api public
*/
function RS() {
this.readyState = RS.UNKNOWN;
this._events = {};
}
/**
* The environment can be in different states. The following states are
* generated:
*
* - ALL: The I don't really give a fuck state.
* - UNKNOWN: We got an unknown readyState we should start listening for events.
* - LOADING: Environment is currently loading.
* - INTERACTIVE: Environment is ready for modification.
* - COMPLETE: All resources have been loaded.
*
* Please note that the order of the `states` string/array is of vital
* importance as it's used in the readyState check.
*
* @type {Number}
* @private
*/
RS.states = 'ALL,UNKNOWN,LOADING,INTERACTIVE,COMPLETE'.split(',');
for (var s = 0, state; s < RS.states.length; s++) {
state = RS.states[s];
RS[state] = RS.prototype[state] = s;
RS.prototype[state.toLowerCase()] = generate(state);
}
/**
* A change in the environment has been detected so we need to change our
* readyState and call assigned event listeners and those of the previous
* states.
*
* @param {Number} state The new readyState that we detected.
* @returns {RS}
* @api private
*/
RS.prototype.change = function change(state) {
state = this.clean(state, true);
var j
, name
, i = 0
, listener
, rs = this
, previously = rs.readyState;
if (previously >= state) return rs;
rs.readyState = state;
for (; i < RS.states.length; i++) {
if (i > state) break;
name = RS.states[i];
if (name in rs._events) {
for (j = 0; j < rs._events[name].length; j++) {
listener = rs._events[name][j];
listener.fn.call(listener.context || rs, previously);
}
delete rs._events[name];
}
}
return rs;
};
/**
* Check if we're currently in a given readyState.
*
* @param {String|Number} state The required readyState.
* @returns {Boolean} Indication if this state has been reached.
* @api public
*/
RS.prototype.is = function is(state) {
return this.readyState >= this.clean(state, true);
};
/**
* Transform a state to a number or toUpperCase.
*
* @param {Mixed} state State to transform.
* @param {Boolean} nr Change to number.
* @returns {Mixed}
* @api public
*/
RS.prototype.clean = function transform(state, nr) {
var type = typeof state;
if (nr) return 'number' !== type
? +RS[state.toUpperCase()] || 0
: state;
return ('number' === type ? RS.states[state] : state).toUpperCase();
};
/**
* Removes all event listeners. Useful when you want to unload readystatechange
* completely so that it won't react to any events anymore. See
* https://github.com/unshiftio/readystate/issues/8
*
* @returns {Function} rs so that calls can be chained.
* @api public
*/
RS.prototype.removeAllListeners = function removeAllListeners() {
this._events = {};
return this;
}
//
// Expose the module.
//
module.exports = new RS();