Upgrade to ExtJS 3.1.1 - Released 02/08/2010
[extjs.git] / src / widgets / chart / swfobject.js
1 /*!
2  * Ext JS Library 3.1.1
3  * Copyright(c) 2006-2010 Ext JS, LLC
4  * licensing@extjs.com
5  * http://www.extjs.com/license
6  */
7 /*! SWFObject v2.2 <http://code.google.com/p/swfobject/> \r
8     is released under the MIT License <http://www.opensource.org/licenses/mit-license.php> \r
9 */\r
10 \r
11 var swfobject = function() {\r
12     \r
13     var UNDEF = "undefined",\r
14         OBJECT = "object",\r
15         SHOCKWAVE_FLASH = "Shockwave Flash",\r
16         SHOCKWAVE_FLASH_AX = "ShockwaveFlash.ShockwaveFlash",\r
17         FLASH_MIME_TYPE = "application/x-shockwave-flash",\r
18         EXPRESS_INSTALL_ID = "SWFObjectExprInst",\r
19         ON_READY_STATE_CHANGE = "onreadystatechange",\r
20         \r
21         win = window,\r
22         doc = document,\r
23         nav = navigator,\r
24         \r
25         plugin = false,\r
26         domLoadFnArr = [main],\r
27         regObjArr = [],\r
28         objIdArr = [],\r
29         listenersArr = [],\r
30         storedAltContent,\r
31         storedAltContentId,\r
32         storedCallbackFn,\r
33         storedCallbackObj,\r
34         isDomLoaded = false,\r
35         isExpressInstallActive = false,\r
36         dynamicStylesheet,\r
37         dynamicStylesheetMedia,\r
38         autoHideShow = true,\r
39     \r
40     /* Centralized function for browser feature detection\r
41         - User agent string detection is only used when no good alternative is possible\r
42         - Is executed directly for optimal performance\r
43     */  \r
44     ua = function() {\r
45         var w3cdom = typeof doc.getElementById != UNDEF && typeof doc.getElementsByTagName != UNDEF && typeof doc.createElement != UNDEF,\r
46             u = nav.userAgent.toLowerCase(),\r
47             p = nav.platform.toLowerCase(),\r
48             windows = p ? /win/.test(p) : /win/.test(u),\r
49             mac = p ? /mac/.test(p) : /mac/.test(u),\r
50             webkit = /webkit/.test(u) ? parseFloat(u.replace(/^.*webkit\/(\d+(\.\d+)?).*$/, "$1")) : false, // returns either the webkit version or false if not webkit\r
51             ie = !+"\v1", // feature detection based on Andrea Giammarchi's solution: http://webreflection.blogspot.com/2009/01/32-bytes-to-know-if-your-browser-is-ie.html\r
52             playerVersion = [0,0,0],\r
53             d = null;\r
54         if (typeof nav.plugins != UNDEF && typeof nav.plugins[SHOCKWAVE_FLASH] == OBJECT) {\r
55             d = nav.plugins[SHOCKWAVE_FLASH].description;\r
56             if (d && !(typeof nav.mimeTypes != UNDEF && nav.mimeTypes[FLASH_MIME_TYPE] && !nav.mimeTypes[FLASH_MIME_TYPE].enabledPlugin)) { // navigator.mimeTypes["application/x-shockwave-flash"].enabledPlugin indicates whether plug-ins are enabled or disabled in Safari 3+\r
57                 plugin = true;\r
58                 ie = false; // cascaded feature detection for Internet Explorer\r
59                 d = d.replace(/^.*\s+(\S+\s+\S+$)/, "$1");\r
60                 playerVersion[0] = parseInt(d.replace(/^(.*)\..*$/, "$1"), 10);\r
61                 playerVersion[1] = parseInt(d.replace(/^.*\.(.*)\s.*$/, "$1"), 10);\r
62                 playerVersion[2] = /[a-zA-Z]/.test(d) ? parseInt(d.replace(/^.*[a-zA-Z]+(.*)$/, "$1"), 10) : 0;\r
63             }\r
64         }\r
65         else if (typeof win.ActiveXObject != UNDEF) {\r
66             try {\r
67                 var a = new ActiveXObject(SHOCKWAVE_FLASH_AX);\r
68                 if (a) { // a will return null when ActiveX is disabled\r
69                     d = a.GetVariable("$version");\r
70                     if (d) {\r
71                         ie = true; // cascaded feature detection for Internet Explorer\r
72                         d = d.split(" ")[1].split(",");\r
73                         playerVersion = [parseInt(d[0], 10), parseInt(d[1], 10), parseInt(d[2], 10)];\r
74                     }\r
75                 }\r
76             }\r
77             catch(e) {}\r
78         }\r
79         return { w3:w3cdom, pv:playerVersion, wk:webkit, ie:ie, win:windows, mac:mac };\r
80     }(),\r
81     \r
82     /* Cross-browser onDomLoad\r
83         - Will fire an event as soon as the DOM of a web page is loaded\r
84         - Internet Explorer workaround based on Diego Perini's solution: http://javascript.nwbox.com/IEContentLoaded/\r
85         - Regular onload serves as fallback\r
86     */ \r
87     onDomLoad = function() {\r
88         if (!ua.w3) { return; }\r
89         if ((typeof doc.readyState != UNDEF && doc.readyState == "complete") || (typeof doc.readyState == UNDEF && (doc.getElementsByTagName("body")[0] || doc.body))) { // function is fired after onload, e.g. when script is inserted dynamically \r
90             callDomLoadFunctions();\r
91         }\r
92         if (!isDomLoaded) {\r
93             if (typeof doc.addEventListener != UNDEF) {\r
94                 doc.addEventListener("DOMContentLoaded", callDomLoadFunctions, false);\r
95             }       \r
96             if (ua.ie && ua.win) {\r
97                 doc.attachEvent(ON_READY_STATE_CHANGE, function() {\r
98                     if (doc.readyState == "complete") {\r
99                         doc.detachEvent(ON_READY_STATE_CHANGE, arguments.callee);\r
100                         callDomLoadFunctions();\r
101                     }\r
102                 });\r
103                 if (win == top) { // if not inside an iframe\r
104                     (function(){\r
105                         if (isDomLoaded) { return; }\r
106                         try {\r
107                             doc.documentElement.doScroll("left");\r
108                         }\r
109                         catch(e) {\r
110                             setTimeout(arguments.callee, 0);\r
111                             return;\r
112                         }\r
113                         callDomLoadFunctions();\r
114                     })();\r
115                 }\r
116             }\r
117             if (ua.wk) {\r
118                 (function(){\r
119                     if (isDomLoaded) { return; }\r
120                     if (!/loaded|complete/.test(doc.readyState)) {\r
121                         setTimeout(arguments.callee, 0);\r
122                         return;\r
123                     }\r
124                     callDomLoadFunctions();\r
125                 })();\r
126             }\r
127             addLoadEvent(callDomLoadFunctions);\r
128         }\r
129     }();\r
130     \r
131     function callDomLoadFunctions() {\r
132         if (isDomLoaded) { return; }\r
133         try { // test if we can really add/remove elements to/from the DOM; we don't want to fire it too early\r
134             var t = doc.getElementsByTagName("body")[0].appendChild(createElement("span"));\r
135             t.parentNode.removeChild(t);\r
136         }\r
137         catch (e) { return; }\r
138         isDomLoaded = true;\r
139         var dl = domLoadFnArr.length;\r
140         for (var i = 0; i < dl; i++) {\r
141             domLoadFnArr[i]();\r
142         }\r
143     }\r
144     \r
145     function addDomLoadEvent(fn) {\r
146         if (isDomLoaded) {\r
147             fn();\r
148         }\r
149         else { \r
150             domLoadFnArr[domLoadFnArr.length] = fn; // Array.push() is only available in IE5.5+\r
151         }\r
152     }\r
153     \r
154     /* Cross-browser onload\r
155         - Based on James Edwards' solution: http://brothercake.com/site/resources/scripts/onload/\r
156         - Will fire an event as soon as a web page including all of its assets are loaded \r
157      */\r
158     function addLoadEvent(fn) {\r
159         if (typeof win.addEventListener != UNDEF) {\r
160             win.addEventListener("load", fn, false);\r
161         }\r
162         else if (typeof doc.addEventListener != UNDEF) {\r
163             doc.addEventListener("load", fn, false);\r
164         }\r
165         else if (typeof win.attachEvent != UNDEF) {\r
166             addListener(win, "onload", fn);\r
167         }\r
168         else if (typeof win.onload == "function") {\r
169             var fnOld = win.onload;\r
170             win.onload = function() {\r
171                 fnOld();\r
172                 fn();\r
173             };\r
174         }\r
175         else {\r
176             win.onload = fn;\r
177         }\r
178     }\r
179     \r
180     /* Main function\r
181         - Will preferably execute onDomLoad, otherwise onload (as a fallback)\r
182     */\r
183     function main() { \r
184         if (plugin) {\r
185             testPlayerVersion();\r
186         }\r
187         else {\r
188             matchVersions();\r
189         }\r
190     }\r
191     \r
192     /* Detect the Flash Player version for non-Internet Explorer browsers\r
193         - Detecting the plug-in version via the object element is more precise than using the plugins collection item's description:\r
194           a. Both release and build numbers can be detected\r
195           b. Avoid wrong descriptions by corrupt installers provided by Adobe\r
196           c. Avoid wrong descriptions by multiple Flash Player entries in the plugin Array, caused by incorrect browser imports\r
197         - Disadvantage of this method is that it depends on the availability of the DOM, while the plugins collection is immediately available\r
198     */\r
199     function testPlayerVersion() {\r
200         var b = doc.getElementsByTagName("body")[0];\r
201         var o = createElement(OBJECT);\r
202         o.setAttribute("type", FLASH_MIME_TYPE);\r
203         var t = b.appendChild(o);\r
204         if (t) {\r
205             var counter = 0;\r
206             (function(){\r
207                 if (typeof t.GetVariable != UNDEF) {\r
208                     var d = t.GetVariable("$version");\r
209                     if (d) {\r
210                         d = d.split(" ")[1].split(",");\r
211                         ua.pv = [parseInt(d[0], 10), parseInt(d[1], 10), parseInt(d[2], 10)];\r
212                     }\r
213                 }\r
214                 else if (counter < 10) {\r
215                     counter++;\r
216                     setTimeout(arguments.callee, 10);\r
217                     return;\r
218                 }\r
219                 b.removeChild(o);\r
220                 t = null;\r
221                 matchVersions();\r
222             })();\r
223         }\r
224         else {\r
225             matchVersions();\r
226         }\r
227     }\r
228     \r
229     /* Perform Flash Player and SWF version matching; static publishing only\r
230     */\r
231     function matchVersions() {\r
232         var rl = regObjArr.length;\r
233         if (rl > 0) {\r
234             for (var i = 0; i < rl; i++) { // for each registered object element\r
235                 var id = regObjArr[i].id;\r
236                 var cb = regObjArr[i].callbackFn;\r
237                 var cbObj = {success:false, id:id};\r
238                 if (ua.pv[0] > 0) {\r
239                     var obj = getElementById(id);\r
240                     if (obj) {\r
241                         if (hasPlayerVersion(regObjArr[i].swfVersion) && !(ua.wk && ua.wk < 312)) { // Flash Player version >= published SWF version: Houston, we have a match!\r
242                             setVisibility(id, true);\r
243                             if (cb) {\r
244                                 cbObj.success = true;\r
245                                 cbObj.ref = getObjectById(id);\r
246                                 cb(cbObj);\r
247                             }\r
248                         }\r
249                         else if (regObjArr[i].expressInstall && canExpressInstall()) { // show the Adobe Express Install dialog if set by the web page author and if supported\r
250                             var att = {};\r
251                             att.data = regObjArr[i].expressInstall;\r
252                             att.width = obj.getAttribute("width") || "0";\r
253                             att.height = obj.getAttribute("height") || "0";\r
254                             if (obj.getAttribute("class")) { att.styleclass = obj.getAttribute("class"); }\r
255                             if (obj.getAttribute("align")) { att.align = obj.getAttribute("align"); }\r
256                             // parse HTML object param element's name-value pairs\r
257                             var par = {};\r
258                             var p = obj.getElementsByTagName("param");\r
259                             var pl = p.length;\r
260                             for (var j = 0; j < pl; j++) {\r
261                                 if (p[j].getAttribute("name").toLowerCase() != "movie") {\r
262                                     par[p[j].getAttribute("name")] = p[j].getAttribute("value");\r
263                                 }\r
264                             }\r
265                             showExpressInstall(att, par, id, cb);\r
266                         }\r
267                         else { // Flash Player and SWF version mismatch or an older Webkit engine that ignores the HTML object element's nested param elements: display alternative content instead of SWF\r
268                             displayAltContent(obj);\r
269                             if (cb) { cb(cbObj); }\r
270                         }\r
271                     }\r
272                 }\r
273                 else {  // if no Flash Player is installed or the fp version cannot be detected we let the HTML object element do its job (either show a SWF or alternative content)\r
274                     setVisibility(id, true);\r
275                     if (cb) {\r
276                         var o = getObjectById(id); // test whether there is an HTML object element or not\r
277                         if (o && typeof o.SetVariable != UNDEF) { \r
278                             cbObj.success = true;\r
279                             cbObj.ref = o;\r
280                         }\r
281                         cb(cbObj);\r
282                     }\r
283                 }\r
284             }\r
285         }\r
286     }\r
287     \r
288     function getObjectById(objectIdStr) {\r
289         var r = null;\r
290         var o = getElementById(objectIdStr);\r
291         if (o && o.nodeName == "OBJECT") {\r
292             if (typeof o.SetVariable != UNDEF) {\r
293                 r = o;\r
294             }\r
295             else {\r
296                 var n = o.getElementsByTagName(OBJECT)[0];\r
297                 if (n) {\r
298                     r = n;\r
299                 }\r
300             }\r
301         }\r
302         return r;\r
303     }\r
304     \r
305     /* Requirements for Adobe Express Install\r
306         - only one instance can be active at a time\r
307         - fp 6.0.65 or higher\r
308         - Win/Mac OS only\r
309         - no Webkit engines older than version 312\r
310     */\r
311     function canExpressInstall() {\r
312         return !isExpressInstallActive && hasPlayerVersion("6.0.65") && (ua.win || ua.mac) && !(ua.wk && ua.wk < 312);\r
313     }\r
314     \r
315     /* Show the Adobe Express Install dialog\r
316         - Reference: http://www.adobe.com/cfusion/knowledgebase/index.cfm?id=6a253b75\r
317     */\r
318     function showExpressInstall(att, par, replaceElemIdStr, callbackFn) {\r
319         isExpressInstallActive = true;\r
320         storedCallbackFn = callbackFn || null;\r
321         storedCallbackObj = {success:false, id:replaceElemIdStr};\r
322         var obj = getElementById(replaceElemIdStr);\r
323         if (obj) {\r
324             if (obj.nodeName == "OBJECT") { // static publishing\r
325                 storedAltContent = abstractAltContent(obj);\r
326                 storedAltContentId = null;\r
327             }\r
328             else { // dynamic publishing\r
329                 storedAltContent = obj;\r
330                 storedAltContentId = replaceElemIdStr;\r
331             }\r
332             att.id = EXPRESS_INSTALL_ID;\r
333             if (typeof att.width == UNDEF || (!/%$/.test(att.width) && parseInt(att.width, 10) < 310)) { att.width = "310"; }\r
334             if (typeof att.height == UNDEF || (!/%$/.test(att.height) && parseInt(att.height, 10) < 137)) { att.height = "137"; }\r
335             doc.title = doc.title.slice(0, 47) + " - Flash Player Installation";\r
336             var pt = ua.ie && ua.win ? "ActiveX" : "PlugIn",\r
337                 fv = "MMredirectURL=" + win.location.toString().replace(/&/g,"%26") + "&MMplayerType=" + pt + "&MMdoctitle=" + doc.title;\r
338             if (typeof par.flashvars != UNDEF) {\r
339                 par.flashvars += "&" + fv;\r
340             }\r
341             else {\r
342                 par.flashvars = fv;\r
343             }\r
344             // IE only: when a SWF is loading (AND: not available in cache) wait for the readyState of the object element to become 4 before removing it,\r
345             // because you cannot properly cancel a loading SWF file without breaking browser load references, also obj.onreadystatechange doesn't work\r
346             if (ua.ie && ua.win && obj.readyState != 4) {\r
347                 var newObj = createElement("div");\r
348                 replaceElemIdStr += "SWFObjectNew";\r
349                 newObj.setAttribute("id", replaceElemIdStr);\r
350                 obj.parentNode.insertBefore(newObj, obj); // insert placeholder div that will be replaced by the object element that loads expressinstall.swf\r
351                 obj.style.display = "none";\r
352                 (function(){\r
353                     if (obj.readyState == 4) {\r
354                         obj.parentNode.removeChild(obj);\r
355                     }\r
356                     else {\r
357                         setTimeout(arguments.callee, 10);\r
358                     }\r
359                 })();\r
360             }\r
361             createSWF(att, par, replaceElemIdStr);\r
362         }\r
363     }\r
364     \r
365     /* Functions to abstract and display alternative content\r
366     */\r
367     function displayAltContent(obj) {\r
368         if (ua.ie && ua.win && obj.readyState != 4) {\r
369             // IE only: when a SWF is loading (AND: not available in cache) wait for the readyState of the object element to become 4 before removing it,\r
370             // because you cannot properly cancel a loading SWF file without breaking browser load references, also obj.onreadystatechange doesn't work\r
371             var el = createElement("div");\r
372             obj.parentNode.insertBefore(el, obj); // insert placeholder div that will be replaced by the alternative content\r
373             el.parentNode.replaceChild(abstractAltContent(obj), el);\r
374             obj.style.display = "none";\r
375             (function(){\r
376                 if (obj.readyState == 4) {\r
377                     obj.parentNode.removeChild(obj);\r
378                 }\r
379                 else {\r
380                     setTimeout(arguments.callee, 10);\r
381                 }\r
382             })();\r
383         }\r
384         else {\r
385             obj.parentNode.replaceChild(abstractAltContent(obj), obj);\r
386         }\r
387     } \r
388 \r
389     function abstractAltContent(obj) {\r
390         var ac = createElement("div");\r
391         if (ua.win && ua.ie) {\r
392             ac.innerHTML = obj.innerHTML;\r
393         }\r
394         else {\r
395             var nestedObj = obj.getElementsByTagName(OBJECT)[0];\r
396             if (nestedObj) {\r
397                 var c = nestedObj.childNodes;\r
398                 if (c) {\r
399                     var cl = c.length;\r
400                     for (var i = 0; i < cl; i++) {\r
401                         if (!(c[i].nodeType == 1 && c[i].nodeName == "PARAM") && !(c[i].nodeType == 8)) {\r
402                             ac.appendChild(c[i].cloneNode(true));\r
403                         }\r
404                     }\r
405                 }\r
406             }\r
407         }\r
408         return ac;\r
409     }\r
410     \r
411     /* Cross-browser dynamic SWF creation\r
412     */\r
413     function createSWF(attObj, parObj, id) {\r
414         var r, el = getElementById(id);\r
415         if (ua.wk && ua.wk < 312) { return r; }\r
416         if (el) {\r
417             if (typeof attObj.id == UNDEF) { // if no 'id' is defined for the object element, it will inherit the 'id' from the alternative content\r
418                 attObj.id = id;\r
419             }\r
420             if (ua.ie && ua.win) { // Internet Explorer + the HTML object element + W3C DOM methods do not combine: fall back to outerHTML\r
421                 var att = "";\r
422                 for (var i in attObj) {\r
423                     if (attObj[i] != Object.prototype[i]) { // filter out prototype additions from other potential libraries\r
424                         if (i.toLowerCase() == "data") {\r
425                             parObj.movie = attObj[i];\r
426                         }\r
427                         else if (i.toLowerCase() == "styleclass") { // 'class' is an ECMA4 reserved keyword\r
428                             att += ' class="' + attObj[i] + '"';\r
429                         }\r
430                         else if (i.toLowerCase() != "classid") {\r
431                             att += ' ' + i + '="' + attObj[i] + '"';\r
432                         }\r
433                     }\r
434                 }\r
435                 var par = "";\r
436                 for (var j in parObj) {\r
437                     if (parObj[j] != Object.prototype[j]) { // filter out prototype additions from other potential libraries\r
438                         par += '<param name="' + j + '" value="' + parObj[j] + '" />';\r
439                     }\r
440                 }\r
441                 el.outerHTML = '<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"' + att + '>' + par + '</object>';\r
442                 objIdArr[objIdArr.length] = attObj.id; // stored to fix object 'leaks' on unload (dynamic publishing only)\r
443                 r = getElementById(attObj.id);  \r
444             }\r
445             else { // well-behaving browsers\r
446                 var o = createElement(OBJECT);\r
447                 o.setAttribute("type", FLASH_MIME_TYPE);\r
448                 for (var m in attObj) {\r
449                     if (attObj[m] != Object.prototype[m]) { // filter out prototype additions from other potential libraries\r
450                         if (m.toLowerCase() == "styleclass") { // 'class' is an ECMA4 reserved keyword\r
451                             o.setAttribute("class", attObj[m]);\r
452                         }\r
453                         else if (m.toLowerCase() != "classid") { // filter out IE specific attribute\r
454                             o.setAttribute(m, attObj[m]);\r
455                         }\r
456                     }\r
457                 }\r
458                 for (var n in parObj) {\r
459                     if (parObj[n] != Object.prototype[n] && n.toLowerCase() != "movie") { // filter out prototype additions from other potential libraries and IE specific param element\r
460                         createObjParam(o, n, parObj[n]);\r
461                     }\r
462                 }\r
463                 el.parentNode.replaceChild(o, el);\r
464                 r = o;\r
465             }\r
466         }\r
467         return r;\r
468     }\r
469     \r
470     function createObjParam(el, pName, pValue) {\r
471         var p = createElement("param");\r
472         p.setAttribute("name", pName);  \r
473         p.setAttribute("value", pValue);\r
474         el.appendChild(p);\r
475     }\r
476     \r
477     /* Cross-browser SWF removal\r
478         - Especially needed to safely and completely remove a SWF in Internet Explorer\r
479     */\r
480     function removeSWF(id) {\r
481         var obj = getElementById(id);\r
482         if (obj && obj.nodeName == "OBJECT") {\r
483             if (ua.ie && ua.win) {\r
484                 obj.style.display = "none";\r
485                 (function(){\r
486                     if (obj.readyState == 4) {\r
487                         removeObjectInIE(id);\r
488                     }\r
489                     else {\r
490                         setTimeout(arguments.callee, 10);\r
491                     }\r
492                 })();\r
493             }\r
494             else {\r
495                 obj.parentNode.removeChild(obj);\r
496             }\r
497         }\r
498     }\r
499     \r
500     function removeObjectInIE(id) {\r
501         var obj = getElementById(id);\r
502         if (obj) {\r
503             for (var i in obj) {\r
504                 if (typeof obj[i] == "function") {\r
505                     obj[i] = null;\r
506                 }\r
507             }\r
508             obj.parentNode.removeChild(obj);\r
509         }\r
510     }\r
511     \r
512     /* Functions to optimize JavaScript compression\r
513     */\r
514     function getElementById(id) {\r
515         var el = null;\r
516         try {\r
517             el = doc.getElementById(id);\r
518         }\r
519         catch (e) {}\r
520         return el;\r
521     }\r
522     \r
523     function createElement(el) {\r
524         return doc.createElement(el);\r
525     }\r
526     \r
527     /* Updated attachEvent function for Internet Explorer\r
528         - Stores attachEvent information in an Array, so on unload the detachEvent functions can be called to avoid memory leaks\r
529     */  \r
530     function addListener(target, eventType, fn) {\r
531         target.attachEvent(eventType, fn);\r
532         listenersArr[listenersArr.length] = [target, eventType, fn];\r
533     }\r
534     \r
535     /* Flash Player and SWF content version matching\r
536     */\r
537     function hasPlayerVersion(rv) {\r
538         var pv = ua.pv, v = rv.split(".");\r
539         v[0] = parseInt(v[0], 10);\r
540         v[1] = parseInt(v[1], 10) || 0; // supports short notation, e.g. "9" instead of "9.0.0"\r
541         v[2] = parseInt(v[2], 10) || 0;\r
542         return (pv[0] > v[0] || (pv[0] == v[0] && pv[1] > v[1]) || (pv[0] == v[0] && pv[1] == v[1] && pv[2] >= v[2])) ? true : false;\r
543     }\r
544     \r
545     /* Cross-browser dynamic CSS creation\r
546         - Based on Bobby van der Sluis' solution: http://www.bobbyvandersluis.com/articles/dynamicCSS.php\r
547     */  \r
548     function createCSS(sel, decl, media, newStyle) {\r
549         if (ua.ie && ua.mac) { return; }\r
550         var h = doc.getElementsByTagName("head")[0];\r
551         if (!h) { return; } // to also support badly authored HTML pages that lack a head element\r
552         var m = (media && typeof media == "string") ? media : "screen";\r
553         if (newStyle) {\r
554             dynamicStylesheet = null;\r
555             dynamicStylesheetMedia = null;\r
556         }\r
557         if (!dynamicStylesheet || dynamicStylesheetMedia != m) { \r
558             // create dynamic stylesheet + get a global reference to it\r
559             var s = createElement("style");\r
560             s.setAttribute("type", "text/css");\r
561             s.setAttribute("media", m);\r
562             dynamicStylesheet = h.appendChild(s);\r
563             if (ua.ie && ua.win && typeof doc.styleSheets != UNDEF && doc.styleSheets.length > 0) {\r
564                 dynamicStylesheet = doc.styleSheets[doc.styleSheets.length - 1];\r
565             }\r
566             dynamicStylesheetMedia = m;\r
567         }\r
568         // add style rule\r
569         if (ua.ie && ua.win) {\r
570             if (dynamicStylesheet && typeof dynamicStylesheet.addRule == OBJECT) {\r
571                 dynamicStylesheet.addRule(sel, decl);\r
572             }\r
573         }\r
574         else {\r
575             if (dynamicStylesheet && typeof doc.createTextNode != UNDEF) {\r
576                 dynamicStylesheet.appendChild(doc.createTextNode(sel + " {" + decl + "}"));\r
577             }\r
578         }\r
579     }\r
580     \r
581     function setVisibility(id, isVisible) {\r
582         if (!autoHideShow) { return; }\r
583         var v = isVisible ? "visible" : "hidden";\r
584         if (isDomLoaded && getElementById(id)) {\r
585             getElementById(id).style.visibility = v;\r
586         }\r
587         else {\r
588             createCSS("#" + id, "visibility:" + v);\r
589         }\r
590     }\r
591 \r
592     /* Filter to avoid XSS attacks\r
593     */\r
594     function urlEncodeIfNecessary(s) {\r
595         var regex = /[\\\"<>\.;]/;\r
596         var hasBadChars = regex.exec(s) != null;\r
597         return hasBadChars && typeof encodeURIComponent != UNDEF ? encodeURIComponent(s) : s;\r
598     }\r
599     \r
600     /* Release memory to avoid memory leaks caused by closures, fix hanging audio/video threads and force open sockets/NetConnections to disconnect (Internet Explorer only)\r
601     */\r
602     var cleanup = function() {\r
603         if (ua.ie && ua.win) {\r
604             window.attachEvent("onunload", function() {\r
605                 // remove listeners to avoid memory leaks\r
606                 var ll = listenersArr.length;\r
607                 for (var i = 0; i < ll; i++) {\r
608                     listenersArr[i][0].detachEvent(listenersArr[i][1], listenersArr[i][2]);\r
609                 }\r
610                 // cleanup dynamically embedded objects to fix audio/video threads and force open sockets and NetConnections to disconnect\r
611                 var il = objIdArr.length;\r
612                 for (var j = 0; j < il; j++) {\r
613                     removeSWF(objIdArr[j]);\r
614                 }\r
615                 // cleanup library's main closures to avoid memory leaks\r
616                 for (var k in ua) {\r
617                     ua[k] = null;\r
618                 }\r
619                 ua = null;\r
620                 for (var l in swfobject) {\r
621                     swfobject[l] = null;\r
622                 }\r
623                 swfobject = null;\r
624             });\r
625         }\r
626     }();\r
627     \r
628     return {\r
629         /* Public API\r
630             - Reference: http://code.google.com/p/swfobject/wiki/documentation\r
631         */ \r
632         registerObject: function(objectIdStr, swfVersionStr, xiSwfUrlStr, callbackFn) {\r
633             if (ua.w3 && objectIdStr && swfVersionStr) {\r
634                 var regObj = {};\r
635                 regObj.id = objectIdStr;\r
636                 regObj.swfVersion = swfVersionStr;\r
637                 regObj.expressInstall = xiSwfUrlStr;\r
638                 regObj.callbackFn = callbackFn;\r
639                 regObjArr[regObjArr.length] = regObj;\r
640                 setVisibility(objectIdStr, false);\r
641             }\r
642             else if (callbackFn) {\r
643                 callbackFn({success:false, id:objectIdStr});\r
644             }\r
645         },\r
646         \r
647         getObjectById: function(objectIdStr) {\r
648             if (ua.w3) {\r
649                 return getObjectById(objectIdStr);\r
650             }\r
651         },\r
652         \r
653         embedSWF: function(swfUrlStr, replaceElemIdStr, widthStr, heightStr, swfVersionStr, xiSwfUrlStr, flashvarsObj, parObj, attObj, callbackFn) {\r
654             var callbackObj = {success:false, id:replaceElemIdStr};\r
655             if (ua.w3 && !(ua.wk && ua.wk < 312) && swfUrlStr && replaceElemIdStr && widthStr && heightStr && swfVersionStr) {\r
656                 setVisibility(replaceElemIdStr, false);\r
657                 addDomLoadEvent(function() {\r
658                     widthStr += ""; // auto-convert to string\r
659                     heightStr += "";\r
660                     var att = {};\r
661                     if (attObj && typeof attObj === OBJECT) {\r
662                         for (var i in attObj) { // copy object to avoid the use of references, because web authors often reuse attObj for multiple SWFs\r
663                             att[i] = attObj[i];\r
664                         }\r
665                     }\r
666                     att.data = swfUrlStr;\r
667                     att.width = widthStr;\r
668                     att.height = heightStr;\r
669                     var par = {}; \r
670                     if (parObj && typeof parObj === OBJECT) {\r
671                         for (var j in parObj) { // copy object to avoid the use of references, because web authors often reuse parObj for multiple SWFs\r
672                             par[j] = parObj[j];\r
673                         }\r
674                     }\r
675                     if (flashvarsObj && typeof flashvarsObj === OBJECT) {\r
676                         for (var k in flashvarsObj) { // copy object to avoid the use of references, because web authors often reuse flashvarsObj for multiple SWFs\r
677                             if (typeof par.flashvars != UNDEF) {\r
678                                 par.flashvars += "&" + k + "=" + flashvarsObj[k];\r
679                             }\r
680                             else {\r
681                                 par.flashvars = k + "=" + flashvarsObj[k];\r
682                             }\r
683                         }\r
684                     }\r
685                     if (hasPlayerVersion(swfVersionStr)) { // create SWF\r
686                         var obj = createSWF(att, par, replaceElemIdStr);\r
687                         if (att.id == replaceElemIdStr) {\r
688                             setVisibility(replaceElemIdStr, true);\r
689                         }\r
690                         callbackObj.success = true;\r
691                         callbackObj.ref = obj;\r
692                     }\r
693                     else if (xiSwfUrlStr && canExpressInstall()) { // show Adobe Express Install\r
694                         att.data = xiSwfUrlStr;\r
695                         showExpressInstall(att, par, replaceElemIdStr, callbackFn);\r
696                         return;\r
697                     }\r
698                     else { // show alternative content\r
699                         setVisibility(replaceElemIdStr, true);\r
700                     }\r
701                     if (callbackFn) { callbackFn(callbackObj); }\r
702                 });\r
703             }\r
704             else if (callbackFn) { callbackFn(callbackObj); }\r
705         },\r
706         \r
707         switchOffAutoHideShow: function() {\r
708             autoHideShow = false;\r
709         },\r
710         \r
711         ua: ua,\r
712         \r
713         getFlashPlayerVersion: function() {\r
714             return { major:ua.pv[0], minor:ua.pv[1], release:ua.pv[2] };\r
715         },\r
716         \r
717         hasFlashPlayerVersion: hasPlayerVersion,\r
718         \r
719         createSWF: function(attObj, parObj, replaceElemIdStr) {\r
720             if (ua.w3) {\r
721                 return createSWF(attObj, parObj, replaceElemIdStr);\r
722             }\r
723             else {\r
724                 return undefined;\r
725             }\r
726         },\r
727         \r
728         showExpressInstall: function(att, par, replaceElemIdStr, callbackFn) {\r
729             if (ua.w3 && canExpressInstall()) {\r
730                 showExpressInstall(att, par, replaceElemIdStr, callbackFn);\r
731             }\r
732         },\r
733         \r
734         removeSWF: function(objElemIdStr) {\r
735             if (ua.w3) {\r
736                 removeSWF(objElemIdStr);\r
737             }\r
738         },\r
739         \r
740         createCSS: function(selStr, declStr, mediaStr, newStyleBoolean) {\r
741             if (ua.w3) {\r
742                 createCSS(selStr, declStr, mediaStr, newStyleBoolean);\r
743             }\r
744         },\r
745         \r
746         addDomLoadEvent: addDomLoadEvent,\r
747         \r
748         addLoadEvent: addLoadEvent,\r
749         \r
750         getQueryParamValue: function(param) {\r
751             var q = doc.location.search || doc.location.hash;\r
752             if (q) {\r
753                 if (/\?/.test(q)) { q = q.split("?")[1]; } // strip question mark\r
754                 if (param == null) {\r
755                     return urlEncodeIfNecessary(q);\r
756                 }\r
757                 var pairs = q.split("&");\r
758                 for (var i = 0; i < pairs.length; i++) {\r
759                     if (pairs[i].substring(0, pairs[i].indexOf("=")) == param) {\r
760                         return urlEncodeIfNecessary(pairs[i].substring((pairs[i].indexOf("=") + 1)));\r
761                     }\r
762                 }\r
763             }\r
764             return "";\r
765         },\r
766         \r
767         // For internal usage only\r
768         expressInstallCallback: function() {\r
769             if (isExpressInstallActive) {\r
770                 var obj = getElementById(EXPRESS_INSTALL_ID);\r
771                 if (obj && storedAltContent) {\r
772                     obj.parentNode.replaceChild(storedAltContent, obj);\r
773                     if (storedAltContentId) {\r
774                         setVisibility(storedAltContentId, true);\r
775                         if (ua.ie && ua.win) { storedAltContent.style.display = "block"; }\r
776                     }\r
777                     if (storedCallbackFn) { storedCallbackFn(storedCallbackObj); }\r
778                 }\r
779                 isExpressInstallActive = false;\r
780             } \r
781         }\r
782     };\r
783 }();\r