script.js
8.61 KB
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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
define([
"../_base/connect", /*===== "../_base/declare", =====*/ "../_base/kernel", "../_base/lang",
"../sniff", "../_base/window","../_base/xhr",
"../dom", "../dom-construct", "../request/script", "../aspect"
], function(connect, /*===== declare, =====*/ kernel, lang, has, win, xhr, dom, domConstruct, _script, aspect){
// module:
// dojo/io/script
kernel.deprecated("dojo/io/script", "Use dojo/request/script.", "2.0");
/*=====
var __ioArgs = declare(kernel.__IoArgs, {
// summary:
// All the properties described in the dojo.__ioArgs type, apply to this
// type as well, EXCEPT "handleAs". It is not applicable to
// dojo/io/script.get() calls, since it is implied by the usage of
// "jsonp" (response will be a JSONP call returning JSON)
// or the response is pure JavaScript defined in
// the body of the script that was attached.
// callbackParamName: String
// Deprecated as of Dojo 1.4 in favor of "jsonp", but still supported for
// legacy code. See notes for jsonp property.
// jsonp: String
// The URL parameter name that indicates the JSONP callback string.
// For instance, when using Yahoo JSONP calls it is normally,
// jsonp: "callback". For AOL JSONP calls it is normally
// jsonp: "c".
// checkString: String
// A string of JavaScript that when evaluated like so:
// "typeof(" + checkString + ") != 'undefined'"
// being true means that the script fetched has been loaded.
// Do not use this if doing a JSONP type of call (use callbackParamName instead).
// frameDoc: Document
// The Document object for a child iframe. If this is passed in, the script
// will be attached to that document. This can be helpful in some comet long-polling
// scenarios with Firefox and Opera.
});
=====*/
var script = {
// summary:
// TODOC
get: function(/*__ioArgs*/ args){
// summary:
// sends a get request using a dynamically created script tag.
var rDfd;
var dfd = this._makeScriptDeferred(args, function(dfd){
rDfd && rDfd.cancel();
});
var ioArgs = dfd.ioArgs;
xhr._ioAddQueryToUrl(ioArgs);
xhr._ioNotifyStart(dfd);
rDfd = _script.get(ioArgs.url, {
timeout: args.timeout,
jsonp: ioArgs.jsonp,
checkString: args.checkString,
ioArgs: ioArgs,
frameDoc: args.frameDoc,
canAttach: function(rDfd){
// sync values
ioArgs.requestId = rDfd.id;
ioArgs.scriptId = rDfd.scriptId;
ioArgs.canDelete = rDfd.canDelete;
return script._canAttach(ioArgs);
}
}, true);
// Run _validCheck at the same time dojo/request/watch runs the
// rDfd.isValid function
aspect.around(rDfd, "isValid", function(isValid){
return function(response){
script._validCheck(dfd);
return isValid.call(this, response);
};
});
rDfd.then(function(){
dfd.resolve(dfd);
}).otherwise(function(error){
dfd.ioArgs.error = error;
dfd.reject(error);
});
return dfd;
},
attach: _script._attach,
remove: _script._remove,
_makeScriptDeferred: function(/*Object*/ args, /*Function?*/ cancel){
// summary:
// sets up a Deferred object for an IO request.
var dfd = xhr._ioSetArgs(args, cancel || this._deferredCancel, this._deferredOk, this._deferredError);
var ioArgs = dfd.ioArgs;
ioArgs.id = kernel._scopeName + "IoScript" + (this._counter++);
ioArgs.canDelete = false;
//Special setup for jsonp case
ioArgs.jsonp = args.callbackParamName || args.jsonp;
if(ioArgs.jsonp){
//Add the jsonp parameter.
ioArgs.query = ioArgs.query || "";
if(ioArgs.query.length > 0){
ioArgs.query += "&";
}
ioArgs.query += ioArgs.jsonp +
"=" + (args.frameDoc ? "parent." : "") +
kernel._scopeName + ".io.script.jsonp_" + ioArgs.id + "._jsonpCallback";
ioArgs.frameDoc = args.frameDoc;
//Setup the Deferred to have the jsonp callback.
ioArgs.canDelete = true;
dfd._jsonpCallback = this._jsonpCallback;
this["jsonp_" + ioArgs.id] = dfd;
}
// Make sure this runs no matter what happens to clean things up if need be
dfd.addBoth(function(value){
if(ioArgs.canDelete){
if(value instanceof Error){
// Set up a callback that will clean things up for timeouts and cancels
script["jsonp_" + ioArgs.id]._jsonpCallback = function(){
// Delete the cached deferred
delete script["jsonp_" + ioArgs.id];
if(ioArgs.requestId){
// Call the dojo/request/script callback to clean itself up as well
kernel.global[_script._callbacksProperty][ioArgs.requestId]();
}
};
}else{
script._addDeadScript(ioArgs);
}
}
});
return dfd; // dojo/_base/Deferred
},
_deferredCancel: function(/*Deferred*/ dfd){
// summary:
// canceller function for xhr._ioSetArgs call.
//DO NOT use "this" and expect it to be script.
dfd.canceled = true;
},
_deferredOk: function(/*Deferred*/ dfd){
// summary:
// okHandler function for xhr._ioSetArgs call.
//DO NOT use "this" and expect it to be script.
var ioArgs = dfd.ioArgs;
//Favor JSONP responses, script load events then lastly ioArgs.
//The ioArgs are goofy, but cannot return the dfd since that stops
//the callback chain in Deferred. The return value is not that important
//in that case, probably a checkString case.
return ioArgs.json || ioArgs.scriptLoaded || ioArgs;
},
_deferredError: function(/*Error*/ error, /*Deferred*/ dfd){
// summary:
// errHandler function for xhr._ioSetArgs call.
console.log("dojo.io.script error", error);
return error;
},
_deadScripts: [],
_counter: 1,
_addDeadScript: function(/*Object*/ ioArgs){
// summary:
// sets up an entry in the deadScripts array.
script._deadScripts.push({id: ioArgs.id, frameDoc: ioArgs.frameDoc});
//Being extra paranoid about leaks:
ioArgs.frameDoc = null;
},
_validCheck: function(/*Deferred*/ dfd){
// summary:
// inflight check function to see if dfd is still valid.
// TODO: why isn't dfd accessed?
//Do script cleanup here. We wait for one inflight pass
//to make sure we don't get any weird things by trying to remove a script
//tag that is part of the call chain (IE 6 has been known to
//crash in that case).
var deadScripts = script._deadScripts;
if(deadScripts && deadScripts.length > 0){
for(var i = 0; i < deadScripts.length; i++){
//Remove the script tag
script.remove(deadScripts[i].id, deadScripts[i].frameDoc);
//Clean up the deferreds
delete script["jsonp_" + deadScripts[i].id];
deadScripts[i].frameDoc = null;
}
script._deadScripts = [];
}
return true;
},
_ioCheck: function(dfd){
// summary:
// inflight check function to see if IO finished.
// dfd: Deferred
var ioArgs = dfd.ioArgs;
//Check for finished jsonp
if(ioArgs.json || (ioArgs.scriptLoaded && !ioArgs.args.checkString)){
return true;
}
//Check for finished "checkString" case.
var checkString = ioArgs.args.checkString;
return checkString && eval("typeof(" + checkString + ") != 'undefined'");
},
_resHandle: function(/*Deferred*/ dfd){
// summary:
// inflight function to handle a completed response.
if(script._ioCheck(dfd)){
dfd.callback(dfd);
}else{
//This path should never happen since the only way we can get
//to _resHandle is if _ioCheck is true.
dfd.errback(new Error("inconceivable dojo.io.script._resHandle error"));
}
},
_canAttach: function(/*===== ioArgs =====*/ ){
// summary:
// A method that can be overridden by other modules
// to control when the script attachment occurs.
// ioArgs: Object
return true;
},
_jsonpCallback: function(/*JSON Object*/ json){
// summary:
// generic handler for jsonp callback. A pointer to this function
// is used for all jsonp callbacks. NOTE: the "this" in this
// function will be the Deferred object that represents the script
// request.
this.ioArgs.json = json;
if(this.ioArgs.requestId){
kernel.global[_script._callbacksProperty][this.ioArgs.requestId](json);
}
}
};
lang.setObject("dojo.io.script", script);
/*=====
script.attach = function(id, url, frameDocument){
// summary:
// creates a new `<script>` tag pointing to the specified URL and
// adds it to the document.
// description:
// Attaches the script element to the DOM. Use this method if you
// just want to attach a script to the DOM and do not care when or
// if it loads.
};
script.remove = function(id, frameDocument){
// summary:
// removes the script element with the given id, from the given frameDocument.
// If no frameDocument is passed, the current document is used.
};
=====*/
return script;
});