Upgrade to ExtJS 3.0.3 - Released 10/11/2009
[extjs.git] / pkgs / pkg-charts-debug.js
1 /*!
2  * Ext JS Library 3.0.3
3  * Copyright(c) 2006-2009 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
784 /**
785  * @class Ext.FlashComponent
786  * @extends Ext.BoxComponent
787  * @constructor
788  * @xtype flash
789  */
790 Ext.FlashComponent = Ext.extend(Ext.BoxComponent, {
791     /**
792      * @cfg {String} flashVersion
793      * Indicates the version the flash content was published for. Defaults to <tt>'9.0.45'</tt>.
794      */
795     flashVersion : '9.0.45',
796
797     /**
798      * @cfg {String} backgroundColor
799      * The background color of the chart. Defaults to <tt>'#ffffff'</tt>.
800      */
801     backgroundColor: '#ffffff',
802
803     /**
804      * @cfg {String} wmode
805      * The wmode of the flash object. This can be used to control layering. Defaults to <tt>'opaque'</tt>.
806      */
807     wmode: 'opaque',
808     
809     /**
810      * @cfg {Object} flashVars
811      * A set of key value pairs to be passed to the flash object as flash variables. Defaults to <tt>undefined</tt>.
812      */
813     flashVars: undefined,
814     
815     /**
816      * @cfg {Object} flashParams
817      * A set of key value pairs to be passed to the flash object as parameters. Possible parameters can be found here:
818      * http://kb2.adobe.com/cps/127/tn_12701.html Defaults to <tt>undefined</tt>.
819      */
820     flashParams: undefined,
821
822     /**
823      * @cfg {String} url
824      * The URL of the chart to include. Defaults to <tt>undefined</tt>.
825      */
826     url: undefined,
827     swfId : undefined,
828     swfWidth: '100%',
829     swfHeight: '100%',
830
831     /**
832      * @cfg {Boolean} expressInstall
833      * True to prompt the user to install flash if not installed. Note that this uses
834      * Ext.FlashComponent.EXPRESS_INSTALL_URL, which should be set to the local resource. Defaults to <tt>false</tt>.
835      */
836     expressInstall: false,
837
838     initComponent : function(){
839         Ext.FlashComponent.superclass.initComponent.call(this);
840
841         this.addEvents('initialize');
842     },
843
844     onRender : function(){
845         Ext.FlashComponent.superclass.onRender.apply(this, arguments);
846
847         var params = Ext.apply({
848             allowScriptAccess: 'always',
849             bgcolor: this.backgroundColor,
850             wmode: this.wmode
851         }, this.flashParams), vars = Ext.apply({
852             allowedDomain: document.location.hostname,
853             elementID: this.getId(),
854             eventHandler: 'Ext.FlashEventProxy.onEvent'
855         }, this.flashVars);
856
857         new swfobject.embedSWF(this.url, this.id, this.swfWidth, this.swfHeight, this.flashVersion,
858             this.expressInstall ? Ext.FlashComponent.EXPRESS_INSTALL_URL : undefined, vars, params);
859
860         this.swf = Ext.getDom(this.id);
861         this.el = Ext.get(this.swf);
862     },
863
864     getSwfId : function(){
865         return this.swfId || (this.swfId = "extswf" + (++Ext.Component.AUTO_ID));
866     },
867
868     getId : function(){
869         return this.id || (this.id = "extflashcmp" + (++Ext.Component.AUTO_ID));
870     },
871
872     onFlashEvent : function(e){
873         switch(e.type){
874             case "swfReady":
875                 this.initSwf();
876                 return;
877             case "log":
878                 return;
879         }
880         e.component = this;
881         this.fireEvent(e.type.toLowerCase().replace(/event$/, ''), e);
882     },
883
884     initSwf : function(){
885         this.onSwfReady(!!this.isInitialized);
886         this.isInitialized = true;
887         this.fireEvent('initialize', this);
888     },
889
890     beforeDestroy: function(){
891         if(this.rendered){
892             swfobject.removeSWF(this.swf.id);
893         }
894         Ext.FlashComponent.superclass.beforeDestroy.call(this);
895     },
896
897     onSwfReady : Ext.emptyFn
898 });
899
900 /**
901  * Sets the url for installing flash if it doesn't exist. This should be set to a local resource.
902  * @static
903  * @type String
904  */
905 Ext.FlashComponent.EXPRESS_INSTALL_URL = 'http:/' + '/swfobject.googlecode.com/svn/trunk/swfobject/expressInstall.swf';
906
907 Ext.reg('flash', Ext.FlashComponent);/**\r
908  * @class Ext.FlashProxy\r
909  * @singleton\r
910  */\r
911 Ext.FlashEventProxy = {\r
912     onEvent : function(id, e){\r
913         var fp = Ext.getCmp(id);\r
914         if(fp){\r
915             fp.onFlashEvent(e);\r
916         }else{\r
917             arguments.callee.defer(10, this, [id, e]);\r
918         }\r
919     }\r
920 }/**\r
921  * @class Ext.chart.Chart\r
922  * @extends Ext.FlashComponent\r
923  * The Ext.chart package provides the capability to visualize data with flash based charting.\r
924  * Each chart binds directly to an Ext.data.Store enabling automatic updates of the chart.\r
925  * @constructor\r
926  * @xtype chart\r
927  */\r
928  \r
929  Ext.chart.Chart = Ext.extend(Ext.FlashComponent, {\r
930     refreshBuffer: 100,\r
931 \r
932     /**\r
933      * @cfg {Object} chartStyle\r
934      * Sets styles for this chart. Contains a number of default values. Modifying this property will override\r
935      * the base styles on the chart.\r
936      */\r
937     chartStyle: {\r
938         padding: 10,\r
939         animationEnabled: true,\r
940         font: {\r
941             name: 'Tahoma',\r
942             color: 0x444444,\r
943             size: 11\r
944         },\r
945         dataTip: {\r
946             padding: 5,\r
947             border: {\r
948                 color: 0x99bbe8,\r
949                 size:1\r
950             },\r
951             background: {\r
952                 color: 0xDAE7F6,\r
953                 alpha: .9\r
954             },\r
955             font: {\r
956                 name: 'Tahoma',\r
957                 color: 0x15428B,\r
958                 size: 10,\r
959                 bold: true\r
960             }\r
961         }\r
962     },\r
963     \r
964     /**\r
965      * @cfg {String} url\r
966      * The url to load the chart from. This defaults to Ext.chart.Chart.CHART_URL, which should\r
967      * be modified to point to the local charts resource.\r
968      */\r
969     \r
970     /**\r
971      * @cfg {Object} extraStyle\r
972      * Contains extra styles that will be added or overwritten to the default chartStyle. Defaults to <tt>null</tt>.\r
973      */\r
974     extraStyle: null,\r
975     \r
976     /**\r
977      * @cfg {Boolean} disableCaching\r
978      * True to add a "cache buster" to the end of the chart url. Defaults to true for Opera and IE.\r
979      */\r
980     disableCaching: Ext.isIE || Ext.isOpera,\r
981     disableCacheParam: '_dc',\r
982 \r
983     initComponent : function(){\r
984         Ext.chart.Chart.superclass.initComponent.call(this);\r
985         if(!this.url){\r
986             this.url = Ext.chart.Chart.CHART_URL;\r
987         }\r
988         if(this.disableCaching){\r
989             this.url = Ext.urlAppend(this.url, String.format('{0}={1}', this.disableCacheParam, new Date().getTime()));\r
990         }\r
991         this.addEvents(\r
992             'itemmouseover',\r
993             'itemmouseout',\r
994             'itemclick',\r
995             'itemdoubleclick',\r
996             'itemdragstart',\r
997             'itemdrag',\r
998             'itemdragend'\r
999         );\r
1000         this.store = Ext.StoreMgr.lookup(this.store);\r
1001     },\r
1002 \r
1003     /**\r
1004      * Sets a single style value on the Chart instance.\r
1005      *\r
1006      * @param name {String} Name of the Chart style value to change.\r
1007      * @param value {Object} New value to pass to the Chart style.\r
1008      */\r
1009      setStyle: function(name, value){\r
1010          this.swf.setStyle(name, Ext.encode(value));\r
1011      },\r
1012 \r
1013     /**\r
1014      * Resets all styles on the Chart instance.\r
1015      *\r
1016      * @param styles {Object} Initializer for all Chart styles.\r
1017      */\r
1018     setStyles: function(styles){\r
1019         this.swf.setStyles(Ext.encode(styles));\r
1020     },\r
1021 \r
1022     /**\r
1023      * Sets the styles on all series in the Chart.\r
1024      *\r
1025      * @param styles {Array} Initializer for all Chart series styles.\r
1026      */\r
1027     setSeriesStyles: function(styles){\r
1028         var s = [];\r
1029         Ext.each(styles, function(style){\r
1030             s.push(Ext.encode(style));\r
1031         });\r
1032         this.swf.setSeriesStyles(s);\r
1033     },\r
1034 \r
1035     setCategoryNames : function(names){\r
1036         this.swf.setCategoryNames(names);\r
1037     },\r
1038 \r
1039     setTipRenderer : function(fn){\r
1040         var chart = this;\r
1041         this.tipFnName = this.createFnProxy(function(item, index, series){\r
1042             var record = chart.store.getAt(index);\r
1043             return fn(chart, record, index, series);\r
1044         }, this.tipFnName);\r
1045         this.swf.setDataTipFunction(this.tipFnName);\r
1046     },\r
1047 \r
1048     setSeries : function(series){\r
1049         this.series = series;\r
1050         this.refresh();\r
1051     },\r
1052 \r
1053     /**\r
1054      * Changes the data store bound to this chart and refreshes it.\r
1055      * @param {Store} store The store to bind to this chart\r
1056      */\r
1057     bindStore : function(store, initial){\r
1058         if(!initial && this.store){\r
1059             if(store !== this.store && this.store.autoDestroy){\r
1060                 this.store.destroy();\r
1061             }else{\r
1062                 this.store.un("datachanged", this.refresh, this);\r
1063                 this.store.un("add", this.delayRefresh, this);\r
1064                 this.store.un("remove", this.delayRefresh, this);\r
1065                 this.store.un("update", this.delayRefresh, this);\r
1066                 this.store.un("clear", this.refresh, this);\r
1067             }\r
1068         }\r
1069         if(store){\r
1070             store = Ext.StoreMgr.lookup(store);\r
1071             store.on({\r
1072                 scope: this,\r
1073                 datachanged: this.refresh,\r
1074                 add: this.delayRefresh,\r
1075                 remove: this.delayRefresh,\r
1076                 update: this.delayRefresh,\r
1077                 clear: this.refresh\r
1078             });\r
1079         }\r
1080         this.store = store;\r
1081         if(store && !initial){\r
1082             this.refresh();\r
1083         }\r
1084     },\r
1085 \r
1086     onSwfReady : function(isReset){\r
1087         Ext.chart.Chart.superclass.onSwfReady.call(this, isReset);\r
1088         this.swf.setType(this.type);\r
1089 \r
1090         if(this.chartStyle){\r
1091             this.setStyles(Ext.apply({}, this.extraStyle, this.chartStyle));\r
1092         }\r
1093 \r
1094         if(this.categoryNames){\r
1095             this.setCategoryNames(this.categoryNames);\r
1096         }\r
1097 \r
1098         if(this.tipRenderer){\r
1099             this.setTipRenderer(this.tipRenderer);\r
1100         }\r
1101         if(!isReset){\r
1102             this.bindStore(this.store, true);\r
1103         }\r
1104         this.refresh.defer(10, this);\r
1105     },\r
1106 \r
1107     delayRefresh : function(){\r
1108         if(!this.refreshTask){\r
1109             this.refreshTask = new Ext.util.DelayedTask(this.refresh, this);\r
1110         }\r
1111         this.refreshTask.delay(this.refreshBuffer);\r
1112     },\r
1113 \r
1114     refresh : function(){\r
1115         var styleChanged = false;\r
1116         // convert the store data into something YUI charts can understand\r
1117         var data = [], rs = this.store.data.items;\r
1118         for(var j = 0, len = rs.length; j < len; j++){\r
1119             data[j] = rs[j].data;\r
1120         }\r
1121         //make a copy of the series definitions so that we aren't\r
1122         //editing them directly.\r
1123         var dataProvider = [];\r
1124         var seriesCount = 0;\r
1125         var currentSeries = null;\r
1126         var i = 0;\r
1127         if(this.series){\r
1128             seriesCount = this.series.length;\r
1129             for(i = 0; i < seriesCount; i++){\r
1130                 currentSeries = this.series[i];\r
1131                 var clonedSeries = {};\r
1132                 for(var prop in currentSeries){\r
1133                     if(prop == "style" && currentSeries.style !== null){\r
1134                         clonedSeries.style = Ext.encode(currentSeries.style);\r
1135                         styleChanged = true;\r
1136                         //we don't want to modify the styles again next time\r
1137                         //so null out the style property.\r
1138                         // this causes issues\r
1139                         // currentSeries.style = null;\r
1140                     } else{\r
1141                         clonedSeries[prop] = currentSeries[prop];\r
1142                     }\r
1143                 }\r
1144                 dataProvider.push(clonedSeries);\r
1145             }\r
1146         }\r
1147 \r
1148         if(seriesCount > 0){\r
1149             for(i = 0; i < seriesCount; i++){\r
1150                 currentSeries = dataProvider[i];\r
1151                 if(!currentSeries.type){\r
1152                     currentSeries.type = this.type;\r
1153                 }\r
1154                 currentSeries.dataProvider = data;\r
1155             }\r
1156         } else{\r
1157             dataProvider.push({type: this.type, dataProvider: data});\r
1158         }\r
1159         this.swf.setDataProvider(dataProvider);\r
1160     },\r
1161 \r
1162     createFnProxy : function(fn, old){\r
1163         if(old){\r
1164             delete window[old];\r
1165         }\r
1166         var fnName = "extFnProxy" + (++Ext.chart.Chart.PROXY_FN_ID);\r
1167         window[fnName] = fn;\r
1168         return fnName;\r
1169     },\r
1170     \r
1171     onDestroy: function(){\r
1172         Ext.chart.Chart.superclass.onDestroy.call(this);\r
1173         this.bindStore(null);\r
1174         var tip = this.tipFnName;\r
1175         if(!Ext.isEmpty(tip)){\r
1176             delete window[tip];\r
1177         }\r
1178     }\r
1179 });\r
1180 Ext.reg('chart', Ext.chart.Chart);\r
1181 Ext.chart.Chart.PROXY_FN_ID = 0;\r
1182 \r
1183 /**\r
1184  * Sets the url to load the chart from. This should be set to a local resource.\r
1185  * @static\r
1186  * @type String\r
1187  */\r
1188 Ext.chart.Chart.CHART_URL = 'http:/' + '/yui.yahooapis.com/2.7.0/build/charts/assets/charts.swf';\r
1189 \r
1190 /**\r
1191  * @class Ext.chart.PieChart\r
1192  * @extends Ext.chart.Chart\r
1193  * @constructor\r
1194  * @xtype piechart\r
1195  */\r
1196 Ext.chart.PieChart = Ext.extend(Ext.chart.Chart, {\r
1197     type: 'pie',\r
1198 \r
1199     onSwfReady : function(isReset){\r
1200         Ext.chart.PieChart.superclass.onSwfReady.call(this, isReset);\r
1201 \r
1202         this.setDataField(this.dataField);\r
1203         this.setCategoryField(this.categoryField);\r
1204     },\r
1205 \r
1206     setDataField : function(field){\r
1207         this.dataField = field;\r
1208         this.swf.setDataField(field);\r
1209     },\r
1210 \r
1211     setCategoryField : function(field){\r
1212         this.categoryField = field;\r
1213         this.swf.setCategoryField(field);\r
1214     }\r
1215 });\r
1216 Ext.reg('piechart', Ext.chart.PieChart);\r
1217 \r
1218 /**\r
1219  * @class Ext.chart.CartesianChart\r
1220  * @extends Ext.chart.Chart\r
1221  * @constructor\r
1222  * @xtype cartesianchart\r
1223  */\r
1224 Ext.chart.CartesianChart = Ext.extend(Ext.chart.Chart, {\r
1225     onSwfReady : function(isReset){\r
1226         Ext.chart.CartesianChart.superclass.onSwfReady.call(this, isReset);\r
1227 \r
1228         if(this.xField){\r
1229             this.setXField(this.xField);\r
1230         }\r
1231         if(this.yField){\r
1232             this.setYField(this.yField);\r
1233         }\r
1234         if(this.xAxis){\r
1235             this.setXAxis(this.xAxis);\r
1236         }\r
1237         if(this.yAxis){\r
1238             this.setYAxis(this.yAxis);\r
1239         }\r
1240     },\r
1241 \r
1242     setXField : function(value){\r
1243         this.xField = value;\r
1244         this.swf.setHorizontalField(value);\r
1245     },\r
1246 \r
1247     setYField : function(value){\r
1248         this.yField = value;\r
1249         this.swf.setVerticalField(value);\r
1250     },\r
1251 \r
1252     setXAxis : function(value){\r
1253         this.xAxis = this.createAxis('xAxis', value);\r
1254         this.swf.setHorizontalAxis(this.xAxis);\r
1255     },\r
1256 \r
1257     setYAxis : function(value){\r
1258         this.yAxis = this.createAxis('yAxis', value);\r
1259         this.swf.setVerticalAxis(this.yAxis);\r
1260     },\r
1261 \r
1262     createAxis : function(axis, value){\r
1263         var o = Ext.apply({}, value), oldFn = null;\r
1264         if(this[axis]){\r
1265             oldFn = this[axis].labelFunction;\r
1266         }\r
1267         if(o.labelRenderer){\r
1268             var fn = o.labelRenderer;\r
1269             o.labelFunction = this.createFnProxy(function(v){\r
1270                 return fn(v);\r
1271             }, oldFn);\r
1272             delete o.labelRenderer;\r
1273         }\r
1274         return o;\r
1275     }\r
1276 });\r
1277 Ext.reg('cartesianchart', Ext.chart.CartesianChart);\r
1278 \r
1279 /**\r
1280  * @class Ext.chart.LineChart\r
1281  * @extends Ext.chart.CartesianChart\r
1282  * @constructor\r
1283  * @xtype linechart\r
1284  */\r
1285 Ext.chart.LineChart = Ext.extend(Ext.chart.CartesianChart, {\r
1286     type: 'line'\r
1287 });\r
1288 Ext.reg('linechart', Ext.chart.LineChart);\r
1289 \r
1290 /**\r
1291  * @class Ext.chart.ColumnChart\r
1292  * @extends Ext.chart.CartesianChart\r
1293  * @constructor\r
1294  * @xtype columnchart\r
1295  */\r
1296 Ext.chart.ColumnChart = Ext.extend(Ext.chart.CartesianChart, {\r
1297     type: 'column'\r
1298 });\r
1299 Ext.reg('columnchart', Ext.chart.ColumnChart);\r
1300 \r
1301 /**\r
1302  * @class Ext.chart.StackedColumnChart\r
1303  * @extends Ext.chart.CartesianChart\r
1304  * @constructor\r
1305  * @xtype stackedcolumnchart\r
1306  */\r
1307 Ext.chart.StackedColumnChart = Ext.extend(Ext.chart.CartesianChart, {\r
1308     type: 'stackcolumn'\r
1309 });\r
1310 Ext.reg('stackedcolumnchart', Ext.chart.StackedColumnChart);\r
1311 \r
1312 /**\r
1313  * @class Ext.chart.BarChart\r
1314  * @extends Ext.chart.CartesianChart\r
1315  * @constructor\r
1316  * @xtype barchart\r
1317  */\r
1318 Ext.chart.BarChart = Ext.extend(Ext.chart.CartesianChart, {\r
1319     type: 'bar'\r
1320 });\r
1321 Ext.reg('barchart', Ext.chart.BarChart);\r
1322 \r
1323 /**\r
1324  * @class Ext.chart.StackedBarChart\r
1325  * @extends Ext.chart.CartesianChart\r
1326  * @constructor\r
1327  * @xtype stackedbarchart\r
1328  */\r
1329 Ext.chart.StackedBarChart = Ext.extend(Ext.chart.CartesianChart, {\r
1330     type: 'stackbar'\r
1331 });\r
1332 Ext.reg('stackedbarchart', Ext.chart.StackedBarChart);\r
1333 \r
1334 \r
1335 \r
1336 /**\r
1337  * @class Ext.chart.Axis\r
1338  * Defines a CartesianChart's vertical or horizontal axis.\r
1339  * @constructor\r
1340  */\r
1341 Ext.chart.Axis = function(config){\r
1342     Ext.apply(this, config);\r
1343 };\r
1344 \r
1345 Ext.chart.Axis.prototype =\r
1346 {\r
1347     /**\r
1348      * The type of axis.\r
1349      *\r
1350      * @property type\r
1351      * @type String\r
1352      */\r
1353     type: null,\r
1354 \r
1355     /**\r
1356      * The direction in which the axis is drawn. May be "horizontal" or "vertical".\r
1357      *\r
1358      * @property orientation\r
1359      * @type String\r
1360      */\r
1361     orientation: "horizontal",\r
1362 \r
1363     /**\r
1364      * If true, the items on the axis will be drawn in opposite direction.\r
1365      *\r
1366      * @property reverse\r
1367      * @type Boolean\r
1368      */\r
1369     reverse: false,\r
1370 \r
1371     /**\r
1372      * A string reference to the globally-accessible function that may be called to\r
1373      * determine each of the label values for this axis.\r
1374      *\r
1375      * @property labelFunction\r
1376      * @type String\r
1377      */\r
1378     labelFunction: null,\r
1379 \r
1380     /**\r
1381      * If true, labels that overlap previously drawn labels on the axis will be hidden.\r
1382      *\r
1383      * @property hideOverlappingLabels\r
1384      * @type Boolean\r
1385      */\r
1386     hideOverlappingLabels: true\r
1387 };\r
1388 \r
1389 /**\r
1390  * @class Ext.chart.NumericAxis\r
1391  * @extends Ext.chart.Axis\r
1392  * A type of axis whose units are measured in numeric values.\r
1393  * @constructor\r
1394  */\r
1395 Ext.chart.NumericAxis = Ext.extend(Ext.chart.Axis, {\r
1396     type: "numeric",\r
1397 \r
1398     /**\r
1399      * The minimum value drawn by the axis. If not set explicitly, the axis minimum\r
1400      * will be calculated automatically.\r
1401      *\r
1402      * @property minimum\r
1403      * @type Number\r
1404      */\r
1405     minimum: NaN,\r
1406 \r
1407     /**\r
1408      * The maximum value drawn by the axis. If not set explicitly, the axis maximum\r
1409      * will be calculated automatically.\r
1410      *\r
1411      * @property maximum\r
1412      * @type Number\r
1413      */\r
1414     maximum: NaN,\r
1415 \r
1416     /**\r
1417      * The spacing between major intervals on this axis.\r
1418      *\r
1419      * @property majorUnit\r
1420      * @type Number\r
1421      */\r
1422     majorUnit: NaN,\r
1423 \r
1424     /**\r
1425      * The spacing between minor intervals on this axis.\r
1426      *\r
1427      * @property minorUnit\r
1428      * @type Number\r
1429      */\r
1430     minorUnit: NaN,\r
1431 \r
1432     /**\r
1433      * If true, the labels, ticks, gridlines, and other objects will snap to\r
1434      * the nearest major or minor unit. If false, their position will be based\r
1435      * on the minimum value.\r
1436      *\r
1437      * @property snapToUnits\r
1438      * @type Boolean\r
1439      */\r
1440     snapToUnits: true,\r
1441 \r
1442     /**\r
1443      * If true, and the bounds are calculated automatically, either the minimum or\r
1444      * maximum will be set to zero.\r
1445      *\r
1446      * @property alwaysShowZero\r
1447      * @type Boolean\r
1448      */\r
1449     alwaysShowZero: true,\r
1450 \r
1451     /**\r
1452      * The scaling algorithm to use on this axis. May be "linear" or "logarithmic".\r
1453      *\r
1454      * @property scale\r
1455      * @type String\r
1456      */\r
1457     scale: "linear"\r
1458 });\r
1459 \r
1460 /**\r
1461  * @class Ext.chart.TimeAxis\r
1462  * @extends Ext.chart.Axis\r
1463  * A type of axis whose units are measured in time-based values.\r
1464  * @constructor\r
1465  */\r
1466 Ext.chart.TimeAxis = Ext.extend(Ext.chart.Axis, {\r
1467     type: "time",\r
1468 \r
1469     /**\r
1470      * The minimum value drawn by the axis. If not set explicitly, the axis minimum\r
1471      * will be calculated automatically.\r
1472      *\r
1473      * @property minimum\r
1474      * @type Date\r
1475      */\r
1476     minimum: null,\r
1477 \r
1478     /**\r
1479      * The maximum value drawn by the axis. If not set explicitly, the axis maximum\r
1480      * will be calculated automatically.\r
1481      *\r
1482      * @property maximum\r
1483      * @type Number\r
1484      */\r
1485     maximum: null,\r
1486 \r
1487     /**\r
1488      * The spacing between major intervals on this axis.\r
1489      *\r
1490      * @property majorUnit\r
1491      * @type Number\r
1492      */\r
1493     majorUnit: NaN,\r
1494 \r
1495     /**\r
1496      * The time unit used by the majorUnit.\r
1497      *\r
1498      * @property majorTimeUnit\r
1499      * @type String\r
1500      */\r
1501     majorTimeUnit: null,\r
1502 \r
1503     /**\r
1504      * The spacing between minor intervals on this axis.\r
1505      *\r
1506      * @property majorUnit\r
1507      * @type Number\r
1508      */\r
1509     minorUnit: NaN,\r
1510 \r
1511     /**\r
1512      * The time unit used by the minorUnit.\r
1513      *\r
1514      * @property majorTimeUnit\r
1515      * @type String\r
1516      */\r
1517     minorTimeUnit: null,\r
1518 \r
1519     /**\r
1520      * If true, the labels, ticks, gridlines, and other objects will snap to\r
1521      * the nearest major or minor unit. If false, their position will be based\r
1522      * on the minimum value.\r
1523      *\r
1524      * @property snapToUnits\r
1525      * @type Boolean\r
1526      */\r
1527     snapToUnits: true\r
1528 });\r
1529 \r
1530 /**\r
1531  * @class Ext.chart.CategoryAxis\r
1532  * @extends Ext.chart.Axis\r
1533  * A type of axis that displays items in categories.\r
1534  * @constructor\r
1535  */\r
1536 Ext.chart.CategoryAxis = Ext.extend(Ext.chart.Axis, {\r
1537     type: "category",\r
1538 \r
1539     /**\r
1540      * A list of category names to display along this axis.\r
1541      *\r
1542      * @property categoryNames\r
1543      * @type Array\r
1544      */\r
1545     categoryNames: null\r
1546 });\r
1547 \r
1548 /**\r
1549  * @class Ext.chart.Series\r
1550  * Series class for the charts widget.\r
1551  * @constructor\r
1552  */\r
1553 Ext.chart.Series = function(config) { Ext.apply(this, config); };\r
1554 \r
1555 Ext.chart.Series.prototype =\r
1556 {\r
1557     /**\r
1558      * The type of series.\r
1559      *\r
1560      * @property type\r
1561      * @type String\r
1562      */\r
1563     type: null,\r
1564 \r
1565     /**\r
1566      * The human-readable name of the series.\r
1567      *\r
1568      * @property displayName\r
1569      * @type String\r
1570      */\r
1571     displayName: null\r
1572 };\r
1573 \r
1574 /**\r
1575  * @class Ext.chart.CartesianSeries\r
1576  * @extends Ext.chart.Series\r
1577  * CartesianSeries class for the charts widget.\r
1578  * @constructor\r
1579  */\r
1580 Ext.chart.CartesianSeries = Ext.extend(Ext.chart.Series, {\r
1581     /**\r
1582      * The field used to access the x-axis value from the items from the data source.\r
1583      *\r
1584      * @property xField\r
1585      * @type String\r
1586      */\r
1587     xField: null,\r
1588 \r
1589     /**\r
1590      * The field used to access the y-axis value from the items from the data source.\r
1591      *\r
1592      * @property yField\r
1593      * @type String\r
1594      */\r
1595     yField: null\r
1596 });\r
1597 \r
1598 /**\r
1599  * @class Ext.chart.ColumnSeries\r
1600  * @extends Ext.chart.CartesianSeries\r
1601  * ColumnSeries class for the charts widget.\r
1602  * @constructor\r
1603  */\r
1604 Ext.chart.ColumnSeries = Ext.extend(Ext.chart.CartesianSeries, {\r
1605     type: "column"\r
1606 });\r
1607 \r
1608 /**\r
1609  * @class Ext.chart.LineSeries\r
1610  * @extends Ext.chart.CartesianSeries\r
1611  * LineSeries class for the charts widget.\r
1612  * @constructor\r
1613  */\r
1614 Ext.chart.LineSeries = Ext.extend(Ext.chart.CartesianSeries, {\r
1615     type: "line"\r
1616 });\r
1617 \r
1618 /**\r
1619  * @class Ext.chart.BarSeries\r
1620  * @extends Ext.chart.CartesianSeries\r
1621  * BarSeries class for the charts widget.\r
1622  * @constructor\r
1623  */\r
1624 Ext.chart.BarSeries = Ext.extend(Ext.chart.CartesianSeries, {\r
1625     type: "bar"\r
1626 });\r
1627 \r
1628 \r
1629 /**\r
1630  * @class Ext.chart.PieSeries\r
1631  * @extends Ext.chart.Series\r
1632  * PieSeries class for the charts widget.\r
1633  * @constructor\r
1634  */\r
1635 Ext.chart.PieSeries = Ext.extend(Ext.chart.Series, {\r
1636     type: "pie",\r
1637     dataField: null,\r
1638     categoryField: null\r
1639 });