ddb1297cecea63f80c7b42dfbde025f567cbbb73
[roojs1] / roojs-debug.js
1 /*
2  * Based on:
3  * Ext JS Library 1.1.1
4  * Copyright(c) 2006-2007, Ext JS, LLC.
5  *
6  * Originally Released Under LGPL - original licence link has changed is not relivant.
7  *
8  * Fork - LGPL
9  * <script type="text/javascript">
10  */
11  
12
13
14
15
16 // for old browsers
17 window["undefined"] = window["undefined"];
18
19 /**
20  * @class Roo
21  * Roo core utilities and functions.
22  * @singleton
23  */
24 var Roo = {}; 
25 /**
26  * Copies all the properties of config to obj.
27  * @param {Object} obj The receiver of the properties
28  * @param {Object} config The source of the properties
29  * @param {Object} defaults A different object that will also be applied for default values
30  * @return {Object} returns obj
31  * @member Roo apply
32  */
33
34  
35 Roo.apply = function(o, c, defaults){
36     if(defaults){
37         // no "this" reference for friendly out of scope calls
38         Roo.apply(o, defaults);
39     }
40     if(o && c && typeof c == 'object'){
41         for(var p in c){
42             o[p] = c[p];
43         }
44     }
45     return o;
46 };
47
48
49 (function(){
50     var idSeed = 0;
51     var ua = navigator.userAgent.toLowerCase();
52
53     var isStrict = document.compatMode == "CSS1Compat",
54         isOpera = ua.indexOf("opera") > -1,
55         isSafari = (/webkit|khtml/).test(ua),
56         isFirefox = ua.indexOf("firefox") > -1,
57         isIE = ua.indexOf("msie") > -1,
58         isIE7 = ua.indexOf("msie 7") > -1,
59         isIE11 = /trident.*rv\:11\./.test(ua),
60         isEdge = ua.indexOf("edge") > -1,
61         isGecko = !isSafari && ua.indexOf("gecko") > -1,
62         isBorderBox = isIE && !isStrict,
63         isWindows = (ua.indexOf("windows") != -1 || ua.indexOf("win32") != -1),
64         isMac = (ua.indexOf("macintosh") != -1 || ua.indexOf("mac os x") != -1),
65         isLinux = (ua.indexOf("linux") != -1),
66         isSecure = window.location.href.toLowerCase().indexOf("https") === 0,
67         isIOS = /iphone|ipad/.test(ua),
68         isAndroid = /android/.test(ua),
69         isTouch =  (function() {
70             try {
71                 if (ua.indexOf('chrome') != -1 && ua.indexOf('android') == -1) {
72                     window.addEventListener('touchstart', function __set_has_touch__ () {
73                         Roo.isTouch = true;
74                         window.removeEventListener('touchstart', __set_has_touch__);
75                     });
76                     return false; // no touch on chrome!?
77                 }
78                 document.createEvent("TouchEvent");  
79                 return true;  
80             } catch (e) {  
81                 return false;  
82             } 
83             
84         })();
85     // remove css image flicker
86         if(isIE && !isIE7){
87         try{
88             document.execCommand("BackgroundImageCache", false, true);
89         }catch(e){}
90     }
91     
92     Roo.apply(Roo, {
93         /**
94          * True if the browser is in strict mode
95          * @type Boolean
96          */
97         isStrict : isStrict,
98         /**
99          * True if the page is running over SSL
100          * @type Boolean
101          */
102         isSecure : isSecure,
103         /**
104          * True when the document is fully initialized and ready for action
105          * @type Boolean
106          */
107         isReady : false,
108         /**
109          * Turn on debugging output (currently only the factory uses this)
110          * @type Boolean
111          */
112         
113         debug: false,
114
115         /**
116          * True to automatically uncache orphaned Roo.Elements periodically (defaults to true)
117          * @type Boolean
118          */
119         enableGarbageCollector : true,
120
121         /**
122          * True to automatically purge event listeners after uncaching an element (defaults to false).
123          * Note: this only happens if enableGarbageCollector is true.
124          * @type Boolean
125          */
126         enableListenerCollection:false,
127
128         /**
129          * URL to a blank file used by Roo when in secure mode for iframe src and onReady src to prevent
130          * the IE insecure content warning (defaults to javascript:false).
131          * @type String
132          */
133         SSL_SECURE_URL : "javascript:false",
134
135         /**
136          * URL to a 1x1 transparent gif image used by Roo to create inline icons with CSS background images. (Defaults to
137          * "http://Roojs.com/s.gif" and you should change this to a URL on your server).
138          * @type String
139          */
140         BLANK_IMAGE_URL : "http:/"+"/localhost/s.gif",
141
142         emptyFn : function(){},
143         
144         /**
145          * Copies all the properties of config to obj if they don't already exist.
146          * @param {Object} obj The receiver of the properties
147          * @param {Object} config The source of the properties
148          * @return {Object} returns obj
149          */
150         applyIf : function(o, c){
151             if(o && c){
152                 for(var p in c){
153                     if(typeof o[p] == "undefined"){ o[p] = c[p]; }
154                 }
155             }
156             return o;
157         },
158
159         /**
160          * Applies event listeners to elements by selectors when the document is ready.
161          * The event name is specified with an @ suffix.
162 <pre><code>
163 Roo.addBehaviors({
164    // add a listener for click on all anchors in element with id foo
165    '#foo a@click' : function(e, t){
166        // do something
167    },
168
169    // add the same listener to multiple selectors (separated by comma BEFORE the @)
170    '#foo a, #bar span.some-class@mouseover' : function(){
171        // do something
172    }
173 });
174 </code></pre>
175          * @param {Object} obj The list of behaviors to apply
176          */
177         addBehaviors : function(o){
178             if(!Roo.isReady){
179                 Roo.onReady(function(){
180                     Roo.addBehaviors(o);
181                 });
182                 return;
183             }
184             var cache = {}; // simple cache for applying multiple behaviors to same selector does query multiple times
185             for(var b in o){
186                 var parts = b.split('@');
187                 if(parts[1]){ // for Object prototype breakers
188                     var s = parts[0];
189                     if(!cache[s]){
190                         cache[s] = Roo.select(s);
191                     }
192                     cache[s].on(parts[1], o[b]);
193                 }
194             }
195             cache = null;
196         },
197
198         /**
199          * Generates unique ids. If the element already has an id, it is unchanged
200          * @param {String/HTMLElement/Element} el (optional) The element to generate an id for
201          * @param {String} prefix (optional) Id prefix (defaults "Roo-gen")
202          * @return {String} The generated Id.
203          */
204         id : function(el, prefix){
205             prefix = prefix || "roo-gen";
206             el = Roo.getDom(el);
207             var id = prefix + (++idSeed);
208             return el ? (el.id ? el.id : (el.id = id)) : id;
209         },
210          
211        
212         /**
213          * Extends one class with another class and optionally overrides members with the passed literal. This class
214          * also adds the function "override()" to the class that can be used to override
215          * members on an instance.
216          * @param {Object} subclass The class inheriting the functionality
217          * @param {Object} superclass The class being extended
218          * @param {Object} overrides (optional) A literal with members
219          * @method extend
220          */
221         extend : function(){
222             // inline overrides
223             var io = function(o){
224                 for(var m in o){
225                     this[m] = o[m];
226                 }
227             };
228             return function(sb, sp, overrides){
229                 if(typeof sp == 'object'){ // eg. prototype, rather than function constructor..
230                     overrides = sp;
231                     sp = sb;
232                     sb = function(){sp.apply(this, arguments);};
233                 }
234                 var F = function(){}, sbp, spp = sp.prototype;
235                 F.prototype = spp;
236                 sbp = sb.prototype = new F();
237                 sbp.constructor=sb;
238                 sb.superclass=spp;
239                 
240                 if(spp.constructor == Object.prototype.constructor){
241                     spp.constructor=sp;
242                    
243                 }
244                 
245                 sb.override = function(o){
246                     Roo.override(sb, o);
247                 };
248                 sbp.override = io;
249                 Roo.override(sb, overrides);
250                 return sb;
251             };
252         }(),
253
254         /**
255          * Adds a list of functions to the prototype of an existing class, overwriting any existing methods with the same name.
256          * Usage:<pre><code>
257 Roo.override(MyClass, {
258     newMethod1: function(){
259         // etc.
260     },
261     newMethod2: function(foo){
262         // etc.
263     }
264 });
265  </code></pre>
266          * @param {Object} origclass The class to override
267          * @param {Object} overrides The list of functions to add to origClass.  This should be specified as an object literal
268          * containing one or more methods.
269          * @method override
270          */
271         override : function(origclass, overrides){
272             if(overrides){
273                 var p = origclass.prototype;
274                 for(var method in overrides){
275                     p[method] = overrides[method];
276                 }
277             }
278         },
279         /**
280          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
281          * <pre><code>
282 Roo.namespace('Company', 'Company.data');
283 Company.Widget = function() { ... }
284 Company.data.CustomStore = function(config) { ... }
285 </code></pre>
286          * @param {String} namespace1
287          * @param {String} namespace2
288          * @param {String} etc
289          * @method namespace
290          */
291         namespace : function(){
292             var a=arguments, o=null, i, j, d, rt;
293             for (i=0; i<a.length; ++i) {
294                 d=a[i].split(".");
295                 rt = d[0];
296                 /** eval:var:o */
297                 eval('if (typeof ' + rt + ' == "undefined"){' + rt + ' = {};} o = ' + rt + ';');
298                 for (j=1; j<d.length; ++j) {
299                     o[d[j]]=o[d[j]] || {};
300                     o=o[d[j]];
301                 }
302             }
303         },
304         /**
305          * Creates namespaces to be used for scoping variables and classes so that they are not global.  Usage:
306          * <pre><code>
307 Roo.factory({ xns: Roo.data, xtype : 'Store', .....});
308 Roo.factory(conf, Roo.data);
309 </code></pre>
310          * @param {String} classname
311          * @param {String} namespace (optional)
312          * @method factory
313          */
314          
315         factory : function(c, ns)
316         {
317             // no xtype, no ns or c.xns - or forced off by c.xns
318             if (!c.xtype   || (!ns && !c.xns) ||  (c.xns === false)) { // not enough info...
319                 return c;
320             }
321             ns = c.xns ? c.xns : ns; // if c.xns is set, then use that..
322             if (c.constructor == ns[c.xtype]) {// already created...
323                 return c;
324             }
325             if (ns[c.xtype]) {
326                 if (Roo.debug) { Roo.log("Roo.Factory(" + c.xtype + ")"); }
327                 var ret = new ns[c.xtype](c);
328                 ret.xns = false;
329                 return ret;
330             }
331             c.xns = false; // prevent recursion..
332             return c;
333         },
334          /**
335          * Logs to console if it can.
336          *
337          * @param {String|Object} string
338          * @method log
339          */
340         log : function(s)
341         {
342             if ((typeof(console) == 'undefined') || (typeof(console.log) == 'undefined')) {
343                 return; // alerT?
344             }
345             
346             console.log(s);
347         },
348         /**
349          * Takes an object and converts it to an encoded URL. e.g. Roo.urlEncode({foo: 1, bar: 2}); would return "foo=1&bar=2".  Optionally, property values can be arrays, instead of keys and the resulting string that's returned will contain a name/value pair for each array value.
350          * @param {Object} o
351          * @return {String}
352          */
353         urlEncode : function(o){
354             if(!o){
355                 return "";
356             }
357             var buf = [];
358             for(var key in o){
359                 var ov = o[key], k = Roo.encodeURIComponent(key);
360                 var type = typeof ov;
361                 if(type == 'undefined'){
362                     buf.push(k, "=&");
363                 }else if(type != "function" && type != "object"){
364                     buf.push(k, "=", Roo.encodeURIComponent(ov), "&");
365                 }else if(ov instanceof Array){
366                     if (ov.length) {
367                             for(var i = 0, len = ov.length; i < len; i++) {
368                                 buf.push(k, "=", Roo.encodeURIComponent(ov[i] === undefined ? '' : ov[i]), "&");
369                             }
370                         } else {
371                             buf.push(k, "=&");
372                         }
373                 }
374             }
375             buf.pop();
376             return buf.join("");
377         },
378          /**
379          * Safe version of encodeURIComponent
380          * @param {String} data 
381          * @return {String} 
382          */
383         
384         encodeURIComponent : function (data)
385         {
386             try {
387                 return encodeURIComponent(data);
388             } catch(e) {} // should be an uri encode error.
389             
390             if (data == '' || data == null){
391                return '';
392             }
393             // http://stackoverflow.com/questions/2596483/unicode-and-uri-encoding-decoding-and-escaping-in-javascript
394             function nibble_to_hex(nibble){
395                 var chars = '0123456789ABCDEF';
396                 return chars.charAt(nibble);
397             }
398             data = data.toString();
399             var buffer = '';
400             for(var i=0; i<data.length; i++){
401                 var c = data.charCodeAt(i);
402                 var bs = new Array();
403                 if (c > 0x10000){
404                         // 4 bytes
405                     bs[0] = 0xF0 | ((c & 0x1C0000) >>> 18);
406                     bs[1] = 0x80 | ((c & 0x3F000) >>> 12);
407                     bs[2] = 0x80 | ((c & 0xFC0) >>> 6);
408                     bs[3] = 0x80 | (c & 0x3F);
409                 }else if (c > 0x800){
410                          // 3 bytes
411                     bs[0] = 0xE0 | ((c & 0xF000) >>> 12);
412                     bs[1] = 0x80 | ((c & 0xFC0) >>> 6);
413                     bs[2] = 0x80 | (c & 0x3F);
414                 }else if (c > 0x80){
415                        // 2 bytes
416                     bs[0] = 0xC0 | ((c & 0x7C0) >>> 6);
417                     bs[1] = 0x80 | (c & 0x3F);
418                 }else{
419                         // 1 byte
420                     bs[0] = c;
421                 }
422                 for(var j=0; j<bs.length; j++){
423                     var b = bs[j];
424                     var hex = nibble_to_hex((b & 0xF0) >>> 4) 
425                             + nibble_to_hex(b &0x0F);
426                     buffer += '%'+hex;
427                }
428             }
429             return buffer;    
430              
431         },
432
433         /**
434          * Takes an encoded URL and and converts it to an object. e.g. Roo.urlDecode("foo=1&bar=2"); would return {foo: 1, bar: 2} or Roo.urlDecode("foo=1&bar=2&bar=3&bar=4", true); would return {foo: 1, bar: [2, 3, 4]}.
435          * @param {String} string
436          * @param {Boolean} overwrite (optional) Items of the same name will overwrite previous values instead of creating an an array (Defaults to false).
437          * @return {Object} A literal with members
438          */
439         urlDecode : function(string, overwrite){
440             if(!string || !string.length){
441                 return {};
442             }
443             var obj = {};
444             var pairs = string.split('&');
445             var pair, name, value;
446             for(var i = 0, len = pairs.length; i < len; i++){
447                 pair = pairs[i].split('=');
448                 name = decodeURIComponent(pair[0]);
449                 value = decodeURIComponent(pair[1]);
450                 if(overwrite !== true){
451                     if(typeof obj[name] == "undefined"){
452                         obj[name] = value;
453                     }else if(typeof obj[name] == "string"){
454                         obj[name] = [obj[name]];
455                         obj[name].push(value);
456                     }else{
457                         obj[name].push(value);
458                     }
459                 }else{
460                     obj[name] = value;
461                 }
462             }
463             return obj;
464         },
465
466         /**
467          * Iterates an array calling the passed function with each item, stopping if your function returns false. If the
468          * passed array is not really an array, your function is called once with it.
469          * The supplied function is called with (Object item, Number index, Array allItems).
470          * @param {Array/NodeList/Mixed} array
471          * @param {Function} fn
472          * @param {Object} scope
473          */
474         each : function(array, fn, scope){
475             if(typeof array.length == "undefined" || typeof array == "string"){
476                 array = [array];
477             }
478             for(var i = 0, len = array.length; i < len; i++){
479                 if(fn.call(scope || array[i], array[i], i, array) === false){ return i; };
480             }
481         },
482
483         // deprecated
484         combine : function(){
485             var as = arguments, l = as.length, r = [];
486             for(var i = 0; i < l; i++){
487                 var a = as[i];
488                 if(a instanceof Array){
489                     r = r.concat(a);
490                 }else if(a.length !== undefined && !a.substr){
491                     r = r.concat(Array.prototype.slice.call(a, 0));
492                 }else{
493                     r.push(a);
494                 }
495             }
496             return r;
497         },
498
499         /**
500          * Escapes the passed string for use in a regular expression
501          * @param {String} str
502          * @return {String}
503          */
504         escapeRe : function(s) {
505             return s.replace(/([.*+?^${}()|[\]\/\\])/g, "\\$1");
506         },
507
508         // internal
509         callback : function(cb, scope, args, delay){
510             if(typeof cb == "function"){
511                 if(delay){
512                     cb.defer(delay, scope, args || []);
513                 }else{
514                     cb.apply(scope, args || []);
515                 }
516             }
517         },
518
519         /**
520          * Return the dom node for the passed string (id), dom node, or Roo.Element
521          * @param {String/HTMLElement/Roo.Element} el
522          * @return HTMLElement
523          */
524         getDom : function(el){
525             if(!el){
526                 return null;
527             }
528             return el.dom ? el.dom : (typeof el == 'string' ? document.getElementById(el) : el);
529         },
530
531         /**
532         * Shorthand for {@link Roo.ComponentMgr#get}
533         * @param {String} id
534         * @return Roo.Component
535         */
536         getCmp : function(id){
537             return Roo.ComponentMgr.get(id);
538         },
539          
540         num : function(v, defaultValue){
541             if(typeof v != 'number'){
542                 return defaultValue;
543             }
544             return v;
545         },
546
547         destroy : function(){
548             for(var i = 0, a = arguments, len = a.length; i < len; i++) {
549                 var as = a[i];
550                 if(as){
551                     if(as.dom){
552                         as.removeAllListeners();
553                         as.remove();
554                         continue;
555                     }
556                     if(typeof as.purgeListeners == 'function'){
557                         as.purgeListeners();
558                     }
559                     if(typeof as.destroy == 'function'){
560                         as.destroy();
561                     }
562                 }
563             }
564         },
565
566         // inpired by a similar function in mootools library
567         /**
568          * Returns the type of object that is passed in. If the object passed in is null or undefined it
569          * return false otherwise it returns one of the following values:<ul>
570          * <li><b>string</b>: If the object passed is a string</li>
571          * <li><b>number</b>: If the object passed is a number</li>
572          * <li><b>boolean</b>: If the object passed is a boolean value</li>
573          * <li><b>function</b>: If the object passed is a function reference</li>
574          * <li><b>object</b>: If the object passed is an object</li>
575          * <li><b>array</b>: If the object passed is an array</li>
576          * <li><b>regexp</b>: If the object passed is a regular expression</li>
577          * <li><b>element</b>: If the object passed is a DOM Element</li>
578          * <li><b>nodelist</b>: If the object passed is a DOM NodeList</li>
579          * <li><b>textnode</b>: If the object passed is a DOM text node and contains something other than whitespace</li>
580          * <li><b>whitespace</b>: If the object passed is a DOM text node and contains only whitespace</li>
581          * @param {Mixed} object
582          * @return {String}
583          */
584         type : function(o){
585             if(o === undefined || o === null){
586                 return false;
587             }
588             if(o.htmlElement){
589                 return 'element';
590             }
591             var t = typeof o;
592             if(t == 'object' && o.nodeName) {
593                 switch(o.nodeType) {
594                     case 1: return 'element';
595                     case 3: return (/\S/).test(o.nodeValue) ? 'textnode' : 'whitespace';
596                 }
597             }
598             if(t == 'object' || t == 'function') {
599                 switch(o.constructor) {
600                     case Array: return 'array';
601                     case RegExp: return 'regexp';
602                 }
603                 if(typeof o.length == 'number' && typeof o.item == 'function') {
604                     return 'nodelist';
605                 }
606             }
607             return t;
608         },
609
610         /**
611          * Returns true if the passed value is null, undefined or an empty string (optional).
612          * @param {Mixed} value The value to test
613          * @param {Boolean} allowBlank (optional) Pass true if an empty string is not considered empty
614          * @return {Boolean}
615          */
616         isEmpty : function(v, allowBlank){
617             return v === null || v === undefined || (!allowBlank ? v === '' : false);
618         },
619         
620         /** @type Boolean */
621         isOpera : isOpera,
622         /** @type Boolean */
623         isSafari : isSafari,
624         /** @type Boolean */
625         isFirefox : isFirefox,
626         /** @type Boolean */
627         isIE : isIE,
628         /** @type Boolean */
629         isIE7 : isIE7,
630         /** @type Boolean */
631         isIE11 : isIE11,
632         /** @type Boolean */
633         isEdge : isEdge,
634         /** @type Boolean */
635         isGecko : isGecko,
636         /** @type Boolean */
637         isBorderBox : isBorderBox,
638         /** @type Boolean */
639         isWindows : isWindows,
640         /** @type Boolean */
641         isLinux : isLinux,
642         /** @type Boolean */
643         isMac : isMac,
644         /** @type Boolean */
645         isIOS : isIOS,
646         /** @type Boolean */
647         isAndroid : isAndroid,
648         /** @type Boolean */
649         isTouch : isTouch,
650
651         /**
652          * By default, Ext intelligently decides whether floating elements should be shimmed. If you are using flash,
653          * you may want to set this to true.
654          * @type Boolean
655          */
656         useShims : ((isIE && !isIE7) || (isGecko && isMac)),
657         
658         
659                 
660         /**
661          * Selects a single element as a Roo Element
662          * This is about as close as you can get to jQuery's $('do crazy stuff')
663          * @param {String} selector The selector/xpath query
664          * @param {Node} root (optional) The start of the query (defaults to document).
665          * @return {Roo.Element}
666          */
667         selectNode : function(selector, root) 
668         {
669             var node = Roo.DomQuery.selectNode(selector,root);
670             return node ? Roo.get(node) : new Roo.Element(false);
671         }
672         
673     });
674
675
676 })();
677
678 Roo.namespace("Roo", "Roo.util", "Roo.grid", "Roo.dd", "Roo.tree", "Roo.data",
679                 "Roo.form", "Roo.menu", "Roo.state", "Roo.lib", "Roo.layout",
680                 "Roo.app", "Roo.ux",
681                 "Roo.bootstrap",
682                 "Roo.bootstrap.dash");
683 /*
684  * Based on:
685  * Ext JS Library 1.1.1
686  * Copyright(c) 2006-2007, Ext JS, LLC.
687  *
688  * Originally Released Under LGPL - original licence link has changed is not relivant.
689  *
690  * Fork - LGPL
691  * <script type="text/javascript">
692  */
693
694 (function() {    
695     // wrappedn so fnCleanup is not in global scope...
696     if(Roo.isIE) {
697         function fnCleanUp() {
698             var p = Function.prototype;
699             delete p.createSequence;
700             delete p.defer;
701             delete p.createDelegate;
702             delete p.createCallback;
703             delete p.createInterceptor;
704
705             window.detachEvent("onunload", fnCleanUp);
706         }
707         window.attachEvent("onunload", fnCleanUp);
708     }
709 })();
710
711
712 /**
713  * @class Function
714  * These functions are available on every Function object (any JavaScript function).
715  */
716 Roo.apply(Function.prototype, {
717      /**
718      * Creates a callback that passes arguments[0], arguments[1], arguments[2], ...
719      * Call directly on any function. Example: <code>myFunction.createCallback(myarg, myarg2)</code>
720      * Will create a function that is bound to those 2 args.
721      * @return {Function} The new function
722     */
723     createCallback : function(/*args...*/){
724         // make args available, in function below
725         var args = arguments;
726         var method = this;
727         return function() {
728             return method.apply(window, args);
729         };
730     },
731
732     /**
733      * Creates a delegate (callback) that sets the scope to obj.
734      * Call directly on any function. Example: <code>this.myFunction.createDelegate(this)</code>
735      * Will create a function that is automatically scoped to this.
736      * @param {Object} obj (optional) The object for which the scope is set
737      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
738      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
739      *                                             if a number the args are inserted at the specified position
740      * @return {Function} The new function
741      */
742     createDelegate : function(obj, args, appendArgs){
743         var method = this;
744         return function() {
745             var callArgs = args || arguments;
746             if(appendArgs === true){
747                 callArgs = Array.prototype.slice.call(arguments, 0);
748                 callArgs = callArgs.concat(args);
749             }else if(typeof appendArgs == "number"){
750                 callArgs = Array.prototype.slice.call(arguments, 0); // copy arguments first
751                 var applyArgs = [appendArgs, 0].concat(args); // create method call params
752                 Array.prototype.splice.apply(callArgs, applyArgs); // splice them in
753             }
754             return method.apply(obj || window, callArgs);
755         };
756     },
757
758     /**
759      * Calls this function after the number of millseconds specified.
760      * @param {Number} millis The number of milliseconds for the setTimeout call (if 0 the function is executed immediately)
761      * @param {Object} obj (optional) The object for which the scope is set
762      * @param {Array} args (optional) Overrides arguments for the call. (Defaults to the arguments passed by the caller)
763      * @param {Boolean/Number} appendArgs (optional) if True args are appended to call args instead of overriding,
764      *                                             if a number the args are inserted at the specified position
765      * @return {Number} The timeout id that can be used with clearTimeout
766      */
767     defer : function(millis, obj, args, appendArgs){
768         var fn = this.createDelegate(obj, args, appendArgs);
769         if(millis){
770             return setTimeout(fn, millis);
771         }
772         fn();
773         return 0;
774     },
775     /**
776      * Create a combined function call sequence of the original function + the passed function.
777      * The resulting function returns the results of the original function.
778      * The passed fcn is called with the parameters of the original function
779      * @param {Function} fcn The function to sequence
780      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
781      * @return {Function} The new function
782      */
783     createSequence : function(fcn, scope){
784         if(typeof fcn != "function"){
785             return this;
786         }
787         var method = this;
788         return function() {
789             var retval = method.apply(this || window, arguments);
790             fcn.apply(scope || this || window, arguments);
791             return retval;
792         };
793     },
794
795     /**
796      * Creates an interceptor function. The passed fcn is called before the original one. If it returns false, the original one is not called.
797      * The resulting function returns the results of the original function.
798      * The passed fcn is called with the parameters of the original function.
799      * @addon
800      * @param {Function} fcn The function to call before the original
801      * @param {Object} scope (optional) The scope of the passed fcn (Defaults to scope of original function or window)
802      * @return {Function} The new function
803      */
804     createInterceptor : function(fcn, scope){
805         if(typeof fcn != "function"){
806             return this;
807         }
808         var method = this;
809         return function() {
810             fcn.target = this;
811             fcn.method = method;
812             if(fcn.apply(scope || this || window, arguments) === false){
813                 return;
814             }
815             return method.apply(this || window, arguments);
816         };
817     }
818 });
819 /*
820  * Based on:
821  * Ext JS Library 1.1.1
822  * Copyright(c) 2006-2007, Ext JS, LLC.
823  *
824  * Originally Released Under LGPL - original licence link has changed is not relivant.
825  *
826  * Fork - LGPL
827  * <script type="text/javascript">
828  */
829
830 Roo.applyIf(String, {
831     
832     /** @scope String */
833     
834     /**
835      * Escapes the passed string for ' and \
836      * @param {String} string The string to escape
837      * @return {String} The escaped string
838      * @static
839      */
840     escape : function(string) {
841         return string.replace(/('|\\)/g, "\\$1");
842     },
843
844     /**
845      * Pads the left side of a string with a specified character.  This is especially useful
846      * for normalizing number and date strings.  Example usage:
847      * <pre><code>
848 var s = String.leftPad('123', 5, '0');
849 // s now contains the string: '00123'
850 </code></pre>
851      * @param {String} string The original string
852      * @param {Number} size The total length of the output string
853      * @param {String} char (optional) The character with which to pad the original string (defaults to empty string " ")
854      * @return {String} The padded string
855      * @static
856      */
857     leftPad : function (val, size, ch) {
858         var result = new String(val);
859         if(ch === null || ch === undefined || ch === '') {
860             ch = " ";
861         }
862         while (result.length < size) {
863             result = ch + result;
864         }
865         return result;
866     },
867
868     /**
869      * Allows you to define a tokenized string and pass an arbitrary number of arguments to replace the tokens.  Each
870      * token must be unique, and must increment in the format {0}, {1}, etc.  Example usage:
871      * <pre><code>
872 var cls = 'my-class', text = 'Some text';
873 var s = String.format('<div class="{0}">{1}</div>', cls, text);
874 // s now contains the string: '<div class="my-class">Some text</div>'
875 </code></pre>
876      * @param {String} string The tokenized string to be formatted
877      * @param {String} value1 The value to replace token {0}
878      * @param {String} value2 Etc...
879      * @return {String} The formatted string
880      * @static
881      */
882     format : function(format){
883         var args = Array.prototype.slice.call(arguments, 1);
884         return format.replace(/\{(\d+)\}/g, function(m, i){
885             return Roo.util.Format.htmlEncode(args[i]);
886         });
887     }
888   
889     
890 });
891
892 /**
893  * Utility function that allows you to easily switch a string between two alternating values.  The passed value
894  * is compared to the current string, and if they are equal, the other value that was passed in is returned.  If
895  * they are already different, the first value passed in is returned.  Note that this method returns the new value
896  * but does not change the current string.
897  * <pre><code>
898 // alternate sort directions
899 sort = sort.toggle('ASC', 'DESC');
900
901 // instead of conditional logic:
902 sort = (sort == 'ASC' ? 'DESC' : 'ASC');
903 </code></pre>
904  * @param {String} value The value to compare to the current string
905  * @param {String} other The new value to use if the string already equals the first value passed in
906  * @return {String} The new value
907  */
908  
909 String.prototype.toggle = function(value, other){
910     return this == value ? other : value;
911 };
912
913
914 /**
915   * Remove invalid unicode characters from a string 
916   *
917   * @return {String} The clean string
918   */
919 String.prototype.unicodeClean = function () {
920     return this.replace(/[\s\S]/g,
921         function(character) {
922             if (character.charCodeAt()< 256) {
923               return character;
924            }
925            try {
926                 encodeURIComponent(character);
927            } catch(e) { 
928               return '';
929            }
930            return character;
931         }
932     );
933 };
934   
935 /*
936  * Based on:
937  * Ext JS Library 1.1.1
938  * Copyright(c) 2006-2007, Ext JS, LLC.
939  *
940  * Originally Released Under LGPL - original licence link has changed is not relivant.
941  *
942  * Fork - LGPL
943  * <script type="text/javascript">
944  */
945
946  /**
947  * @class Number
948  */
949 Roo.applyIf(Number.prototype, {
950     /**
951      * Checks whether or not the current number is within a desired range.  If the number is already within the
952      * range it is returned, otherwise the min or max value is returned depending on which side of the range is
953      * exceeded.  Note that this method returns the constrained value but does not change the current number.
954      * @param {Number} min The minimum number in the range
955      * @param {Number} max The maximum number in the range
956      * @return {Number} The constrained value if outside the range, otherwise the current value
957      */
958     constrain : function(min, max){
959         return Math.min(Math.max(this, min), max);
960     }
961 });/*
962  * Based on:
963  * Ext JS Library 1.1.1
964  * Copyright(c) 2006-2007, Ext JS, LLC.
965  *
966  * Originally Released Under LGPL - original licence link has changed is not relivant.
967  *
968  * Fork - LGPL
969  * <script type="text/javascript">
970  */
971  /**
972  * @class Array
973  */
974 Roo.applyIf(Array.prototype, {
975     /**
976      * 
977      * Checks whether or not the specified object exists in the array.
978      * @param {Object} o The object to check for
979      * @return {Number} The index of o in the array (or -1 if it is not found)
980      */
981     indexOf : function(o){
982        for (var i = 0, len = this.length; i < len; i++){
983               if(this[i] == o) { return i; }
984        }
985            return -1;
986     },
987
988     /**
989      * Removes the specified object from the array.  If the object is not found nothing happens.
990      * @param {Object} o The object to remove
991      */
992     remove : function(o){
993        var index = this.indexOf(o);
994        if(index != -1){
995            this.splice(index, 1);
996        }
997     },
998     /**
999      * Map (JS 1.6 compatibility)
1000      * @param {Function} function  to call
1001      */
1002     map : function(fun )
1003     {
1004         var len = this.length >>> 0;
1005         if (typeof fun != "function") {
1006             throw new TypeError();
1007         }
1008         var res = new Array(len);
1009         var thisp = arguments[1];
1010         for (var i = 0; i < len; i++)
1011         {
1012             if (i in this) {
1013                 res[i] = fun.call(thisp, this[i], i, this);
1014             }
1015         }
1016
1017         return res;
1018     }
1019     
1020 });
1021
1022
1023  
1024 /*
1025  * Based on:
1026  * Ext JS Library 1.1.1
1027  * Copyright(c) 2006-2007, Ext JS, LLC.
1028  *
1029  * Originally Released Under LGPL - original licence link has changed is not relivant.
1030  *
1031  * Fork - LGPL
1032  * <script type="text/javascript">
1033  */
1034
1035 /**
1036  * @class Date
1037  *
1038  * The date parsing and format syntax is a subset of
1039  * <a href="http://www.php.net/date">PHP's date() function</a>, and the formats that are
1040  * supported will provide results equivalent to their PHP versions.
1041  *
1042  * Following is the list of all currently supported formats:
1043  *<pre>
1044 Sample date:
1045 'Wed Jan 10 2007 15:05:01 GMT-0600 (Central Standard Time)'
1046
1047 Format  Output      Description
1048 ------  ----------  --------------------------------------------------------------
1049   d      10         Day of the month, 2 digits with leading zeros
1050   D      Wed        A textual representation of a day, three letters
1051   j      10         Day of the month without leading zeros
1052   l      Wednesday  A full textual representation of the day of the week
1053   S      th         English ordinal day of month suffix, 2 chars (use with j)
1054   w      3          Numeric representation of the day of the week
1055   z      9          The julian date, or day of the year (0-365)
1056   W      01         ISO-8601 2-digit week number of year, weeks starting on Monday (00-52)
1057   F      January    A full textual representation of the month
1058   m      01         Numeric representation of a month, with leading zeros
1059   M      Jan        Month name abbreviation, three letters
1060   n      1          Numeric representation of a month, without leading zeros
1061   t      31         Number of days in the given month
1062   L      0          Whether it's a leap year (1 if it is a leap year, else 0)
1063   Y      2007       A full numeric representation of a year, 4 digits
1064   y      07         A two digit representation of a year
1065   a      pm         Lowercase Ante meridiem and Post meridiem
1066   A      PM         Uppercase Ante meridiem and Post meridiem
1067   g      3          12-hour format of an hour without leading zeros
1068   G      15         24-hour format of an hour without leading zeros
1069   h      03         12-hour format of an hour with leading zeros
1070   H      15         24-hour format of an hour with leading zeros
1071   i      05         Minutes with leading zeros
1072   s      01         Seconds, with leading zeros
1073   O      -0600      Difference to Greenwich time (GMT) in hours (Allows +08, without minutes)
1074   P      -06:00     Difference to Greenwich time (GMT) with colon between hours and minutes
1075   T      CST        Timezone setting of the machine running the code
1076   Z      -21600     Timezone offset in seconds (negative if west of UTC, positive if east)
1077 </pre>
1078  *
1079  * Example usage (note that you must escape format specifiers with '\\' to render them as character literals):
1080  * <pre><code>
1081 var dt = new Date('1/10/2007 03:05:01 PM GMT-0600');
1082 document.write(dt.format('Y-m-d'));                         //2007-01-10
1083 document.write(dt.format('F j, Y, g:i a'));                 //January 10, 2007, 3:05 pm
1084 document.write(dt.format('l, \\t\\he dS of F Y h:i:s A'));  //Wednesday, the 10th of January 2007 03:05:01 PM
1085  </code></pre>
1086  *
1087  * Here are some standard date/time patterns that you might find helpful.  They
1088  * are not part of the source of Date.js, but to use them you can simply copy this
1089  * block of code into any script that is included after Date.js and they will also become
1090  * globally available on the Date object.  Feel free to add or remove patterns as needed in your code.
1091  * <pre><code>
1092 Date.patterns = {
1093     ISO8601Long:"Y-m-d H:i:s",
1094     ISO8601Short:"Y-m-d",
1095     ShortDate: "n/j/Y",
1096     LongDate: "l, F d, Y",
1097     FullDateTime: "l, F d, Y g:i:s A",
1098     MonthDay: "F d",
1099     ShortTime: "g:i A",
1100     LongTime: "g:i:s A",
1101     SortableDateTime: "Y-m-d\\TH:i:s",
1102     UniversalSortableDateTime: "Y-m-d H:i:sO",
1103     YearMonth: "F, Y"
1104 };
1105 </code></pre>
1106  *
1107  * Example usage:
1108  * <pre><code>
1109 var dt = new Date();
1110 document.write(dt.format(Date.patterns.ShortDate));
1111  </code></pre>
1112  */
1113
1114 /*
1115  * Most of the date-formatting functions below are the excellent work of Baron Schwartz.
1116  * They generate precompiled functions from date formats instead of parsing and
1117  * processing the pattern every time you format a date.  These functions are available
1118  * on every Date object (any javascript function).
1119  *
1120  * The original article and download are here:
1121  * http://www.xaprb.com/blog/2005/12/12/javascript-closures-for-runtime-efficiency/
1122  *
1123  */
1124  
1125  
1126  // was in core
1127 /**
1128  Returns the number of milliseconds between this date and date
1129  @param {Date} date (optional) Defaults to now
1130  @return {Number} The diff in milliseconds
1131  @member Date getElapsed
1132  */
1133 Date.prototype.getElapsed = function(date) {
1134         return Math.abs((date || new Date()).getTime()-this.getTime());
1135 };
1136 // was in date file..
1137
1138
1139 // private
1140 Date.parseFunctions = {count:0};
1141 // private
1142 Date.parseRegexes = [];
1143 // private
1144 Date.formatFunctions = {count:0};
1145
1146 // private
1147 Date.prototype.dateFormat = function(format) {
1148     if (Date.formatFunctions[format] == null) {
1149         Date.createNewFormat(format);
1150     }
1151     var func = Date.formatFunctions[format];
1152     return this[func]();
1153 };
1154
1155
1156 /**
1157  * Formats a date given the supplied format string
1158  * @param {String} format The format string
1159  * @return {String} The formatted date
1160  * @method
1161  */
1162 Date.prototype.format = Date.prototype.dateFormat;
1163
1164 // private
1165 Date.createNewFormat = function(format) {
1166     var funcName = "format" + Date.formatFunctions.count++;
1167     Date.formatFunctions[format] = funcName;
1168     var code = "Date.prototype." + funcName + " = function(){return ";
1169     var special = false;
1170     var ch = '';
1171     for (var i = 0; i < format.length; ++i) {
1172         ch = format.charAt(i);
1173         if (!special && ch == "\\") {
1174             special = true;
1175         }
1176         else if (special) {
1177             special = false;
1178             code += "'" + String.escape(ch) + "' + ";
1179         }
1180         else {
1181             code += Date.getFormatCode(ch);
1182         }
1183     }
1184     /** eval:var:zzzzzzzzzzzzz */
1185     eval(code.substring(0, code.length - 3) + ";}");
1186 };
1187
1188 // private
1189 Date.getFormatCode = function(character) {
1190     switch (character) {
1191     case "d":
1192         return "String.leftPad(this.getDate(), 2, '0') + ";
1193     case "D":
1194         return "Date.dayNames[this.getDay()].substring(0, 3) + ";
1195     case "j":
1196         return "this.getDate() + ";
1197     case "l":
1198         return "Date.dayNames[this.getDay()] + ";
1199     case "S":
1200         return "this.getSuffix() + ";
1201     case "w":
1202         return "this.getDay() + ";
1203     case "z":
1204         return "this.getDayOfYear() + ";
1205     case "W":
1206         return "this.getWeekOfYear() + ";
1207     case "F":
1208         return "Date.monthNames[this.getMonth()] + ";
1209     case "m":
1210         return "String.leftPad(this.getMonth() + 1, 2, '0') + ";
1211     case "M":
1212         return "Date.monthNames[this.getMonth()].substring(0, 3) + ";
1213     case "n":
1214         return "(this.getMonth() + 1) + ";
1215     case "t":
1216         return "this.getDaysInMonth() + ";
1217     case "L":
1218         return "(this.isLeapYear() ? 1 : 0) + ";
1219     case "Y":
1220         return "this.getFullYear() + ";
1221     case "y":
1222         return "('' + this.getFullYear()).substring(2, 4) + ";
1223     case "a":
1224         return "(this.getHours() < 12 ? 'am' : 'pm') + ";
1225     case "A":
1226         return "(this.getHours() < 12 ? 'AM' : 'PM') + ";
1227     case "g":
1228         return "((this.getHours() % 12) ? this.getHours() % 12 : 12) + ";
1229     case "G":
1230         return "this.getHours() + ";
1231     case "h":
1232         return "String.leftPad((this.getHours() % 12) ? this.getHours() % 12 : 12, 2, '0') + ";
1233     case "H":
1234         return "String.leftPad(this.getHours(), 2, '0') + ";
1235     case "i":
1236         return "String.leftPad(this.getMinutes(), 2, '0') + ";
1237     case "s":
1238         return "String.leftPad(this.getSeconds(), 2, '0') + ";
1239     case "O":
1240         return "this.getGMTOffset() + ";
1241     case "P":
1242         return "this.getGMTColonOffset() + ";
1243     case "T":
1244         return "this.getTimezone() + ";
1245     case "Z":
1246         return "(this.getTimezoneOffset() * -60) + ";
1247     default:
1248         return "'" + String.escape(character) + "' + ";
1249     }
1250 };
1251
1252 /**
1253  * Parses the passed string using the specified format. Note that this function expects dates in normal calendar
1254  * format, meaning that months are 1-based (1 = January) and not zero-based like in JavaScript dates.  Any part of
1255  * the date format that is not specified will default to the current date value for that part.  Time parts can also
1256  * be specified, but default to 0.  Keep in mind that the input date string must precisely match the specified format
1257  * string or the parse operation will fail.
1258  * Example Usage:
1259 <pre><code>
1260 //dt = Fri May 25 2007 (current date)
1261 var dt = new Date();
1262
1263 //dt = Thu May 25 2006 (today's month/day in 2006)
1264 dt = Date.parseDate("2006", "Y");
1265
1266 //dt = Sun Jan 15 2006 (all date parts specified)
1267 dt = Date.parseDate("2006-1-15", "Y-m-d");
1268
1269 //dt = Sun Jan 15 2006 15:20:01 GMT-0600 (CST)
1270 dt = Date.parseDate("2006-1-15 3:20:01 PM", "Y-m-d h:i:s A" );
1271 </code></pre>
1272  * @param {String} input The unparsed date as a string
1273  * @param {String} format The format the date is in
1274  * @return {Date} The parsed date
1275  * @static
1276  */
1277 Date.parseDate = function(input, format) {
1278     if (Date.parseFunctions[format] == null) {
1279         Date.createParser(format);
1280     }
1281     var func = Date.parseFunctions[format];
1282     return Date[func](input);
1283 };
1284 /**
1285  * @private
1286  */
1287
1288 Date.createParser = function(format) {
1289     var funcName = "parse" + Date.parseFunctions.count++;
1290     var regexNum = Date.parseRegexes.length;
1291     var currentGroup = 1;
1292     Date.parseFunctions[format] = funcName;
1293
1294     var code = "Date." + funcName + " = function(input){\n"
1295         + "var y = -1, m = -1, d = -1, h = -1, i = -1, s = -1, o, z, v;\n"
1296         + "var d = new Date();\n"
1297         + "y = d.getFullYear();\n"
1298         + "m = d.getMonth();\n"
1299         + "d = d.getDate();\n"
1300         + "if (typeof(input) !== 'string') { input = input.toString(); }\n"
1301         + "var results = input.match(Date.parseRegexes[" + regexNum + "]);\n"
1302         + "if (results && results.length > 0) {";
1303     var regex = "";
1304
1305     var special = false;
1306     var ch = '';
1307     for (var i = 0; i < format.length; ++i) {
1308         ch = format.charAt(i);
1309         if (!special && ch == "\\") {
1310             special = true;
1311         }
1312         else if (special) {
1313             special = false;
1314             regex += String.escape(ch);
1315         }
1316         else {
1317             var obj = Date.formatCodeToRegex(ch, currentGroup);
1318             currentGroup += obj.g;
1319             regex += obj.s;
1320             if (obj.g && obj.c) {
1321                 code += obj.c;
1322             }
1323         }
1324     }
1325
1326     code += "if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0 && s >= 0)\n"
1327         + "{v = new Date(y, m, d, h, i, s);}\n"
1328         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0 && i >= 0)\n"
1329         + "{v = new Date(y, m, d, h, i);}\n"
1330         + "else if (y >= 0 && m >= 0 && d > 0 && h >= 0)\n"
1331         + "{v = new Date(y, m, d, h);}\n"
1332         + "else if (y >= 0 && m >= 0 && d > 0)\n"
1333         + "{v = new Date(y, m, d);}\n"
1334         + "else if (y >= 0 && m >= 0)\n"
1335         + "{v = new Date(y, m);}\n"
1336         + "else if (y >= 0)\n"
1337         + "{v = new Date(y);}\n"
1338         + "}return (v && (z || o))?\n" // favour UTC offset over GMT offset
1339         + "    ((z)? v.add(Date.SECOND, (v.getTimezoneOffset() * 60) + (z*1)) :\n" // reset to UTC, then add offset
1340         + "        v.add(Date.HOUR, (v.getGMTOffset() / 100) + (o / -100))) : v\n" // reset to GMT, then add offset
1341         + ";}";
1342
1343     Date.parseRegexes[regexNum] = new RegExp("^" + regex + "$");
1344     /** eval:var:zzzzzzzzzzzzz */
1345     eval(code);
1346 };
1347
1348 // private
1349 Date.formatCodeToRegex = function(character, currentGroup) {
1350     switch (character) {
1351     case "D":
1352         return {g:0,
1353         c:null,
1354         s:"(?:Sun|Mon|Tue|Wed|Thu|Fri|Sat)"};
1355     case "j":
1356         return {g:1,
1357             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1358             s:"(\\d{1,2})"}; // day of month without leading zeroes
1359     case "d":
1360         return {g:1,
1361             c:"d = parseInt(results[" + currentGroup + "], 10);\n",
1362             s:"(\\d{2})"}; // day of month with leading zeroes
1363     case "l":
1364         return {g:0,
1365             c:null,
1366             s:"(?:" + Date.dayNames.join("|") + ")"};
1367     case "S":
1368         return {g:0,
1369             c:null,
1370             s:"(?:st|nd|rd|th)"};
1371     case "w":
1372         return {g:0,
1373             c:null,
1374             s:"\\d"};
1375     case "z":
1376         return {g:0,
1377             c:null,
1378             s:"(?:\\d{1,3})"};
1379     case "W":
1380         return {g:0,
1381             c:null,
1382             s:"(?:\\d{2})"};
1383     case "F":
1384         return {g:1,
1385             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "].substring(0, 3)], 10);\n",
1386             s:"(" + Date.monthNames.join("|") + ")"};
1387     case "M":
1388         return {g:1,
1389             c:"m = parseInt(Date.monthNumbers[results[" + currentGroup + "]], 10);\n",
1390             s:"(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)"};
1391     case "n":
1392         return {g:1,
1393             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1394             s:"(\\d{1,2})"}; // Numeric representation of a month, without leading zeros
1395     case "m":
1396         return {g:1,
1397             c:"m = parseInt(results[" + currentGroup + "], 10) - 1;\n",
1398             s:"(\\d{2})"}; // Numeric representation of a month, with leading zeros
1399     case "t":
1400         return {g:0,
1401             c:null,
1402             s:"\\d{1,2}"};
1403     case "L":
1404         return {g:0,
1405             c:null,
1406             s:"(?:1|0)"};
1407     case "Y":
1408         return {g:1,
1409             c:"y = parseInt(results[" + currentGroup + "], 10);\n",
1410             s:"(\\d{4})"};
1411     case "y":
1412         return {g:1,
1413             c:"var ty = parseInt(results[" + currentGroup + "], 10);\n"
1414                 + "y = ty > Date.y2kYear ? 1900 + ty : 2000 + ty;\n",
1415             s:"(\\d{1,2})"};
1416     case "a":
1417         return {g:1,
1418             c:"if (results[" + currentGroup + "] == 'am') {\n"
1419                 + "if (h == 12) { h = 0; }\n"
1420                 + "} else { if (h < 12) { h += 12; }}",
1421             s:"(am|pm)"};
1422     case "A":
1423         return {g:1,
1424             c:"if (results[" + currentGroup + "] == 'AM') {\n"
1425                 + "if (h == 12) { h = 0; }\n"
1426                 + "} else { if (h < 12) { h += 12; }}",
1427             s:"(AM|PM)"};
1428     case "g":
1429     case "G":
1430         return {g:1,
1431             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1432             s:"(\\d{1,2})"}; // 12/24-hr format  format of an hour without leading zeroes
1433     case "h":
1434     case "H":
1435         return {g:1,
1436             c:"h = parseInt(results[" + currentGroup + "], 10);\n",
1437             s:"(\\d{2})"}; //  12/24-hr format  format of an hour with leading zeroes
1438     case "i":
1439         return {g:1,
1440             c:"i = parseInt(results[" + currentGroup + "], 10);\n",
1441             s:"(\\d{2})"};
1442     case "s":
1443         return {g:1,
1444             c:"s = parseInt(results[" + currentGroup + "], 10);\n",
1445             s:"(\\d{2})"};
1446     case "O":
1447         return {g:1,
1448             c:[
1449                 "o = results[", currentGroup, "];\n",
1450                 "var sn = o.substring(0,1);\n", // get + / - sign
1451                 "var hr = o.substring(1,3)*1 + Math.floor(o.substring(3,5) / 60);\n", // get hours (performs minutes-to-hour conversion also)
1452                 "var mn = o.substring(3,5) % 60;\n", // get minutes
1453                 "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n", // -12hrs <= GMT offset <= 14hrs
1454                 "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1455             ].join(""),
1456             s:"([+\-]\\d{2,4})"};
1457     
1458     
1459     case "P":
1460         return {g:1,
1461                 c:[
1462                    "o = results[", currentGroup, "];\n",
1463                    "var sn = o.substring(0,1);\n",
1464                    "var hr = o.substring(1,3)*1 + Math.floor(o.substring(4,6) / 60);\n",
1465                    "var mn = o.substring(4,6) % 60;\n",
1466                    "o = ((-12 <= (hr*60 + mn)/60) && ((hr*60 + mn)/60 <= 14))?\n",
1467                         "    (sn + String.leftPad(hr, 2, 0) + String.leftPad(mn, 2, 0)) : null;\n"
1468             ].join(""),
1469             s:"([+\-]\\d{4})"};
1470     case "T":
1471         return {g:0,
1472             c:null,
1473             s:"[A-Z]{1,4}"}; // timezone abbrev. may be between 1 - 4 chars
1474     case "Z":
1475         return {g:1,
1476             c:"z = results[" + currentGroup + "];\n" // -43200 <= UTC offset <= 50400
1477                   + "z = (-43200 <= z*1 && z*1 <= 50400)? z : null;\n",
1478             s:"([+\-]?\\d{1,5})"}; // leading '+' sign is optional for UTC offset
1479     default:
1480         return {g:0,
1481             c:null,
1482             s:String.escape(character)};
1483     }
1484 };
1485
1486 /**
1487  * Get the timezone abbreviation of the current date (equivalent to the format specifier 'T').
1488  * @return {String} The abbreviated timezone name (e.g. 'CST')
1489  */
1490 Date.prototype.getTimezone = function() {
1491     return this.toString().replace(/^.*? ([A-Z]{1,4})[\-+][0-9]{4} .*$/, "$1");
1492 };
1493
1494 /**
1495  * Get the offset from GMT of the current date (equivalent to the format specifier 'O').
1496  * @return {String} The 4-character offset string prefixed with + or - (e.g. '-0600')
1497  */
1498 Date.prototype.getGMTOffset = function() {
1499     return (this.getTimezoneOffset() > 0 ? "-" : "+")
1500         + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1501         + String.leftPad(this.getTimezoneOffset() % 60, 2, "0");
1502 };
1503
1504 /**
1505  * Get the offset from GMT of the current date (equivalent to the format specifier 'P').
1506  * @return {String} 2-characters representing hours and 2-characters representing minutes
1507  * seperated by a colon and prefixed with + or - (e.g. '-06:00')
1508  */
1509 Date.prototype.getGMTColonOffset = function() {
1510         return (this.getTimezoneOffset() > 0 ? "-" : "+")
1511                 + String.leftPad(Math.abs(Math.floor(this.getTimezoneOffset() / 60)), 2, "0")
1512                 + ":"
1513                 + String.leftPad(this.getTimezoneOffset() %60, 2, "0");
1514 }
1515
1516 /**
1517  * Get the numeric day number of the year, adjusted for leap year.
1518  * @return {Number} 0 through 364 (365 in leap years)
1519  */
1520 Date.prototype.getDayOfYear = function() {
1521     var num = 0;
1522     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1523     for (var i = 0; i < this.getMonth(); ++i) {
1524         num += Date.daysInMonth[i];
1525     }
1526     return num + this.getDate() - 1;
1527 };
1528
1529 /**
1530  * Get the string representation of the numeric week number of the year
1531  * (equivalent to the format specifier 'W').
1532  * @return {String} '00' through '52'
1533  */
1534 Date.prototype.getWeekOfYear = function() {
1535     // Skip to Thursday of this week
1536     var now = this.getDayOfYear() + (4 - this.getDay());
1537     // Find the first Thursday of the year
1538     var jan1 = new Date(this.getFullYear(), 0, 1);
1539     var then = (7 - jan1.getDay() + 4);
1540     return String.leftPad(((now - then) / 7) + 1, 2, "0");
1541 };
1542
1543 /**
1544  * Whether or not the current date is in a leap year.
1545  * @return {Boolean} True if the current date is in a leap year, else false
1546  */
1547 Date.prototype.isLeapYear = function() {
1548     var year = this.getFullYear();
1549     return ((year & 3) == 0 && (year % 100 || (year % 400 == 0 && year)));
1550 };
1551
1552 /**
1553  * Get the first day of the current month, adjusted for leap year.  The returned value
1554  * is the numeric day index within the week (0-6) which can be used in conjunction with
1555  * the {@link #monthNames} array to retrieve the textual day name.
1556  * Example:
1557  *<pre><code>
1558 var dt = new Date('1/10/2007');
1559 document.write(Date.dayNames[dt.getFirstDayOfMonth()]); //output: 'Monday'
1560 </code></pre>
1561  * @return {Number} The day number (0-6)
1562  */
1563 Date.prototype.getFirstDayOfMonth = function() {
1564     var day = (this.getDay() - (this.getDate() - 1)) % 7;
1565     return (day < 0) ? (day + 7) : day;
1566 };
1567
1568 /**
1569  * Get the last day of the current month, adjusted for leap year.  The returned value
1570  * is the numeric day index within the week (0-6) which can be used in conjunction with
1571  * the {@link #monthNames} array to retrieve the textual day name.
1572  * Example:
1573  *<pre><code>
1574 var dt = new Date('1/10/2007');
1575 document.write(Date.dayNames[dt.getLastDayOfMonth()]); //output: 'Wednesday'
1576 </code></pre>
1577  * @return {Number} The day number (0-6)
1578  */
1579 Date.prototype.getLastDayOfMonth = function() {
1580     var day = (this.getDay() + (Date.daysInMonth[this.getMonth()] - this.getDate())) % 7;
1581     return (day < 0) ? (day + 7) : day;
1582 };
1583
1584
1585 /**
1586  * Get the first date of this date's month
1587  * @return {Date}
1588  */
1589 Date.prototype.getFirstDateOfMonth = function() {
1590     return new Date(this.getFullYear(), this.getMonth(), 1);
1591 };
1592
1593 /**
1594  * Get the last date of this date's month
1595  * @return {Date}
1596  */
1597 Date.prototype.getLastDateOfMonth = function() {
1598     return new Date(this.getFullYear(), this.getMonth(), this.getDaysInMonth());
1599 };
1600 /**
1601  * Get the number of days in the current month, adjusted for leap year.
1602  * @return {Number} The number of days in the month
1603  */
1604 Date.prototype.getDaysInMonth = function() {
1605     Date.daysInMonth[1] = this.isLeapYear() ? 29 : 28;
1606     return Date.daysInMonth[this.getMonth()];
1607 };
1608
1609 /**
1610  * Get the English ordinal suffix of the current day (equivalent to the format specifier 'S').
1611  * @return {String} 'st, 'nd', 'rd' or 'th'
1612  */
1613 Date.prototype.getSuffix = function() {
1614     switch (this.getDate()) {
1615         case 1:
1616         case 21:
1617         case 31:
1618             return "st";
1619         case 2:
1620         case 22:
1621             return "nd";
1622         case 3:
1623         case 23:
1624             return "rd";
1625         default:
1626             return "th";
1627     }
1628 };
1629
1630 // private
1631 Date.daysInMonth = [31,28,31,30,31,30,31,31,30,31,30,31];
1632
1633 /**
1634  * An array of textual month names.
1635  * Override these values for international dates, for example...
1636  * Date.monthNames = ['JanInYourLang', 'FebInYourLang', ...];
1637  * @type Array
1638  * @static
1639  */
1640 Date.monthNames =
1641    ["January",
1642     "February",
1643     "March",
1644     "April",
1645     "May",
1646     "June",
1647     "July",
1648     "August",
1649     "September",
1650     "October",
1651     "November",
1652     "December"];
1653
1654 /**
1655  * An array of textual day names.
1656  * Override these values for international dates, for example...
1657  * Date.dayNames = ['SundayInYourLang', 'MondayInYourLang', ...];
1658  * @type Array
1659  * @static
1660  */
1661 Date.dayNames =
1662    ["Sunday",
1663     "Monday",
1664     "Tuesday",
1665     "Wednesday",
1666     "Thursday",
1667     "Friday",
1668     "Saturday"];
1669
1670 // private
1671 Date.y2kYear = 50;
1672 // private
1673 Date.monthNumbers = {
1674     Jan:0,
1675     Feb:1,
1676     Mar:2,
1677     Apr:3,
1678     May:4,
1679     Jun:5,
1680     Jul:6,
1681     Aug:7,
1682     Sep:8,
1683     Oct:9,
1684     Nov:10,
1685     Dec:11};
1686
1687 /**
1688  * Creates and returns a new Date instance with the exact same date value as the called instance.
1689  * Dates are copied and passed by reference, so if a copied date variable is modified later, the original
1690  * variable will also be changed.  When the intention is to create a new variable that will not
1691  * modify the original instance, you should create a clone.
1692  *
1693  * Example of correctly cloning a date:
1694  * <pre><code>
1695 //wrong way:
1696 var orig = new Date('10/1/2006');
1697 var copy = orig;
1698 copy.setDate(5);
1699 document.write(orig);  //returns 'Thu Oct 05 2006'!
1700
1701 //correct way:
1702 var orig = new Date('10/1/2006');
1703 var copy = orig.clone();
1704 copy.setDate(5);
1705 document.write(orig);  //returns 'Thu Oct 01 2006'
1706 </code></pre>
1707  * @return {Date} The new Date instance
1708  */
1709 Date.prototype.clone = function() {
1710         return new Date(this.getTime());
1711 };
1712
1713 /**
1714  * Clears any time information from this date
1715  @param {Boolean} clone true to create a clone of this date, clear the time and return it
1716  @return {Date} this or the clone
1717  */
1718 Date.prototype.clearTime = function(clone){
1719     if(clone){
1720         return this.clone().clearTime();
1721     }
1722     this.setHours(0);
1723     this.setMinutes(0);
1724     this.setSeconds(0);
1725     this.setMilliseconds(0);
1726     return this;
1727 };
1728
1729 // private
1730 // safari setMonth is broken -- check that this is only donw once...
1731 if(Roo.isSafari && typeof(Date.brokenSetMonth) == 'undefined'){
1732     Date.brokenSetMonth = Date.prototype.setMonth;
1733         Date.prototype.setMonth = function(num){
1734                 if(num <= -1){
1735                         var n = Math.ceil(-num);
1736                         var back_year = Math.ceil(n/12);
1737                         var month = (n % 12) ? 12 - n % 12 : 0 ;
1738                         this.setFullYear(this.getFullYear() - back_year);
1739                         return Date.brokenSetMonth.call(this, month);
1740                 } else {
1741                         return Date.brokenSetMonth.apply(this, arguments);
1742                 }
1743         };
1744 }
1745
1746 /** Date interval constant 
1747 * @static 
1748 * @type String */
1749 Date.MILLI = "ms";
1750 /** Date interval constant 
1751 * @static 
1752 * @type String */
1753 Date.SECOND = "s";
1754 /** Date interval constant 
1755 * @static 
1756 * @type String */
1757 Date.MINUTE = "mi";
1758 /** Date interval constant 
1759 * @static 
1760 * @type String */
1761 Date.HOUR = "h";
1762 /** Date interval constant 
1763 * @static 
1764 * @type String */
1765 Date.DAY = "d";
1766 /** Date interval constant 
1767 * @static 
1768 * @type String */
1769 Date.MONTH = "mo";
1770 /** Date interval constant 
1771 * @static 
1772 * @type String */
1773 Date.YEAR = "y";
1774
1775 /**
1776  * Provides a convenient method of performing basic date arithmetic.  This method
1777  * does not modify the Date instance being called - it creates and returns
1778  * a new Date instance containing the resulting date value.
1779  *
1780  * Examples:
1781  * <pre><code>
1782 //Basic usage:
1783 var dt = new Date('10/29/2006').add(Date.DAY, 5);
1784 document.write(dt); //returns 'Fri Oct 06 2006 00:00:00'
1785
1786 //Negative values will subtract correctly:
1787 var dt2 = new Date('10/1/2006').add(Date.DAY, -5);
1788 document.write(dt2); //returns 'Tue Sep 26 2006 00:00:00'
1789
1790 //You can even chain several calls together in one line!
1791 var dt3 = new Date('10/1/2006').add(Date.DAY, 5).add(Date.HOUR, 8).add(Date.MINUTE, -30);
1792 document.write(dt3); //returns 'Fri Oct 06 2006 07:30:00'
1793  </code></pre>
1794  *
1795  * @param {String} interval   A valid date interval enum value
1796  * @param {Number} value      The amount to add to the current date
1797  * @return {Date} The new Date instance
1798  */
1799 Date.prototype.add = function(interval, value){
1800   var d = this.clone();
1801   if (!interval || value === 0) { return d; }
1802   switch(interval.toLowerCase()){
1803     case Date.MILLI:
1804       d.setMilliseconds(this.getMilliseconds() + value);
1805       break;
1806     case Date.SECOND:
1807       d.setSeconds(this.getSeconds() + value);
1808       break;
1809     case Date.MINUTE:
1810       d.setMinutes(this.getMinutes() + value);
1811       break;
1812     case Date.HOUR:
1813       d.setHours(this.getHours() + value);
1814       break;
1815     case Date.DAY:
1816       d.setDate(this.getDate() + value);
1817       break;
1818     case Date.MONTH:
1819       var day = this.getDate();
1820       if(day > 28){
1821           day = Math.min(day, this.getFirstDateOfMonth().add('mo', value).getLastDateOfMonth().getDate());
1822       }
1823       d.setDate(day);
1824       d.setMonth(this.getMonth() + value);
1825       break;
1826     case Date.YEAR:
1827       d.setFullYear(this.getFullYear() + value);
1828       break;
1829   }
1830   return d;
1831 };
1832 /*
1833  * Based on:
1834  * Ext JS Library 1.1.1
1835  * Copyright(c) 2006-2007, Ext JS, LLC.
1836  *
1837  * Originally Released Under LGPL - original licence link has changed is not relivant.
1838  *
1839  * Fork - LGPL
1840  * <script type="text/javascript">
1841  */
1842
1843 /**
1844  * @class Roo.lib.Dom
1845  * @static
1846  * 
1847  * Dom utils (from YIU afaik)
1848  * 
1849  **/
1850 Roo.lib.Dom = {
1851     /**
1852      * Get the view width
1853      * @param {Boolean} full True will get the full document, otherwise it's the view width
1854      * @return {Number} The width
1855      */
1856      
1857     getViewWidth : function(full) {
1858         return full ? this.getDocumentWidth() : this.getViewportWidth();
1859     },
1860     /**
1861      * Get the view height
1862      * @param {Boolean} full True will get the full document, otherwise it's the view height
1863      * @return {Number} The height
1864      */
1865     getViewHeight : function(full) {
1866         return full ? this.getDocumentHeight() : this.getViewportHeight();
1867     },
1868
1869     getDocumentHeight: function() {
1870         var scrollHeight = (document.compatMode != "CSS1Compat") ? document.body.scrollHeight : document.documentElement.scrollHeight;
1871         return Math.max(scrollHeight, this.getViewportHeight());
1872     },
1873
1874     getDocumentWidth: function() {
1875         var scrollWidth = (document.compatMode != "CSS1Compat") ? document.body.scrollWidth : document.documentElement.scrollWidth;
1876         return Math.max(scrollWidth, this.getViewportWidth());
1877     },
1878
1879     getViewportHeight: function() {
1880         var height = self.innerHeight;
1881         var mode = document.compatMode;
1882
1883         if ((mode || Roo.isIE) && !Roo.isOpera) {
1884             height = (mode == "CSS1Compat") ?
1885                      document.documentElement.clientHeight :
1886                      document.body.clientHeight;
1887         }
1888
1889         return height;
1890     },
1891
1892     getViewportWidth: function() {
1893         var width = self.innerWidth;
1894         var mode = document.compatMode;
1895
1896         if (mode || Roo.isIE) {
1897             width = (mode == "CSS1Compat") ?
1898                     document.documentElement.clientWidth :
1899                     document.body.clientWidth;
1900         }
1901         return width;
1902     },
1903
1904     isAncestor : function(p, c) {
1905         p = Roo.getDom(p);
1906         c = Roo.getDom(c);
1907         if (!p || !c) {
1908             return false;
1909         }
1910
1911         if (p.contains && !Roo.isSafari) {
1912             return p.contains(c);
1913         } else if (p.compareDocumentPosition) {
1914             return !!(p.compareDocumentPosition(c) & 16);
1915         } else {
1916             var parent = c.parentNode;
1917             while (parent) {
1918                 if (parent == p) {
1919                     return true;
1920                 }
1921                 else if (!parent.tagName || parent.tagName.toUpperCase() == "HTML") {
1922                     return false;
1923                 }
1924                 parent = parent.parentNode;
1925             }
1926             return false;
1927         }
1928     },
1929
1930     getRegion : function(el) {
1931         return Roo.lib.Region.getRegion(el);
1932     },
1933
1934     getY : function(el) {
1935         return this.getXY(el)[1];
1936     },
1937
1938     getX : function(el) {
1939         return this.getXY(el)[0];
1940     },
1941
1942     getXY : function(el) {
1943         var p, pe, b, scroll, bd = document.body;
1944         el = Roo.getDom(el);
1945         var fly = Roo.lib.AnimBase.fly;
1946         if (el.getBoundingClientRect) {
1947             b = el.getBoundingClientRect();
1948             scroll = fly(document).getScroll();
1949             return [b.left + scroll.left, b.top + scroll.top];
1950         }
1951         var x = 0, y = 0;
1952
1953         p = el;
1954
1955         var hasAbsolute = fly(el).getStyle("position") == "absolute";
1956
1957         while (p) {
1958
1959             x += p.offsetLeft;
1960             y += p.offsetTop;
1961
1962             if (!hasAbsolute && fly(p).getStyle("position") == "absolute") {
1963                 hasAbsolute = true;
1964             }
1965
1966             if (Roo.isGecko) {
1967                 pe = fly(p);
1968
1969                 var bt = parseInt(pe.getStyle("borderTopWidth"), 10) || 0;
1970                 var bl = parseInt(pe.getStyle("borderLeftWidth"), 10) || 0;
1971
1972
1973                 x += bl;
1974                 y += bt;
1975
1976
1977                 if (p != el && pe.getStyle('overflow') != 'visible') {
1978                     x += bl;
1979                     y += bt;
1980                 }
1981             }
1982             p = p.offsetParent;
1983         }
1984
1985         if (Roo.isSafari && hasAbsolute) {
1986             x -= bd.offsetLeft;
1987             y -= bd.offsetTop;
1988         }
1989
1990         if (Roo.isGecko && !hasAbsolute) {
1991             var dbd = fly(bd);
1992             x += parseInt(dbd.getStyle("borderLeftWidth"), 10) || 0;
1993             y += parseInt(dbd.getStyle("borderTopWidth"), 10) || 0;
1994         }
1995
1996         p = el.parentNode;
1997         while (p && p != bd) {
1998             if (!Roo.isOpera || (p.tagName != 'TR' && fly(p).getStyle("display") != "inline")) {
1999                 x -= p.scrollLeft;
2000                 y -= p.scrollTop;
2001             }
2002             p = p.parentNode;
2003         }
2004         return [x, y];
2005     },
2006  
2007   
2008
2009
2010     setXY : function(el, xy) {
2011         el = Roo.fly(el, '_setXY');
2012         el.position();
2013         var pts = el.translatePoints(xy);
2014         if (xy[0] !== false) {
2015             el.dom.style.left = pts.left + "px";
2016         }
2017         if (xy[1] !== false) {
2018             el.dom.style.top = pts.top + "px";
2019         }
2020     },
2021
2022     setX : function(el, x) {
2023         this.setXY(el, [x, false]);
2024     },
2025
2026     setY : function(el, y) {
2027         this.setXY(el, [false, y]);
2028     }
2029 };
2030 /*
2031  * Portions of this file are based on pieces of Yahoo User Interface Library
2032  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2033  * YUI licensed under the BSD License:
2034  * http://developer.yahoo.net/yui/license.txt
2035  * <script type="text/javascript">
2036  *
2037  */
2038
2039 Roo.lib.Event = function() {
2040     var loadComplete = false;
2041     var listeners = [];
2042     var unloadListeners = [];
2043     var retryCount = 0;
2044     var onAvailStack = [];
2045     var counter = 0;
2046     var lastError = null;
2047
2048     return {
2049         POLL_RETRYS: 200,
2050         POLL_INTERVAL: 20,
2051         EL: 0,
2052         TYPE: 1,
2053         FN: 2,
2054         WFN: 3,
2055         OBJ: 3,
2056         ADJ_SCOPE: 4,
2057         _interval: null,
2058
2059         startInterval: function() {
2060             if (!this._interval) {
2061                 var self = this;
2062                 var callback = function() {
2063                     self._tryPreloadAttach();
2064                 };
2065                 this._interval = setInterval(callback, this.POLL_INTERVAL);
2066
2067             }
2068         },
2069
2070         onAvailable: function(p_id, p_fn, p_obj, p_override) {
2071             onAvailStack.push({ id:         p_id,
2072                 fn:         p_fn,
2073                 obj:        p_obj,
2074                 override:   p_override,
2075                 checkReady: false    });
2076
2077             retryCount = this.POLL_RETRYS;
2078             this.startInterval();
2079         },
2080
2081
2082         addListener: function(el, eventName, fn) {
2083             el = Roo.getDom(el);
2084             if (!el || !fn) {
2085                 return false;
2086             }
2087
2088             if ("unload" == eventName) {
2089                 unloadListeners[unloadListeners.length] =
2090                 [el, eventName, fn];
2091                 return true;
2092             }
2093
2094             var wrappedFn = function(e) {
2095                 return fn(Roo.lib.Event.getEvent(e));
2096             };
2097
2098             var li = [el, eventName, fn, wrappedFn];
2099
2100             var index = listeners.length;
2101             listeners[index] = li;
2102
2103             this.doAdd(el, eventName, wrappedFn, false);
2104             return true;
2105
2106         },
2107
2108
2109         removeListener: function(el, eventName, fn) {
2110             var i, len;
2111
2112             el = Roo.getDom(el);
2113
2114             if(!fn) {
2115                 return this.purgeElement(el, false, eventName);
2116             }
2117
2118
2119             if ("unload" == eventName) {
2120
2121                 for (i = 0,len = unloadListeners.length; i < len; i++) {
2122                     var li = unloadListeners[i];
2123                     if (li &&
2124                         li[0] == el &&
2125                         li[1] == eventName &&
2126                         li[2] == fn) {
2127                         unloadListeners.splice(i, 1);
2128                         return true;
2129                     }
2130                 }
2131
2132                 return false;
2133             }
2134
2135             var cacheItem = null;
2136
2137
2138             var index = arguments[3];
2139
2140             if ("undefined" == typeof index) {
2141                 index = this._getCacheIndex(el, eventName, fn);
2142             }
2143
2144             if (index >= 0) {
2145                 cacheItem = listeners[index];
2146             }
2147
2148             if (!el || !cacheItem) {
2149                 return false;
2150             }
2151
2152             this.doRemove(el, eventName, cacheItem[this.WFN], false);
2153
2154             delete listeners[index][this.WFN];
2155             delete listeners[index][this.FN];
2156             listeners.splice(index, 1);
2157
2158             return true;
2159
2160         },
2161
2162
2163         getTarget: function(ev, resolveTextNode) {
2164             ev = ev.browserEvent || ev;
2165             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2166             var t = ev.target || ev.srcElement;
2167             return this.resolveTextNode(t);
2168         },
2169
2170
2171         resolveTextNode: function(node) {
2172             if (Roo.isSafari && node && 3 == node.nodeType) {
2173                 return node.parentNode;
2174             } else {
2175                 return node;
2176             }
2177         },
2178
2179
2180         getPageX: function(ev) {
2181             ev = ev.browserEvent || ev;
2182             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2183             var x = ev.pageX;
2184             if (!x && 0 !== x) {
2185                 x = ev.clientX || 0;
2186
2187                 if (Roo.isIE) {
2188                     x += this.getScroll()[1];
2189                 }
2190             }
2191
2192             return x;
2193         },
2194
2195
2196         getPageY: function(ev) {
2197             ev = ev.browserEvent || ev;
2198             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2199             var y = ev.pageY;
2200             if (!y && 0 !== y) {
2201                 y = ev.clientY || 0;
2202
2203                 if (Roo.isIE) {
2204                     y += this.getScroll()[0];
2205                 }
2206             }
2207
2208
2209             return y;
2210         },
2211
2212
2213         getXY: function(ev) {
2214             ev = ev.browserEvent || ev;
2215             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2216             return [this.getPageX(ev), this.getPageY(ev)];
2217         },
2218
2219
2220         getRelatedTarget: function(ev) {
2221             ev = ev.browserEvent || ev;
2222             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2223             var t = ev.relatedTarget;
2224             if (!t) {
2225                 if (ev.type == "mouseout") {
2226                     t = ev.toElement;
2227                 } else if (ev.type == "mouseover") {
2228                     t = ev.fromElement;
2229                 }
2230             }
2231
2232             return this.resolveTextNode(t);
2233         },
2234
2235
2236         getTime: function(ev) {
2237             ev = ev.browserEvent || ev;
2238             ev = ev.touches ? (ev.touches[0] || ev.changedTouches[0] || ev )  : ev;
2239             if (!ev.time) {
2240                 var t = new Date().getTime();
2241                 try {
2242                     ev.time = t;
2243                 } catch(ex) {
2244                     this.lastError = ex;
2245                     return t;
2246                 }
2247             }
2248
2249             return ev.time;
2250         },
2251
2252
2253         stopEvent: function(ev) {
2254             this.stopPropagation(ev);
2255             this.preventDefault(ev);
2256         },
2257
2258
2259         stopPropagation: function(ev) {
2260             ev = ev.browserEvent || ev;
2261             if (ev.stopPropagation) {
2262                 ev.stopPropagation();
2263             } else {
2264                 ev.cancelBubble = true;
2265             }
2266         },
2267
2268
2269         preventDefault: function(ev) {
2270             ev = ev.browserEvent || ev;
2271             if(ev.preventDefault) {
2272                 ev.preventDefault();
2273             } else {
2274                 ev.returnValue = false;
2275             }
2276         },
2277
2278
2279         getEvent: function(e) {
2280             var ev = e || window.event;
2281             if (!ev) {
2282                 var c = this.getEvent.caller;
2283                 while (c) {
2284                     ev = c.arguments[0];
2285                     if (ev && Event == ev.constructor) {
2286                         break;
2287                     }
2288                     c = c.caller;
2289                 }
2290             }
2291             return ev;
2292         },
2293
2294
2295         getCharCode: function(ev) {
2296             ev = ev.browserEvent || ev;
2297             return ev.charCode || ev.keyCode || 0;
2298         },
2299
2300
2301         _getCacheIndex: function(el, eventName, fn) {
2302             for (var i = 0,len = listeners.length; i < len; ++i) {
2303                 var li = listeners[i];
2304                 if (li &&
2305                     li[this.FN] == fn &&
2306                     li[this.EL] == el &&
2307                     li[this.TYPE] == eventName) {
2308                     return i;
2309                 }
2310             }
2311
2312             return -1;
2313         },
2314
2315
2316         elCache: {},
2317
2318
2319         getEl: function(id) {
2320             return document.getElementById(id);
2321         },
2322
2323
2324         clearCache: function() {
2325         },
2326
2327
2328         _load: function(e) {
2329             loadComplete = true;
2330             var EU = Roo.lib.Event;
2331
2332
2333             if (Roo.isIE) {
2334                 EU.doRemove(window, "load", EU._load);
2335             }
2336         },
2337
2338
2339         _tryPreloadAttach: function() {
2340
2341             if (this.locked) {
2342                 return false;
2343             }
2344
2345             this.locked = true;
2346
2347
2348             var tryAgain = !loadComplete;
2349             if (!tryAgain) {
2350                 tryAgain = (retryCount > 0);
2351             }
2352
2353
2354             var notAvail = [];
2355             for (var i = 0,len = onAvailStack.length; i < len; ++i) {
2356                 var item = onAvailStack[i];
2357                 if (item) {
2358                     var el = this.getEl(item.id);
2359
2360                     if (el) {
2361                         if (!item.checkReady ||
2362                             loadComplete ||
2363                             el.nextSibling ||
2364                             (document && document.body)) {
2365
2366                             var scope = el;
2367                             if (item.override) {
2368                                 if (item.override === true) {
2369                                     scope = item.obj;
2370                                 } else {
2371                                     scope = item.override;
2372                                 }
2373                             }
2374                             item.fn.call(scope, item.obj);
2375                             onAvailStack[i] = null;
2376                         }
2377                     } else {
2378                         notAvail.push(item);
2379                     }
2380                 }
2381             }
2382
2383             retryCount = (notAvail.length === 0) ? 0 : retryCount - 1;
2384
2385             if (tryAgain) {
2386
2387                 this.startInterval();
2388             } else {
2389                 clearInterval(this._interval);
2390                 this._interval = null;
2391             }
2392
2393             this.locked = false;
2394
2395             return true;
2396
2397         },
2398
2399
2400         purgeElement: function(el, recurse, eventName) {
2401             var elListeners = this.getListeners(el, eventName);
2402             if (elListeners) {
2403                 for (var i = 0,len = elListeners.length; i < len; ++i) {
2404                     var l = elListeners[i];
2405                     this.removeListener(el, l.type, l.fn);
2406                 }
2407             }
2408
2409             if (recurse && el && el.childNodes) {
2410                 for (i = 0,len = el.childNodes.length; i < len; ++i) {
2411                     this.purgeElement(el.childNodes[i], recurse, eventName);
2412                 }
2413             }
2414         },
2415
2416
2417         getListeners: function(el, eventName) {
2418             var results = [], searchLists;
2419             if (!eventName) {
2420                 searchLists = [listeners, unloadListeners];
2421             } else if (eventName == "unload") {
2422                 searchLists = [unloadListeners];
2423             } else {
2424                 searchLists = [listeners];
2425             }
2426
2427             for (var j = 0; j < searchLists.length; ++j) {
2428                 var searchList = searchLists[j];
2429                 if (searchList && searchList.length > 0) {
2430                     for (var i = 0,len = searchList.length; i < len; ++i) {
2431                         var l = searchList[i];
2432                         if (l && l[this.EL] === el &&
2433                             (!eventName || eventName === l[this.TYPE])) {
2434                             results.push({
2435                                 type:   l[this.TYPE],
2436                                 fn:     l[this.FN],
2437                                 obj:    l[this.OBJ],
2438                                 adjust: l[this.ADJ_SCOPE],
2439                                 index:  i
2440                             });
2441                         }
2442                     }
2443                 }
2444             }
2445
2446             return (results.length) ? results : null;
2447         },
2448
2449
2450         _unload: function(e) {
2451
2452             var EU = Roo.lib.Event, i, j, l, len, index;
2453
2454             for (i = 0,len = unloadListeners.length; i < len; ++i) {
2455                 l = unloadListeners[i];
2456                 if (l) {
2457                     var scope = window;
2458                     if (l[EU.ADJ_SCOPE]) {
2459                         if (l[EU.ADJ_SCOPE] === true) {
2460                             scope = l[EU.OBJ];
2461                         } else {
2462                             scope = l[EU.ADJ_SCOPE];
2463                         }
2464                     }
2465                     l[EU.FN].call(scope, EU.getEvent(e), l[EU.OBJ]);
2466                     unloadListeners[i] = null;
2467                     l = null;
2468                     scope = null;
2469                 }
2470             }
2471
2472             unloadListeners = null;
2473
2474             if (listeners && listeners.length > 0) {
2475                 j = listeners.length;
2476                 while (j) {
2477                     index = j - 1;
2478                     l = listeners[index];
2479                     if (l) {
2480                         EU.removeListener(l[EU.EL], l[EU.TYPE],
2481                                 l[EU.FN], index);
2482                     }
2483                     j = j - 1;
2484                 }
2485                 l = null;
2486
2487                 EU.clearCache();
2488             }
2489
2490             EU.doRemove(window, "unload", EU._unload);
2491
2492         },
2493
2494
2495         getScroll: function() {
2496             var dd = document.documentElement, db = document.body;
2497             if (dd && (dd.scrollTop || dd.scrollLeft)) {
2498                 return [dd.scrollTop, dd.scrollLeft];
2499             } else if (db) {
2500                 return [db.scrollTop, db.scrollLeft];
2501             } else {
2502                 return [0, 0];
2503             }
2504         },
2505
2506
2507         doAdd: function () {
2508             if (window.addEventListener) {
2509                 return function(el, eventName, fn, capture) {
2510                     el.addEventListener(eventName, fn, (capture));
2511                 };
2512             } else if (window.attachEvent) {
2513                 return function(el, eventName, fn, capture) {
2514                     el.attachEvent("on" + eventName, fn);
2515                 };
2516             } else {
2517                 return function() {
2518                 };
2519             }
2520         }(),
2521
2522
2523         doRemove: function() {
2524             if (window.removeEventListener) {
2525                 return function (el, eventName, fn, capture) {
2526                     el.removeEventListener(eventName, fn, (capture));
2527                 };
2528             } else if (window.detachEvent) {
2529                 return function (el, eventName, fn) {
2530                     el.detachEvent("on" + eventName, fn);
2531                 };
2532             } else {
2533                 return function() {
2534                 };
2535             }
2536         }()
2537     };
2538     
2539 }();
2540 (function() {     
2541    
2542     var E = Roo.lib.Event;
2543     E.on = E.addListener;
2544     E.un = E.removeListener;
2545
2546     if (document && document.body) {
2547         E._load();
2548     } else {
2549         E.doAdd(window, "load", E._load);
2550     }
2551     E.doAdd(window, "unload", E._unload);
2552     E._tryPreloadAttach();
2553 })();
2554
2555 /*
2556  * Portions of this file are based on pieces of Yahoo User Interface Library
2557  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
2558  * YUI licensed under the BSD License:
2559  * http://developer.yahoo.net/yui/license.txt
2560  * <script type="text/javascript">
2561  *
2562  */
2563
2564 (function() {
2565     /**
2566      * @class Roo.lib.Ajax
2567      *
2568      */
2569     Roo.lib.Ajax = {
2570         /**
2571          * @static 
2572          */
2573         request : function(method, uri, cb, data, options) {
2574             if(options){
2575                 var hs = options.headers;
2576                 if(hs){
2577                     for(var h in hs){
2578                         if(hs.hasOwnProperty(h)){
2579                             this.initHeader(h, hs[h], false);
2580                         }
2581                     }
2582                 }
2583                 if(options.xmlData){
2584                     this.initHeader('Content-Type', 'text/xml', false);
2585                     method = 'POST';
2586                     data = options.xmlData;
2587                 }
2588             }
2589
2590             return this.asyncRequest(method, uri, cb, data);
2591         },
2592
2593         serializeForm : function(form) {
2594             if(typeof form == 'string') {
2595                 form = (document.getElementById(form) || document.forms[form]);
2596             }
2597
2598             var el, name, val, disabled, data = '', hasSubmit = false;
2599             for (var i = 0; i < form.elements.length; i++) {
2600                 el = form.elements[i];
2601                 disabled = form.elements[i].disabled;
2602                 name = form.elements[i].name;
2603                 val = form.elements[i].value;
2604
2605                 if (!disabled && name){
2606                     switch (el.type)
2607                             {
2608                         case 'select-one':
2609                         case 'select-multiple':
2610                             for (var j = 0; j < el.options.length; j++) {
2611                                 if (el.options[j].selected) {
2612                                     if (Roo.isIE) {
2613                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].attributes['value'].specified ? el.options[j].value : el.options[j].text) + '&';
2614                                     }
2615                                     else {
2616                                         data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(el.options[j].hasAttribute('value') ? el.options[j].value : el.options[j].text) + '&';
2617                                     }
2618                                 }
2619                             }
2620                             break;
2621                         case 'radio':
2622                         case 'checkbox':
2623                             if (el.checked) {
2624                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2625                             }
2626                             break;
2627                         case 'file':
2628
2629                         case undefined:
2630
2631                         case 'reset':
2632
2633                         case 'button':
2634
2635                             break;
2636                         case 'submit':
2637                             if(hasSubmit == false) {
2638                                 data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2639                                 hasSubmit = true;
2640                             }
2641                             break;
2642                         default:
2643                             data += Roo.encodeURIComponent(name) + '=' + Roo.encodeURIComponent(val) + '&';
2644                             break;
2645                     }
2646                 }
2647             }
2648             data = data.substr(0, data.length - 1);
2649             return data;
2650         },
2651
2652         headers:{},
2653
2654         hasHeaders:false,
2655
2656         useDefaultHeader:true,
2657
2658         defaultPostHeader:'application/x-www-form-urlencoded',
2659
2660         useDefaultXhrHeader:true,
2661
2662         defaultXhrHeader:'XMLHttpRequest',
2663
2664         hasDefaultHeaders:true,
2665
2666         defaultHeaders:{},
2667
2668         poll:{},
2669
2670         timeout:{},
2671
2672         pollInterval:50,
2673
2674         transactionId:0,
2675
2676         setProgId:function(id)
2677         {
2678             this.activeX.unshift(id);
2679         },
2680
2681         setDefaultPostHeader:function(b)
2682         {
2683             this.useDefaultHeader = b;
2684         },
2685
2686         setDefaultXhrHeader:function(b)
2687         {
2688             this.useDefaultXhrHeader = b;
2689         },
2690
2691         setPollingInterval:function(i)
2692         {
2693             if (typeof i == 'number' && isFinite(i)) {
2694                 this.pollInterval = i;
2695             }
2696         },
2697
2698         createXhrObject:function(transactionId)
2699         {
2700             var obj,http;
2701             try
2702             {
2703
2704                 http = new XMLHttpRequest();
2705
2706                 obj = { conn:http, tId:transactionId };
2707             }
2708             catch(e)
2709             {
2710                 for (var i = 0; i < this.activeX.length; ++i) {
2711                     try
2712                     {
2713
2714                         http = new ActiveXObject(this.activeX[i]);
2715
2716                         obj = { conn:http, tId:transactionId };
2717                         break;
2718                     }
2719                     catch(e) {
2720                     }
2721                 }
2722             }
2723             finally
2724             {
2725                 return obj;
2726             }
2727         },
2728
2729         getConnectionObject:function()
2730         {
2731             var o;
2732             var tId = this.transactionId;
2733
2734             try
2735             {
2736                 o = this.createXhrObject(tId);
2737                 if (o) {
2738                     this.transactionId++;
2739                 }
2740             }
2741             catch(e) {
2742             }
2743             finally
2744             {
2745                 return o;
2746             }
2747         },
2748
2749         asyncRequest:function(method, uri, callback, postData)
2750         {
2751             var o = this.getConnectionObject();
2752
2753             if (!o) {
2754                 return null;
2755             }
2756             else {
2757                 o.conn.open(method, uri, true);
2758
2759                 if (this.useDefaultXhrHeader) {
2760                     if (!this.defaultHeaders['X-Requested-With']) {
2761                         this.initHeader('X-Requested-With', this.defaultXhrHeader, true);
2762                     }
2763                 }
2764
2765                 if(postData && this.useDefaultHeader){
2766                     this.initHeader('Content-Type', this.defaultPostHeader);
2767                 }
2768
2769                  if (this.hasDefaultHeaders || this.hasHeaders) {
2770                     this.setHeader(o);
2771                 }
2772
2773                 this.handleReadyState(o, callback);
2774                 o.conn.send(postData || null);
2775
2776                 return o;
2777             }
2778         },
2779
2780         handleReadyState:function(o, callback)
2781         {
2782             var oConn = this;
2783
2784             if (callback && callback.timeout) {
2785                 
2786                 this.timeout[o.tId] = window.setTimeout(function() {
2787                     oConn.abort(o, callback, true);
2788                 }, callback.timeout);
2789             }
2790
2791             this.poll[o.tId] = window.setInterval(
2792                     function() {
2793                         if (o.conn && o.conn.readyState == 4) {
2794                             window.clearInterval(oConn.poll[o.tId]);
2795                             delete oConn.poll[o.tId];
2796
2797                             if(callback && callback.timeout) {
2798                                 window.clearTimeout(oConn.timeout[o.tId]);
2799                                 delete oConn.timeout[o.tId];
2800                             }
2801
2802                             oConn.handleTransactionResponse(o, callback);
2803                         }
2804                     }
2805                     , this.pollInterval);
2806         },
2807
2808         handleTransactionResponse:function(o, callback, isAbort)
2809         {
2810
2811             if (!callback) {
2812                 this.releaseObject(o);
2813                 return;
2814             }
2815
2816             var httpStatus, responseObject;
2817
2818             try
2819             {
2820                 if (o.conn.status !== undefined && o.conn.status != 0) {
2821                     httpStatus = o.conn.status;
2822                 }
2823                 else {
2824                     httpStatus = 13030;
2825                 }
2826             }
2827             catch(e) {
2828
2829
2830                 httpStatus = 13030;
2831             }
2832
2833             if (httpStatus >= 200 && httpStatus < 300) {
2834                 responseObject = this.createResponseObject(o, callback.argument);
2835                 if (callback.success) {
2836                     if (!callback.scope) {
2837                         callback.success(responseObject);
2838                     }
2839                     else {
2840
2841
2842                         callback.success.apply(callback.scope, [responseObject]);
2843                     }
2844                 }
2845             }
2846             else {
2847                 switch (httpStatus) {
2848
2849                     case 12002:
2850                     case 12029:
2851                     case 12030:
2852                     case 12031:
2853                     case 12152:
2854                     case 13030:
2855                         responseObject = this.createExceptionObject(o.tId, callback.argument, (isAbort ? isAbort : false));
2856                         if (callback.failure) {
2857                             if (!callback.scope) {
2858                                 callback.failure(responseObject);
2859                             }
2860                             else {
2861                                 callback.failure.apply(callback.scope, [responseObject]);
2862                             }
2863                         }
2864                         break;
2865                     default:
2866                         responseObject = this.createResponseObject(o, callback.argument);
2867                         if (callback.failure) {
2868                             if (!callback.scope) {
2869                                 callback.failure(responseObject);
2870                             }
2871                             else {
2872                                 callback.failure.apply(callback.scope, [responseObject]);
2873                             }
2874                         }
2875                 }
2876             }
2877
2878             this.releaseObject(o);
2879             responseObject = null;
2880         },
2881
2882         createResponseObject:function(o, callbackArg)
2883         {
2884             var obj = {};
2885             var headerObj = {};
2886
2887             try
2888             {
2889                 var headerStr = o.conn.getAllResponseHeaders();
2890                 var header = headerStr.split('\n');
2891                 for (var i = 0; i < header.length; i++) {
2892                     var delimitPos = header[i].indexOf(':');
2893                     if (delimitPos != -1) {
2894                         headerObj[header[i].substring(0, delimitPos)] = header[i].substring(delimitPos + 2);
2895                     }
2896                 }
2897             }
2898             catch(e) {
2899             }
2900
2901             obj.tId = o.tId;
2902             obj.status = o.conn.status;
2903             obj.statusText = o.conn.statusText;
2904             obj.getResponseHeader = headerObj;
2905             obj.getAllResponseHeaders = headerStr;
2906             obj.responseText = o.conn.responseText;
2907             obj.responseXML = o.conn.responseXML;
2908
2909             if (typeof callbackArg !== undefined) {
2910                 obj.argument = callbackArg;
2911             }
2912
2913             return obj;
2914         },
2915
2916         createExceptionObject:function(tId, callbackArg, isAbort)
2917         {
2918             var COMM_CODE = 0;
2919             var COMM_ERROR = 'communication failure';
2920             var ABORT_CODE = -1;
2921             var ABORT_ERROR = 'transaction aborted';
2922
2923             var obj = {};
2924
2925             obj.tId = tId;
2926             if (isAbort) {
2927                 obj.status = ABORT_CODE;
2928                 obj.statusText = ABORT_ERROR;
2929             }
2930             else {
2931                 obj.status = COMM_CODE;
2932                 obj.statusText = COMM_ERROR;
2933             }
2934
2935             if (callbackArg) {
2936                 obj.argument = callbackArg;
2937             }
2938
2939             return obj;
2940         },
2941
2942         initHeader:function(label, value, isDefault)
2943         {
2944             var headerObj = (isDefault) ? this.defaultHeaders : this.headers;
2945
2946             if (headerObj[label] === undefined) {
2947                 headerObj[label] = value;
2948             }
2949             else {
2950
2951
2952                 headerObj[label] = value + "," + headerObj[label];
2953             }
2954
2955             if (isDefault) {
2956                 this.hasDefaultHeaders = true;
2957             }
2958             else {
2959                 this.hasHeaders = true;
2960             }
2961         },
2962
2963
2964         setHeader:function(o)
2965         {
2966             if (this.hasDefaultHeaders) {
2967                 for (var prop in this.defaultHeaders) {
2968                     if (this.defaultHeaders.hasOwnProperty(prop)) {
2969                         o.conn.setRequestHeader(prop, this.defaultHeaders[prop]);
2970                     }
2971                 }
2972             }
2973
2974             if (this.hasHeaders) {
2975                 for (var prop in this.headers) {
2976                     if (this.headers.hasOwnProperty(prop)) {
2977                         o.conn.setRequestHeader(prop, this.headers[prop]);
2978                     }
2979                 }
2980                 this.headers = {};
2981                 this.hasHeaders = false;
2982             }
2983         },
2984
2985         resetDefaultHeaders:function() {
2986             delete this.defaultHeaders;
2987             this.defaultHeaders = {};
2988             this.hasDefaultHeaders = false;
2989         },
2990
2991         abort:function(o, callback, isTimeout)
2992         {
2993             if(this.isCallInProgress(o)) {
2994                 o.conn.abort();
2995                 window.clearInterval(this.poll[o.tId]);
2996                 delete this.poll[o.tId];
2997                 if (isTimeout) {
2998                     delete this.timeout[o.tId];
2999                 }
3000
3001                 this.handleTransactionResponse(o, callback, true);
3002
3003                 return true;
3004             }
3005             else {
3006                 return false;
3007             }
3008         },
3009
3010
3011         isCallInProgress:function(o)
3012         {
3013             if (o && o.conn) {
3014                 return o.conn.readyState != 4 && o.conn.readyState != 0;
3015             }
3016             else {
3017
3018                 return false;
3019             }
3020         },
3021
3022
3023         releaseObject:function(o)
3024         {
3025
3026             o.conn = null;
3027
3028             o = null;
3029         },
3030
3031         activeX:[
3032         'MSXML2.XMLHTTP.3.0',
3033         'MSXML2.XMLHTTP',
3034         'Microsoft.XMLHTTP'
3035         ]
3036
3037
3038     };
3039 })();/*
3040  * Portions of this file are based on pieces of Yahoo User Interface Library
3041  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3042  * YUI licensed under the BSD License:
3043  * http://developer.yahoo.net/yui/license.txt
3044  * <script type="text/javascript">
3045  *
3046  */
3047
3048 Roo.lib.Region = function(t, r, b, l) {
3049     this.top = t;
3050     this[1] = t;
3051     this.right = r;
3052     this.bottom = b;
3053     this.left = l;
3054     this[0] = l;
3055 };
3056
3057
3058 Roo.lib.Region.prototype = {
3059     contains : function(region) {
3060         return ( region.left >= this.left &&
3061                  region.right <= this.right &&
3062                  region.top >= this.top &&
3063                  region.bottom <= this.bottom    );
3064
3065     },
3066
3067     getArea : function() {
3068         return ( (this.bottom - this.top) * (this.right - this.left) );
3069     },
3070
3071     intersect : function(region) {
3072         var t = Math.max(this.top, region.top);
3073         var r = Math.min(this.right, region.right);
3074         var b = Math.min(this.bottom, region.bottom);
3075         var l = Math.max(this.left, region.left);
3076
3077         if (b >= t && r >= l) {
3078             return new Roo.lib.Region(t, r, b, l);
3079         } else {
3080             return null;
3081         }
3082     },
3083     union : function(region) {
3084         var t = Math.min(this.top, region.top);
3085         var r = Math.max(this.right, region.right);
3086         var b = Math.max(this.bottom, region.bottom);
3087         var l = Math.min(this.left, region.left);
3088
3089         return new Roo.lib.Region(t, r, b, l);
3090     },
3091
3092     adjust : function(t, l, b, r) {
3093         this.top += t;
3094         this.left += l;
3095         this.right += r;
3096         this.bottom += b;
3097         return this;
3098     }
3099 };
3100
3101 Roo.lib.Region.getRegion = function(el) {
3102     var p = Roo.lib.Dom.getXY(el);
3103
3104     var t = p[1];
3105     var r = p[0] + el.offsetWidth;
3106     var b = p[1] + el.offsetHeight;
3107     var l = p[0];
3108
3109     return new Roo.lib.Region(t, r, b, l);
3110 };
3111 /*
3112  * Portions of this file are based on pieces of Yahoo User Interface Library
3113  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3114  * YUI licensed under the BSD License:
3115  * http://developer.yahoo.net/yui/license.txt
3116  * <script type="text/javascript">
3117  *
3118  */
3119 //@@dep Roo.lib.Region
3120
3121
3122 Roo.lib.Point = function(x, y) {
3123     if (x instanceof Array) {
3124         y = x[1];
3125         x = x[0];
3126     }
3127     this.x = this.right = this.left = this[0] = x;
3128     this.y = this.top = this.bottom = this[1] = y;
3129 };
3130
3131 Roo.lib.Point.prototype = new Roo.lib.Region();
3132 /*
3133  * Portions of this file are based on pieces of Yahoo User Interface Library
3134  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3135  * YUI licensed under the BSD License:
3136  * http://developer.yahoo.net/yui/license.txt
3137  * <script type="text/javascript">
3138  *
3139  */
3140  
3141 (function() {   
3142
3143     Roo.lib.Anim = {
3144         scroll : function(el, args, duration, easing, cb, scope) {
3145             this.run(el, args, duration, easing, cb, scope, Roo.lib.Scroll);
3146         },
3147
3148         motion : function(el, args, duration, easing, cb, scope) {
3149             this.run(el, args, duration, easing, cb, scope, Roo.lib.Motion);
3150         },
3151
3152         color : function(el, args, duration, easing, cb, scope) {
3153             this.run(el, args, duration, easing, cb, scope, Roo.lib.ColorAnim);
3154         },
3155
3156         run : function(el, args, duration, easing, cb, scope, type) {
3157             type = type || Roo.lib.AnimBase;
3158             if (typeof easing == "string") {
3159                 easing = Roo.lib.Easing[easing];
3160             }
3161             var anim = new type(el, args, duration, easing);
3162             anim.animateX(function() {
3163                 Roo.callback(cb, scope);
3164             });
3165             return anim;
3166         }
3167     };
3168 })();/*
3169  * Portions of this file are based on pieces of Yahoo User Interface Library
3170  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3171  * YUI licensed under the BSD License:
3172  * http://developer.yahoo.net/yui/license.txt
3173  * <script type="text/javascript">
3174  *
3175  */
3176
3177 (function() {    
3178     var libFlyweight;
3179     
3180     function fly(el) {
3181         if (!libFlyweight) {
3182             libFlyweight = new Roo.Element.Flyweight();
3183         }
3184         libFlyweight.dom = el;
3185         return libFlyweight;
3186     }
3187
3188     // since this uses fly! - it cant be in DOM (which does not have fly yet..)
3189     
3190    
3191     
3192     Roo.lib.AnimBase = function(el, attributes, duration, method) {
3193         if (el) {
3194             this.init(el, attributes, duration, method);
3195         }
3196     };
3197
3198     Roo.lib.AnimBase.fly = fly;
3199     
3200     
3201     
3202     Roo.lib.AnimBase.prototype = {
3203
3204         toString: function() {
3205             var el = this.getEl();
3206             var id = el.id || el.tagName;
3207             return ("Anim " + id);
3208         },
3209
3210         patterns: {
3211             noNegatives:        /width|height|opacity|padding/i,
3212             offsetAttribute:  /^((width|height)|(top|left))$/,
3213             defaultUnit:        /width|height|top$|bottom$|left$|right$/i,
3214             offsetUnit:         /\d+(em|%|en|ex|pt|in|cm|mm|pc)$/i
3215         },
3216
3217
3218         doMethod: function(attr, start, end) {
3219             return this.method(this.currentFrame, start, end - start, this.totalFrames);
3220         },
3221
3222
3223         setAttribute: function(attr, val, unit) {
3224             if (this.patterns.noNegatives.test(attr)) {
3225                 val = (val > 0) ? val : 0;
3226             }
3227
3228             Roo.fly(this.getEl(), '_anim').setStyle(attr, val + unit);
3229         },
3230
3231
3232         getAttribute: function(attr) {
3233             var el = this.getEl();
3234             var val = fly(el).getStyle(attr);
3235
3236             if (val !== 'auto' && !this.patterns.offsetUnit.test(val)) {
3237                 return parseFloat(val);
3238             }
3239
3240             var a = this.patterns.offsetAttribute.exec(attr) || [];
3241             var pos = !!( a[3] );
3242             var box = !!( a[2] );
3243
3244
3245             if (box || (fly(el).getStyle('position') == 'absolute' && pos)) {
3246                 val = el['offset' + a[0].charAt(0).toUpperCase() + a[0].substr(1)];
3247             } else {
3248                 val = 0;
3249             }
3250
3251             return val;
3252         },
3253
3254
3255         getDefaultUnit: function(attr) {
3256             if (this.patterns.defaultUnit.test(attr)) {
3257                 return 'px';
3258             }
3259
3260             return '';
3261         },
3262
3263         animateX : function(callback, scope) {
3264             var f = function() {
3265                 this.onComplete.removeListener(f);
3266                 if (typeof callback == "function") {
3267                     callback.call(scope || this, this);
3268                 }
3269             };
3270             this.onComplete.addListener(f, this);
3271             this.animate();
3272         },
3273
3274
3275         setRuntimeAttribute: function(attr) {
3276             var start;
3277             var end;
3278             var attributes = this.attributes;
3279
3280             this.runtimeAttributes[attr] = {};
3281
3282             var isset = function(prop) {
3283                 return (typeof prop !== 'undefined');
3284             };
3285
3286             if (!isset(attributes[attr]['to']) && !isset(attributes[attr]['by'])) {
3287                 return false;
3288             }
3289
3290             start = ( isset(attributes[attr]['from']) ) ? attributes[attr]['from'] : this.getAttribute(attr);
3291
3292
3293             if (isset(attributes[attr]['to'])) {
3294                 end = attributes[attr]['to'];
3295             } else if (isset(attributes[attr]['by'])) {
3296                 if (start.constructor == Array) {
3297                     end = [];
3298                     for (var i = 0, len = start.length; i < len; ++i) {
3299                         end[i] = start[i] + attributes[attr]['by'][i];
3300                     }
3301                 } else {
3302                     end = start + attributes[attr]['by'];
3303                 }
3304             }
3305
3306             this.runtimeAttributes[attr].start = start;
3307             this.runtimeAttributes[attr].end = end;
3308
3309
3310             this.runtimeAttributes[attr].unit = ( isset(attributes[attr].unit) ) ? attributes[attr]['unit'] : this.getDefaultUnit(attr);
3311         },
3312
3313
3314         init: function(el, attributes, duration, method) {
3315
3316             var isAnimated = false;
3317
3318
3319             var startTime = null;
3320
3321
3322             var actualFrames = 0;
3323
3324
3325             el = Roo.getDom(el);
3326
3327
3328             this.attributes = attributes || {};
3329
3330
3331             this.duration = duration || 1;
3332
3333
3334             this.method = method || Roo.lib.Easing.easeNone;
3335
3336
3337             this.useSeconds = true;
3338
3339
3340             this.currentFrame = 0;
3341
3342
3343             this.totalFrames = Roo.lib.AnimMgr.fps;
3344
3345
3346             this.getEl = function() {
3347                 return el;
3348             };
3349
3350
3351             this.isAnimated = function() {
3352                 return isAnimated;
3353             };
3354
3355
3356             this.getStartTime = function() {
3357                 return startTime;
3358             };
3359
3360             this.runtimeAttributes = {};
3361
3362
3363             this.animate = function() {
3364                 if (this.isAnimated()) {
3365                     return false;
3366                 }
3367
3368                 this.currentFrame = 0;
3369
3370                 this.totalFrames = ( this.useSeconds ) ? Math.ceil(Roo.lib.AnimMgr.fps * this.duration) : this.duration;
3371
3372                 Roo.lib.AnimMgr.registerElement(this);
3373             };
3374
3375
3376             this.stop = function(finish) {
3377                 if (finish) {
3378                     this.currentFrame = this.totalFrames;
3379                     this._onTween.fire();
3380                 }
3381                 Roo.lib.AnimMgr.stop(this);
3382             };
3383
3384             var onStart = function() {
3385                 this.onStart.fire();
3386
3387                 this.runtimeAttributes = {};
3388                 for (var attr in this.attributes) {
3389                     this.setRuntimeAttribute(attr);
3390                 }
3391
3392                 isAnimated = true;
3393                 actualFrames = 0;
3394                 startTime = new Date();
3395             };
3396
3397
3398             var onTween = function() {
3399                 var data = {
3400                     duration: new Date() - this.getStartTime(),
3401                     currentFrame: this.currentFrame
3402                 };
3403
3404                 data.toString = function() {
3405                     return (
3406                             'duration: ' + data.duration +
3407                             ', currentFrame: ' + data.currentFrame
3408                             );
3409                 };
3410
3411                 this.onTween.fire(data);
3412
3413                 var runtimeAttributes = this.runtimeAttributes;
3414
3415                 for (var attr in runtimeAttributes) {
3416                     this.setAttribute(attr, this.doMethod(attr, runtimeAttributes[attr].start, runtimeAttributes[attr].end), runtimeAttributes[attr].unit);
3417                 }
3418
3419                 actualFrames += 1;
3420             };
3421
3422             var onComplete = function() {
3423                 var actual_duration = (new Date() - startTime) / 1000 ;
3424
3425                 var data = {
3426                     duration: actual_duration,
3427                     frames: actualFrames,
3428                     fps: actualFrames / actual_duration
3429                 };
3430
3431                 data.toString = function() {
3432                     return (
3433                             'duration: ' + data.duration +
3434                             ', frames: ' + data.frames +
3435                             ', fps: ' + data.fps
3436                             );
3437                 };
3438
3439                 isAnimated = false;
3440                 actualFrames = 0;
3441                 this.onComplete.fire(data);
3442             };
3443
3444
3445             this._onStart = new Roo.util.Event(this);
3446             this.onStart = new Roo.util.Event(this);
3447             this.onTween = new Roo.util.Event(this);
3448             this._onTween = new Roo.util.Event(this);
3449             this.onComplete = new Roo.util.Event(this);
3450             this._onComplete = new Roo.util.Event(this);
3451             this._onStart.addListener(onStart);
3452             this._onTween.addListener(onTween);
3453             this._onComplete.addListener(onComplete);
3454         }
3455     };
3456 })();
3457 /*
3458  * Portions of this file are based on pieces of Yahoo User Interface Library
3459  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3460  * YUI licensed under the BSD License:
3461  * http://developer.yahoo.net/yui/license.txt
3462  * <script type="text/javascript">
3463  *
3464  */
3465
3466 Roo.lib.AnimMgr = new function() {
3467
3468     var thread = null;
3469
3470
3471     var queue = [];
3472
3473
3474     var tweenCount = 0;
3475
3476
3477     this.fps = 1000;
3478
3479
3480     this.delay = 1;
3481
3482
3483     this.registerElement = function(tween) {
3484         queue[queue.length] = tween;
3485         tweenCount += 1;
3486         tween._onStart.fire();
3487         this.start();
3488     };
3489
3490
3491     this.unRegister = function(tween, index) {
3492         tween._onComplete.fire();
3493         index = index || getIndex(tween);
3494         if (index != -1) {
3495             queue.splice(index, 1);
3496         }
3497
3498         tweenCount -= 1;
3499         if (tweenCount <= 0) {
3500             this.stop();
3501         }
3502     };
3503
3504
3505     this.start = function() {
3506         if (thread === null) {
3507             thread = setInterval(this.run, this.delay);
3508         }
3509     };
3510
3511
3512     this.stop = function(tween) {
3513         if (!tween) {
3514             clearInterval(thread);
3515
3516             for (var i = 0, len = queue.length; i < len; ++i) {
3517                 if (queue[0].isAnimated()) {
3518                     this.unRegister(queue[0], 0);
3519                 }
3520             }
3521
3522             queue = [];
3523             thread = null;
3524             tweenCount = 0;
3525         }
3526         else {
3527             this.unRegister(tween);
3528         }
3529     };
3530
3531
3532     this.run = function() {
3533         for (var i = 0, len = queue.length; i < len; ++i) {
3534             var tween = queue[i];
3535             if (!tween || !tween.isAnimated()) {
3536                 continue;
3537             }
3538
3539             if (tween.currentFrame < tween.totalFrames || tween.totalFrames === null)
3540             {
3541                 tween.currentFrame += 1;
3542
3543                 if (tween.useSeconds) {
3544                     correctFrame(tween);
3545                 }
3546                 tween._onTween.fire();
3547             }
3548             else {
3549                 Roo.lib.AnimMgr.stop(tween, i);
3550             }
3551         }
3552     };
3553
3554     var getIndex = function(anim) {
3555         for (var i = 0, len = queue.length; i < len; ++i) {
3556             if (queue[i] == anim) {
3557                 return i;
3558             }
3559         }
3560         return -1;
3561     };
3562
3563
3564     var correctFrame = function(tween) {
3565         var frames = tween.totalFrames;
3566         var frame = tween.currentFrame;
3567         var expected = (tween.currentFrame * tween.duration * 1000 / tween.totalFrames);
3568         var elapsed = (new Date() - tween.getStartTime());
3569         var tweak = 0;
3570
3571         if (elapsed < tween.duration * 1000) {
3572             tweak = Math.round((elapsed / expected - 1) * tween.currentFrame);
3573         } else {
3574             tweak = frames - (frame + 1);
3575         }
3576         if (tweak > 0 && isFinite(tweak)) {
3577             if (tween.currentFrame + tweak >= frames) {
3578                 tweak = frames - (frame + 1);
3579             }
3580
3581             tween.currentFrame += tweak;
3582         }
3583     };
3584 };
3585
3586     /*
3587  * Portions of this file are based on pieces of Yahoo User Interface Library
3588  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3589  * YUI licensed under the BSD License:
3590  * http://developer.yahoo.net/yui/license.txt
3591  * <script type="text/javascript">
3592  *
3593  */
3594 Roo.lib.Bezier = new function() {
3595
3596         this.getPosition = function(points, t) {
3597             var n = points.length;
3598             var tmp = [];
3599
3600             for (var i = 0; i < n; ++i) {
3601                 tmp[i] = [points[i][0], points[i][1]];
3602             }
3603
3604             for (var j = 1; j < n; ++j) {
3605                 for (i = 0; i < n - j; ++i) {
3606                     tmp[i][0] = (1 - t) * tmp[i][0] + t * tmp[parseInt(i + 1, 10)][0];
3607                     tmp[i][1] = (1 - t) * tmp[i][1] + t * tmp[parseInt(i + 1, 10)][1];
3608                 }
3609             }
3610
3611             return [ tmp[0][0], tmp[0][1] ];
3612
3613         };
3614     };/*
3615  * Portions of this file are based on pieces of Yahoo User Interface Library
3616  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3617  * YUI licensed under the BSD License:
3618  * http://developer.yahoo.net/yui/license.txt
3619  * <script type="text/javascript">
3620  *
3621  */
3622 (function() {
3623
3624     Roo.lib.ColorAnim = function(el, attributes, duration, method) {
3625         Roo.lib.ColorAnim.superclass.constructor.call(this, el, attributes, duration, method);
3626     };
3627
3628     Roo.extend(Roo.lib.ColorAnim, Roo.lib.AnimBase);
3629
3630     var fly = Roo.lib.AnimBase.fly;
3631     var Y = Roo.lib;
3632     var superclass = Y.ColorAnim.superclass;
3633     var proto = Y.ColorAnim.prototype;
3634
3635     proto.toString = function() {
3636         var el = this.getEl();
3637         var id = el.id || el.tagName;
3638         return ("ColorAnim " + id);
3639     };
3640
3641     proto.patterns.color = /color$/i;
3642     proto.patterns.rgb = /^rgb\(([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\)$/i;
3643     proto.patterns.hex = /^#?([0-9A-F]{2})([0-9A-F]{2})([0-9A-F]{2})$/i;
3644     proto.patterns.hex3 = /^#?([0-9A-F]{1})([0-9A-F]{1})([0-9A-F]{1})$/i;
3645     proto.patterns.transparent = /^transparent|rgba\(0, 0, 0, 0\)$/;
3646
3647
3648     proto.parseColor = function(s) {
3649         if (s.length == 3) {
3650             return s;
3651         }
3652
3653         var c = this.patterns.hex.exec(s);
3654         if (c && c.length == 4) {
3655             return [ parseInt(c[1], 16), parseInt(c[2], 16), parseInt(c[3], 16) ];
3656         }
3657
3658         c = this.patterns.rgb.exec(s);
3659         if (c && c.length == 4) {
3660             return [ parseInt(c[1], 10), parseInt(c[2], 10), parseInt(c[3], 10) ];
3661         }
3662
3663         c = this.patterns.hex3.exec(s);
3664         if (c && c.length == 4) {
3665             return [ parseInt(c[1] + c[1], 16), parseInt(c[2] + c[2], 16), parseInt(c[3] + c[3], 16) ];
3666         }
3667
3668         return null;
3669     };
3670     // since this uses fly! - it cant be in ColorAnim (which does not have fly yet..)
3671     proto.getAttribute = function(attr) {
3672         var el = this.getEl();
3673         if (this.patterns.color.test(attr)) {
3674             var val = fly(el).getStyle(attr);
3675
3676             if (this.patterns.transparent.test(val)) {
3677                 var parent = el.parentNode;
3678                 val = fly(parent).getStyle(attr);
3679
3680                 while (parent && this.patterns.transparent.test(val)) {
3681                     parent = parent.parentNode;
3682                     val = fly(parent).getStyle(attr);
3683                     if (parent.tagName.toUpperCase() == 'HTML') {
3684                         val = '#fff';
3685                     }
3686                 }
3687             }
3688         } else {
3689             val = superclass.getAttribute.call(this, attr);
3690         }
3691
3692         return val;
3693     };
3694     proto.getAttribute = function(attr) {
3695         var el = this.getEl();
3696         if (this.patterns.color.test(attr)) {
3697             var val = fly(el).getStyle(attr);
3698
3699             if (this.patterns.transparent.test(val)) {
3700                 var parent = el.parentNode;
3701                 val = fly(parent).getStyle(attr);
3702
3703                 while (parent && this.patterns.transparent.test(val)) {
3704                     parent = parent.parentNode;
3705                     val = fly(parent).getStyle(attr);
3706                     if (parent.tagName.toUpperCase() == 'HTML') {
3707                         val = '#fff';
3708                     }
3709                 }
3710             }
3711         } else {
3712             val = superclass.getAttribute.call(this, attr);
3713         }
3714
3715         return val;
3716     };
3717
3718     proto.doMethod = function(attr, start, end) {
3719         var val;
3720
3721         if (this.patterns.color.test(attr)) {
3722             val = [];
3723             for (var i = 0, len = start.length; i < len; ++i) {
3724                 val[i] = superclass.doMethod.call(this, attr, start[i], end[i]);
3725             }
3726
3727             val = 'rgb(' + Math.floor(val[0]) + ',' + Math.floor(val[1]) + ',' + Math.floor(val[2]) + ')';
3728         }
3729         else {
3730             val = superclass.doMethod.call(this, attr, start, end);
3731         }
3732
3733         return val;
3734     };
3735
3736     proto.setRuntimeAttribute = function(attr) {
3737         superclass.setRuntimeAttribute.call(this, attr);
3738
3739         if (this.patterns.color.test(attr)) {
3740             var attributes = this.attributes;
3741             var start = this.parseColor(this.runtimeAttributes[attr].start);
3742             var end = this.parseColor(this.runtimeAttributes[attr].end);
3743
3744             if (typeof attributes[attr]['to'] === 'undefined' && typeof attributes[attr]['by'] !== 'undefined') {
3745                 end = this.parseColor(attributes[attr].by);
3746
3747                 for (var i = 0, len = start.length; i < len; ++i) {
3748                     end[i] = start[i] + end[i];
3749                 }
3750             }
3751
3752             this.runtimeAttributes[attr].start = start;
3753             this.runtimeAttributes[attr].end = end;
3754         }
3755     };
3756 })();
3757
3758 /*
3759  * Portions of this file are based on pieces of Yahoo User Interface Library
3760  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3761  * YUI licensed under the BSD License:
3762  * http://developer.yahoo.net/yui/license.txt
3763  * <script type="text/javascript">
3764  *
3765  */
3766 Roo.lib.Easing = {
3767
3768
3769     easeNone: function (t, b, c, d) {
3770         return c * t / d + b;
3771     },
3772
3773
3774     easeIn: function (t, b, c, d) {
3775         return c * (t /= d) * t + b;
3776     },
3777
3778
3779     easeOut: function (t, b, c, d) {
3780         return -c * (t /= d) * (t - 2) + b;
3781     },
3782
3783
3784     easeBoth: function (t, b, c, d) {
3785         if ((t /= d / 2) < 1) {
3786             return c / 2 * t * t + b;
3787         }
3788
3789         return -c / 2 * ((--t) * (t - 2) - 1) + b;
3790     },
3791
3792
3793     easeInStrong: function (t, b, c, d) {
3794         return c * (t /= d) * t * t * t + b;
3795     },
3796
3797
3798     easeOutStrong: function (t, b, c, d) {
3799         return -c * ((t = t / d - 1) * t * t * t - 1) + b;
3800     },
3801
3802
3803     easeBothStrong: function (t, b, c, d) {
3804         if ((t /= d / 2) < 1) {
3805             return c / 2 * t * t * t * t + b;
3806         }
3807
3808         return -c / 2 * ((t -= 2) * t * t * t - 2) + b;
3809     },
3810
3811
3812
3813     elasticIn: function (t, b, c, d, a, p) {
3814         if (t == 0) {
3815             return b;
3816         }
3817         if ((t /= d) == 1) {
3818             return b + c;
3819         }
3820         if (!p) {
3821             p = d * .3;
3822         }
3823
3824         if (!a || a < Math.abs(c)) {
3825             a = c;
3826             var s = p / 4;
3827         }
3828         else {
3829             var s = p / (2 * Math.PI) * Math.asin(c / a);
3830         }
3831
3832         return -(a * Math.pow(2, 10 * (t -= 1)) * Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3833     },
3834
3835
3836     elasticOut: function (t, b, c, d, a, p) {
3837         if (t == 0) {
3838             return b;
3839         }
3840         if ((t /= d) == 1) {
3841             return b + c;
3842         }
3843         if (!p) {
3844             p = d * .3;
3845         }
3846
3847         if (!a || a < Math.abs(c)) {
3848             a = c;
3849             var s = p / 4;
3850         }
3851         else {
3852             var s = p / (2 * Math.PI) * Math.asin(c / a);
3853         }
3854
3855         return a * Math.pow(2, -10 * t) * Math.sin((t * d - s) * (2 * Math.PI) / p) + c + b;
3856     },
3857
3858
3859     elasticBoth: function (t, b, c, d, a, p) {
3860         if (t == 0) {
3861             return b;
3862         }
3863
3864         if ((t /= d / 2) == 2) {
3865             return b + c;
3866         }
3867
3868         if (!p) {
3869             p = d * (.3 * 1.5);
3870         }
3871
3872         if (!a || a < Math.abs(c)) {
3873             a = c;
3874             var s = p / 4;
3875         }
3876         else {
3877             var s = p / (2 * Math.PI) * Math.asin(c / a);
3878         }
3879
3880         if (t < 1) {
3881             return -.5 * (a * Math.pow(2, 10 * (t -= 1)) *
3882                           Math.sin((t * d - s) * (2 * Math.PI) / p)) + b;
3883         }
3884         return a * Math.pow(2, -10 * (t -= 1)) *
3885                Math.sin((t * d - s) * (2 * Math.PI) / p) * .5 + c + b;
3886     },
3887
3888
3889
3890     backIn: function (t, b, c, d, s) {
3891         if (typeof s == 'undefined') {
3892             s = 1.70158;
3893         }
3894         return c * (t /= d) * t * ((s + 1) * t - s) + b;
3895     },
3896
3897
3898     backOut: function (t, b, c, d, s) {
3899         if (typeof s == 'undefined') {
3900             s = 1.70158;
3901         }
3902         return c * ((t = t / d - 1) * t * ((s + 1) * t + s) + 1) + b;
3903     },
3904
3905
3906     backBoth: function (t, b, c, d, s) {
3907         if (typeof s == 'undefined') {
3908             s = 1.70158;
3909         }
3910
3911         if ((t /= d / 2 ) < 1) {
3912             return c / 2 * (t * t * (((s *= (1.525)) + 1) * t - s)) + b;
3913         }
3914         return c / 2 * ((t -= 2) * t * (((s *= (1.525)) + 1) * t + s) + 2) + b;
3915     },
3916
3917
3918     bounceIn: function (t, b, c, d) {
3919         return c - Roo.lib.Easing.bounceOut(d - t, 0, c, d) + b;
3920     },
3921
3922
3923     bounceOut: function (t, b, c, d) {
3924         if ((t /= d) < (1 / 2.75)) {
3925             return c * (7.5625 * t * t) + b;
3926         } else if (t < (2 / 2.75)) {
3927             return c * (7.5625 * (t -= (1.5 / 2.75)) * t + .75) + b;
3928         } else if (t < (2.5 / 2.75)) {
3929             return c * (7.5625 * (t -= (2.25 / 2.75)) * t + .9375) + b;
3930         }
3931         return c * (7.5625 * (t -= (2.625 / 2.75)) * t + .984375) + b;
3932     },
3933
3934
3935     bounceBoth: function (t, b, c, d) {
3936         if (t < d / 2) {
3937             return Roo.lib.Easing.bounceIn(t * 2, 0, c, d) * .5 + b;
3938         }
3939         return Roo.lib.Easing.bounceOut(t * 2 - d, 0, c, d) * .5 + c * .5 + b;
3940     }
3941 };/*
3942  * Portions of this file are based on pieces of Yahoo User Interface Library
3943  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
3944  * YUI licensed under the BSD License:
3945  * http://developer.yahoo.net/yui/license.txt
3946  * <script type="text/javascript">
3947  *
3948  */
3949     (function() {
3950         Roo.lib.Motion = function(el, attributes, duration, method) {
3951             if (el) {
3952                 Roo.lib.Motion.superclass.constructor.call(this, el, attributes, duration, method);
3953             }
3954         };
3955
3956         Roo.extend(Roo.lib.Motion, Roo.lib.ColorAnim);
3957
3958
3959         var Y = Roo.lib;
3960         var superclass = Y.Motion.superclass;
3961         var proto = Y.Motion.prototype;
3962
3963         proto.toString = function() {
3964             var el = this.getEl();
3965             var id = el.id || el.tagName;
3966             return ("Motion " + id);
3967         };
3968
3969         proto.patterns.points = /^points$/i;
3970
3971         proto.setAttribute = function(attr, val, unit) {
3972             if (this.patterns.points.test(attr)) {
3973                 unit = unit || 'px';
3974                 superclass.setAttribute.call(this, 'left', val[0], unit);
3975                 superclass.setAttribute.call(this, 'top', val[1], unit);
3976             } else {
3977                 superclass.setAttribute.call(this, attr, val, unit);
3978             }
3979         };
3980
3981         proto.getAttribute = function(attr) {
3982             if (this.patterns.points.test(attr)) {
3983                 var val = [
3984                         superclass.getAttribute.call(this, 'left'),
3985                         superclass.getAttribute.call(this, 'top')
3986                         ];
3987             } else {
3988                 val = superclass.getAttribute.call(this, attr);
3989             }
3990
3991             return val;
3992         };
3993
3994         proto.doMethod = function(attr, start, end) {
3995             var val = null;
3996
3997             if (this.patterns.points.test(attr)) {
3998                 var t = this.method(this.currentFrame, 0, 100, this.totalFrames) / 100;
3999                 val = Y.Bezier.getPosition(this.runtimeAttributes[attr], t);
4000             } else {
4001                 val = superclass.doMethod.call(this, attr, start, end);
4002             }
4003             return val;
4004         };
4005
4006         proto.setRuntimeAttribute = function(attr) {
4007             if (this.patterns.points.test(attr)) {
4008                 var el = this.getEl();
4009                 var attributes = this.attributes;
4010                 var start;
4011                 var control = attributes['points']['control'] || [];
4012                 var end;
4013                 var i, len;
4014
4015                 if (control.length > 0 && !(control[0] instanceof Array)) {
4016                     control = [control];
4017                 } else {
4018                     var tmp = [];
4019                     for (i = 0,len = control.length; i < len; ++i) {
4020                         tmp[i] = control[i];
4021                     }
4022                     control = tmp;
4023                 }
4024
4025                 Roo.fly(el).position();
4026
4027                 if (isset(attributes['points']['from'])) {
4028                     Roo.lib.Dom.setXY(el, attributes['points']['from']);
4029                 }
4030                 else {
4031                     Roo.lib.Dom.setXY(el, Roo.lib.Dom.getXY(el));
4032                 }
4033
4034                 start = this.getAttribute('points');
4035
4036
4037                 if (isset(attributes['points']['to'])) {
4038                     end = translateValues.call(this, attributes['points']['to'], start);
4039
4040                     var pageXY = Roo.lib.Dom.getXY(this.getEl());
4041                     for (i = 0,len = control.length; i < len; ++i) {
4042                         control[i] = translateValues.call(this, control[i], start);
4043                     }
4044
4045
4046                 } else if (isset(attributes['points']['by'])) {
4047                     end = [ start[0] + attributes['points']['by'][0], start[1] + attributes['points']['by'][1] ];
4048
4049                     for (i = 0,len = control.length; i < len; ++i) {
4050                         control[i] = [ start[0] + control[i][0], start[1] + control[i][1] ];
4051                     }
4052                 }
4053
4054                 this.runtimeAttributes[attr] = [start];
4055
4056                 if (control.length > 0) {
4057                     this.runtimeAttributes[attr] = this.runtimeAttributes[attr].concat(control);
4058                 }
4059
4060                 this.runtimeAttributes[attr][this.runtimeAttributes[attr].length] = end;
4061             }
4062             else {
4063                 superclass.setRuntimeAttribute.call(this, attr);
4064             }
4065         };
4066
4067         var translateValues = function(val, start) {
4068             var pageXY = Roo.lib.Dom.getXY(this.getEl());
4069             val = [ val[0] - pageXY[0] + start[0], val[1] - pageXY[1] + start[1] ];
4070
4071             return val;
4072         };
4073
4074         var isset = function(prop) {
4075             return (typeof prop !== 'undefined');
4076         };
4077     })();
4078 /*
4079  * Portions of this file are based on pieces of Yahoo User Interface Library
4080  * Copyright (c) 2007, Yahoo! Inc. All rights reserved.
4081  * YUI licensed under the BSD License:
4082  * http://developer.yahoo.net/yui/license.txt
4083  * <script type="text/javascript">
4084  *
4085  */
4086     (function() {
4087         Roo.lib.Scroll = function(el, attributes, duration, method) {
4088             if (el) {
4089                 Roo.lib.Scroll.superclass.constructor.call(this, el, attributes, duration, method);
4090             }
4091         };
4092
4093         Roo.extend(Roo.lib.Scroll, Roo.lib.ColorAnim);
4094
4095
4096         var Y = Roo.lib;
4097         var superclass = Y.Scroll.superclass;
4098         var proto = Y.Scroll.prototype;
4099
4100         proto.toString = function() {
4101             var el = this.getEl();
4102             var id = el.id || el.tagName;
4103             return ("Scroll " + id);
4104         };
4105
4106         proto.doMethod = function(attr, start, end) {
4107             var val = null;
4108
4109             if (attr == 'scroll') {
4110                 val = [
4111                         this.method(this.currentFrame, start[0], end[0] - start[0], this.totalFrames),
4112                         this.method(this.currentFrame, start[1], end[1] - start[1], this.totalFrames)
4113                         ];
4114
4115             } else {
4116                 val = superclass.doMethod.call(this, attr, start, end);
4117             }
4118             return val;
4119         };
4120
4121         proto.getAttribute = function(attr) {
4122             var val = null;
4123             var el = this.getEl();
4124
4125             if (attr == 'scroll') {
4126                 val = [ el.scrollLeft, el.scrollTop ];
4127             } else {
4128                 val = superclass.getAttribute.call(this, attr);
4129             }
4130
4131             return val;
4132         };
4133
4134         proto.setAttribute = function(attr, val, unit) {
4135             var el = this.getEl();
4136
4137             if (attr == 'scroll') {
4138                 el.scrollLeft = val[0];
4139                 el.scrollTop = val[1];
4140             } else {
4141                 superclass.setAttribute.call(this, attr, val, unit);
4142             }
4143         };
4144     })();
4145 /*
4146  * Based on:
4147  * Ext JS Library 1.1.1
4148  * Copyright(c) 2006-2007, Ext JS, LLC.
4149  *
4150  * Originally Released Under LGPL - original licence link has changed is not relivant.
4151  *
4152  * Fork - LGPL
4153  * <script type="text/javascript">
4154  */
4155
4156
4157 // nasty IE9 hack - what a pile of crap that is..
4158
4159  if (typeof Range != "undefined" && typeof Range.prototype.createContextualFragment == "undefined") {
4160     Range.prototype.createContextualFragment = function (html) {
4161         var doc = window.document;
4162         var container = doc.createElement("div");
4163         container.innerHTML = html;
4164         var frag = doc.createDocumentFragment(), n;
4165         while ((n = container.firstChild)) {
4166             frag.appendChild(n);
4167         }
4168         return frag;
4169     };
4170 }
4171
4172 /**
4173  * @class Roo.DomHelper
4174  * Utility class for working with DOM and/or Templates. It transparently supports using HTML fragments or DOM.
4175  * For more information see <a href="http://web.archive.org/web/20071221063734/http://www.jackslocum.com/blog/2006/10/06/domhelper-create-elements-using-dom-html-fragments-or-templates/">this blog post with examples</a>.
4176  * @singleton
4177  */
4178 Roo.DomHelper = function(){
4179     var tempTableEl = null;
4180     var emptyTags = /^(?:br|frame|hr|img|input|link|meta|range|spacer|wbr|area|param|col)$/i;
4181     var tableRe = /^table|tbody|tr|td$/i;
4182     var xmlns = {};
4183     // build as innerHTML where available
4184     /** @ignore */
4185     var createHtml = function(o){
4186         if(typeof o == 'string'){
4187             return o;
4188         }
4189         var b = "";
4190         if(!o.tag){
4191             o.tag = "div";
4192         }
4193         b += "<" + o.tag;
4194         for(var attr in o){
4195             if(attr == "tag" || attr == "children" || attr == "cn" || attr == "html" || typeof o[attr] == "function") { continue; }
4196             if(attr == "style"){
4197                 var s = o["style"];
4198                 if(typeof s == "function"){
4199                     s = s.call();
4200                 }
4201                 if(typeof s == "string"){
4202                     b += ' style="' + s + '"';
4203                 }else if(typeof s == "object"){
4204                     b += ' style="';
4205                     for(var key in s){
4206                         if(typeof s[key] != "function"){
4207                             b += key + ":" + s[key] + ";";
4208                         }
4209                     }
4210                     b += '"';
4211                 }
4212             }else{
4213                 if(attr == "cls"){
4214                     b += ' class="' + o["cls"] + '"';
4215                 }else if(attr == "htmlFor"){
4216                     b += ' for="' + o["htmlFor"] + '"';
4217                 }else{
4218                     b += " " + attr + '="' + o[attr] + '"';
4219                 }
4220             }
4221         }
4222         if(emptyTags.test(o.tag)){
4223             b += "/>";
4224         }else{
4225             b += ">";
4226             var cn = o.children || o.cn;
4227             if(cn){
4228                 //http://bugs.kde.org/show_bug.cgi?id=71506
4229                 if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4230                     for(var i = 0, len = cn.length; i < len; i++) {
4231                         b += createHtml(cn[i], b);
4232                     }
4233                 }else{
4234                     b += createHtml(cn, b);
4235                 }
4236             }
4237             if(o.html){
4238                 b += o.html;
4239             }
4240             b += "</" + o.tag + ">";
4241         }
4242         return b;
4243     };
4244
4245     // build as dom
4246     /** @ignore */
4247     var createDom = function(o, parentNode){
4248          
4249         // defininition craeted..
4250         var ns = false;
4251         if (o.ns && o.ns != 'html') {
4252                
4253             if (o.xmlns && typeof(xmlns[o.ns]) == 'undefined') {
4254                 xmlns[o.ns] = o.xmlns;
4255                 ns = o.xmlns;
4256             }
4257             if (typeof(xmlns[o.ns]) == 'undefined') {
4258                 console.log("Trying to create namespace element " + o.ns + ", however no xmlns was sent to builder previously");
4259             }
4260             ns = xmlns[o.ns];
4261         }
4262         
4263         
4264         if (typeof(o) == 'string') {
4265             return parentNode.appendChild(document.createTextNode(o));
4266         }
4267         o.tag = o.tag || div;
4268         if (o.ns && Roo.isIE) {
4269             ns = false;
4270             o.tag = o.ns + ':' + o.tag;
4271             
4272         }
4273         var el = ns ? document.createElementNS( ns, o.tag||'div') :  document.createElement(o.tag||'div');
4274         var useSet = el.setAttribute ? true : false; // In IE some elements don't have setAttribute
4275         for(var attr in o){
4276             
4277             if(attr == "tag" || attr == "ns" ||attr == "xmlns" ||attr == "children" || attr == "cn" || attr == "html" || 
4278                     attr == "style" || typeof o[attr] == "function") { continue; }
4279                     
4280             if(attr=="cls" && Roo.isIE){
4281                 el.className = o["cls"];
4282             }else{
4283                 if(useSet) { el.setAttribute(attr=="cls" ? 'class' : attr, o[attr]);}
4284                 else { 
4285                     el[attr] = o[attr];
4286                 }
4287             }
4288         }
4289         Roo.DomHelper.applyStyles(el, o.style);
4290         var cn = o.children || o.cn;
4291         if(cn){
4292             //http://bugs.kde.org/show_bug.cgi?id=71506
4293              if((cn instanceof Array) || (Roo.isSafari && typeof(cn.join) == "function")){
4294                 for(var i = 0, len = cn.length; i < len; i++) {
4295                     createDom(cn[i], el);
4296                 }
4297             }else{
4298                 createDom(cn, el);
4299             }
4300         }
4301         if(o.html){
4302             el.innerHTML = o.html;
4303         }
4304         if(parentNode){
4305            parentNode.appendChild(el);
4306         }
4307         return el;
4308     };
4309
4310     var ieTable = function(depth, s, h, e){
4311         tempTableEl.innerHTML = [s, h, e].join('');
4312         var i = -1, el = tempTableEl;
4313         while(++i < depth){
4314             el = el.firstChild;
4315         }
4316         return el;
4317     };
4318
4319     // kill repeat to save bytes
4320     var ts = '<table>',
4321         te = '</table>',
4322         tbs = ts+'<tbody>',
4323         tbe = '</tbody>'+te,
4324         trs = tbs + '<tr>',
4325         tre = '</tr>'+tbe;
4326
4327     /**
4328      * @ignore
4329      * Nasty code for IE's broken table implementation
4330      */
4331     var insertIntoTable = function(tag, where, el, html){
4332         if(!tempTableEl){
4333             tempTableEl = document.createElement('div');
4334         }
4335         var node;
4336         var before = null;
4337         if(tag == 'td'){
4338             if(where == 'afterbegin' || where == 'beforeend'){ // INTO a TD
4339                 return;
4340             }
4341             if(where == 'beforebegin'){
4342                 before = el;
4343                 el = el.parentNode;
4344             } else{
4345                 before = el.nextSibling;
4346                 el = el.parentNode;
4347             }
4348             node = ieTable(4, trs, html, tre);
4349         }
4350         else if(tag == 'tr'){
4351             if(where == 'beforebegin'){
4352                 before = el;
4353                 el = el.parentNode;
4354                 node = ieTable(3, tbs, html, tbe);
4355             } else if(where == 'afterend'){
4356                 before = el.nextSibling;
4357                 el = el.parentNode;
4358                 node = ieTable(3, tbs, html, tbe);
4359             } else{ // INTO a TR
4360                 if(where == 'afterbegin'){
4361                     before = el.firstChild;
4362                 }
4363                 node = ieTable(4, trs, html, tre);
4364             }
4365         } else if(tag == 'tbody'){
4366             if(where == 'beforebegin'){
4367                 before = el;
4368                 el = el.parentNode;
4369                 node = ieTable(2, ts, html, te);
4370             } else if(where == 'afterend'){
4371                 before = el.nextSibling;
4372                 el = el.parentNode;
4373                 node = ieTable(2, ts, html, te);
4374             } else{
4375                 if(where == 'afterbegin'){
4376                     before = el.firstChild;
4377                 }
4378                 node = ieTable(3, tbs, html, tbe);
4379             }
4380         } else{ // TABLE
4381             if(where == 'beforebegin' || where == 'afterend'){ // OUTSIDE the table
4382                 return;
4383             }
4384             if(where == 'afterbegin'){
4385                 before = el.firstChild;
4386             }
4387             node = ieTable(2, ts, html, te);
4388         }
4389         el.insertBefore(node, before);
4390         return node;
4391     };
4392
4393     return {
4394     /** True to force the use of DOM instead of html fragments @type Boolean */
4395     useDom : false,
4396
4397     /**
4398      * Returns the markup for the passed Element(s) config
4399      * @param {Object} o The Dom object spec (and children)
4400      * @return {String}
4401      */
4402     markup : function(o){
4403         return createHtml(o);
4404     },
4405
4406     /**
4407      * Applies a style specification to an element
4408      * @param {String/HTMLElement} el The element to apply styles to
4409      * @param {String/Object/Function} styles A style specification string eg "width:100px", or object in the form {width:"100px"}, or
4410      * a function which returns such a specification.
4411      */
4412     applyStyles : function(el, styles){
4413         if(styles){
4414            el = Roo.fly(el);
4415            if(typeof styles == "string"){
4416                var re = /\s?([a-z\-]*)\:\s?([^;]*);?/gi;
4417                var matches;
4418                while ((matches = re.exec(styles)) != null){
4419                    el.setStyle(matches[1], matches[2]);
4420                }
4421            }else if (typeof styles == "object"){
4422                for (var style in styles){
4423                   el.setStyle(style, styles[style]);
4424                }
4425            }else if (typeof styles == "function"){
4426                 Roo.DomHelper.applyStyles(el, styles.call());
4427            }
4428         }
4429     },
4430
4431     /**
4432      * Inserts an HTML fragment into the Dom
4433      * @param {String} where Where to insert the html in relation to el - beforeBegin, afterBegin, beforeEnd, afterEnd.
4434      * @param {HTMLElement} el The context element
4435      * @param {String} html The HTML fragmenet
4436      * @return {HTMLElement} The new node
4437      */
4438     insertHtml : function(where, el, html){
4439         where = where.toLowerCase();
4440         if(el.insertAdjacentHTML){
4441             if(tableRe.test(el.tagName)){
4442                 var rs;
4443                 if(rs = insertIntoTable(el.tagName.toLowerCase(), where, el, html)){
4444                     return rs;
4445                 }
4446             }
4447             switch(where){
4448                 case "beforebegin":
4449                     el.insertAdjacentHTML('BeforeBegin', html);
4450                     return el.previousSibling;
4451                 case "afterbegin":
4452                     el.insertAdjacentHTML('AfterBegin', html);
4453                     return el.firstChild;
4454                 case "beforeend":
4455                     el.insertAdjacentHTML('BeforeEnd', html);
4456                     return el.lastChild;
4457                 case "afterend":
4458                     el.insertAdjacentHTML('AfterEnd', html);
4459                     return el.nextSibling;
4460             }
4461             throw 'Illegal insertion point -> "' + where + '"';
4462         }
4463         var range = el.ownerDocument.createRange();
4464         var frag;
4465         switch(where){
4466              case "beforebegin":
4467                 range.setStartBefore(el);
4468                 frag = range.createContextualFragment(html);
4469                 el.parentNode.insertBefore(frag, el);
4470                 return el.previousSibling;
4471              case "afterbegin":
4472                 if(el.firstChild){
4473                     range.setStartBefore(el.firstChild);
4474                     frag = range.createContextualFragment(html);
4475                     el.insertBefore(frag, el.firstChild);
4476                     return el.firstChild;
4477                 }else{
4478                     el.innerHTML = html;
4479                     return el.firstChild;
4480                 }
4481             case "beforeend":
4482                 if(el.lastChild){
4483                     range.setStartAfter(el.lastChild);
4484                     frag = range.createContextualFragment(html);
4485                     el.appendChild(frag);
4486                     return el.lastChild;
4487                 }else{
4488                     el.innerHTML = html;
4489                     return el.lastChild;
4490                 }
4491             case "afterend":
4492                 range.setStartAfter(el);
4493                 frag = range.createContextualFragment(html);
4494                 el.parentNode.insertBefore(frag, el.nextSibling);
4495                 return el.nextSibling;
4496             }
4497             throw 'Illegal insertion point -> "' + where + '"';
4498     },
4499
4500     /**
4501      * Creates new Dom element(s) and inserts them before el
4502      * @param {String/HTMLElement/Element} el The context element
4503      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4504      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4505      * @return {HTMLElement/Roo.Element} The new node
4506      */
4507     insertBefore : function(el, o, returnElement){
4508         return this.doInsert(el, o, returnElement, "beforeBegin");
4509     },
4510
4511     /**
4512      * Creates new Dom element(s) and inserts them after el
4513      * @param {String/HTMLElement/Element} el The context element
4514      * @param {Object} o The Dom object spec (and children)
4515      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4516      * @return {HTMLElement/Roo.Element} The new node
4517      */
4518     insertAfter : function(el, o, returnElement){
4519         return this.doInsert(el, o, returnElement, "afterEnd", "nextSibling");
4520     },
4521
4522     /**
4523      * Creates new Dom element(s) and inserts them as the first child of el
4524      * @param {String/HTMLElement/Element} el The context element
4525      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4526      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4527      * @return {HTMLElement/Roo.Element} The new node
4528      */
4529     insertFirst : function(el, o, returnElement){
4530         return this.doInsert(el, o, returnElement, "afterBegin");
4531     },
4532
4533     // private
4534     doInsert : function(el, o, returnElement, pos, sibling){
4535         el = Roo.getDom(el);
4536         var newNode;
4537         if(this.useDom || o.ns){
4538             newNode = createDom(o, null);
4539             el.parentNode.insertBefore(newNode, sibling ? el[sibling] : el);
4540         }else{
4541             var html = createHtml(o);
4542             newNode = this.insertHtml(pos, el, html);
4543         }
4544         return returnElement ? Roo.get(newNode, true) : newNode;
4545     },
4546
4547     /**
4548      * Creates new Dom element(s) and appends them to el
4549      * @param {String/HTMLElement/Element} el The context element
4550      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4551      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4552      * @return {HTMLElement/Roo.Element} The new node
4553      */
4554     append : function(el, o, returnElement){
4555         el = Roo.getDom(el);
4556         var newNode;
4557         if(this.useDom || o.ns){
4558             newNode = createDom(o, null);
4559             el.appendChild(newNode);
4560         }else{
4561             var html = createHtml(o);
4562             newNode = this.insertHtml("beforeEnd", el, html);
4563         }
4564         return returnElement ? Roo.get(newNode, true) : newNode;
4565     },
4566
4567     /**
4568      * Creates new Dom element(s) and overwrites the contents of el with them
4569      * @param {String/HTMLElement/Element} el The context element
4570      * @param {Object/String} o The Dom object spec (and children) or raw HTML blob
4571      * @param {Boolean} returnElement (optional) true to return a Roo.Element
4572      * @return {HTMLElement/Roo.Element} The new node
4573      */
4574     overwrite : function(el, o, returnElement){
4575         el = Roo.getDom(el);
4576         if (o.ns) {
4577           
4578             while (el.childNodes.length) {
4579                 el.removeChild(el.firstChild);
4580             }
4581             createDom(o, el);
4582         } else {
4583             el.innerHTML = createHtml(o);   
4584         }
4585         
4586         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4587     },
4588
4589     /**
4590      * Creates a new Roo.DomHelper.Template from the Dom object spec
4591      * @param {Object} o The Dom object spec (and children)
4592      * @return {Roo.DomHelper.Template} The new template
4593      */
4594     createTemplate : function(o){
4595         var html = createHtml(o);
4596         return new Roo.Template(html);
4597     }
4598     };
4599 }();
4600 /*
4601  * Based on:
4602  * Ext JS Library 1.1.1
4603  * Copyright(c) 2006-2007, Ext JS, LLC.
4604  *
4605  * Originally Released Under LGPL - original licence link has changed is not relivant.
4606  *
4607  * Fork - LGPL
4608  * <script type="text/javascript">
4609  */
4610  
4611 /**
4612 * @class Roo.Template
4613 * Represents an HTML fragment template. Templates can be precompiled for greater performance.
4614 * For a list of available format functions, see {@link Roo.util.Format}.<br />
4615 * Usage:
4616 <pre><code>
4617 var t = new Roo.Template({
4618     html :  '&lt;div name="{id}"&gt;' + 
4619         '&lt;span class="{cls}"&gt;{name:trim} {someval:this.myformat}{value:ellipsis(10)}&lt;/span&gt;' +
4620         '&lt;/div&gt;',
4621     myformat: function (value, allValues) {
4622         return 'XX' + value;
4623     }
4624 });
4625 t.append('some-element', {id: 'myid', cls: 'myclass', name: 'foo', value: 'bar'});
4626 </code></pre>
4627 * For more information see this blog post with examples:
4628 *  <a href="http://www.cnitblog.com/seeyeah/archive/2011/12/30/38728.html/">DomHelper
4629      - Create Elements using DOM, HTML fragments and Templates</a>. 
4630 * @constructor
4631 * @param {Object} cfg - Configuration object.
4632 */
4633 Roo.Template = function(cfg){
4634     // BC!
4635     if(cfg instanceof Array){
4636         cfg = cfg.join("");
4637     }else if(arguments.length > 1){
4638         cfg = Array.prototype.join.call(arguments, "");
4639     }
4640     
4641     
4642     if (typeof(cfg) == 'object') {
4643         Roo.apply(this,cfg)
4644     } else {
4645         // bc
4646         this.html = cfg;
4647     }
4648     if (this.url) {
4649         this.load();
4650     }
4651     
4652 };
4653 Roo.Template.prototype = {
4654     
4655     /**
4656      * @cfg {String} url  The Url to load the template from. beware if you are loading from a url, the data may not be ready if you use it instantly..
4657      *                    it should be fixed so that template is observable...
4658      */
4659     url : false,
4660     /**
4661      * @cfg {String} html  The HTML fragment or an array of fragments to join("") or multiple arguments to join("")
4662      */
4663     html : '',
4664     /**
4665      * Returns an HTML fragment of this template with the specified values applied.
4666      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4667      * @return {String} The HTML fragment
4668      */
4669     applyTemplate : function(values){
4670         try {
4671            
4672             if(this.compiled){
4673                 return this.compiled(values);
4674             }
4675             var useF = this.disableFormats !== true;
4676             var fm = Roo.util.Format, tpl = this;
4677             var fn = function(m, name, format, args){
4678                 if(format && useF){
4679                     if(format.substr(0, 5) == "this."){
4680                         return tpl.call(format.substr(5), values[name], values);
4681                     }else{
4682                         if(args){
4683                             // quoted values are required for strings in compiled templates, 
4684                             // but for non compiled we need to strip them
4685                             // quoted reversed for jsmin
4686                             var re = /^\s*['"](.*)["']\s*$/;
4687                             args = args.split(',');
4688                             for(var i = 0, len = args.length; i < len; i++){
4689                                 args[i] = args[i].replace(re, "$1");
4690                             }
4691                             args = [values[name]].concat(args);
4692                         }else{
4693                             args = [values[name]];
4694                         }
4695                         return fm[format].apply(fm, args);
4696                     }
4697                 }else{
4698                     return values[name] !== undefined ? values[name] : "";
4699                 }
4700             };
4701             return this.html.replace(this.re, fn);
4702         } catch (e) {
4703             Roo.log(e);
4704             throw e;
4705         }
4706          
4707     },
4708     
4709     loading : false,
4710       
4711     load : function ()
4712     {
4713          
4714         if (this.loading) {
4715             return;
4716         }
4717         var _t = this;
4718         
4719         this.loading = true;
4720         this.compiled = false;
4721         
4722         var cx = new Roo.data.Connection();
4723         cx.request({
4724             url : this.url,
4725             method : 'GET',
4726             success : function (response) {
4727                 _t.loading = false;
4728                 _t.html = response.responseText;
4729                 _t.url = false;
4730                 _t.compile();
4731              },
4732             failure : function(response) {
4733                 Roo.log("Template failed to load from " + _t.url);
4734                 _t.loading = false;
4735             }
4736         });
4737     },
4738
4739     /**
4740      * Sets the HTML used as the template and optionally compiles it.
4741      * @param {String} html
4742      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4743      * @return {Roo.Template} this
4744      */
4745     set : function(html, compile){
4746         this.html = html;
4747         this.compiled = null;
4748         if(compile){
4749             this.compile();
4750         }
4751         return this;
4752     },
4753     
4754     /**
4755      * True to disable format functions (defaults to false)
4756      * @type Boolean
4757      */
4758     disableFormats : false,
4759     
4760     /**
4761     * The regular expression used to match template variables 
4762     * @type RegExp
4763     * @property 
4764     */
4765     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4766     
4767     /**
4768      * Compiles the template into an internal function, eliminating the RegEx overhead.
4769      * @return {Roo.Template} this
4770      */
4771     compile : function(){
4772         var fm = Roo.util.Format;
4773         var useF = this.disableFormats !== true;
4774         var sep = Roo.isGecko ? "+" : ",";
4775         var fn = function(m, name, format, args){
4776             if(format && useF){
4777                 args = args ? ',' + args : "";
4778                 if(format.substr(0, 5) != "this."){
4779                     format = "fm." + format + '(';
4780                 }else{
4781                     format = 'this.call("'+ format.substr(5) + '", ';
4782                     args = ", values";
4783                 }
4784             }else{
4785                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4786             }
4787             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4788         };
4789         var body;
4790         // branched to use + in gecko and [].join() in others
4791         if(Roo.isGecko){
4792             body = "this.compiled = function(values){ return '" +
4793                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4794                     "';};";
4795         }else{
4796             body = ["this.compiled = function(values){ return ['"];
4797             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4798             body.push("'].join('');};");
4799             body = body.join('');
4800         }
4801         /**
4802          * eval:var:values
4803          * eval:var:fm
4804          */
4805         eval(body);
4806         return this;
4807     },
4808     
4809     // private function used to call members
4810     call : function(fnName, value, allValues){
4811         return this[fnName](value, allValues);
4812     },
4813     
4814     /**
4815      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4816      * @param {String/HTMLElement/Roo.Element} el The context element
4817      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4818      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4819      * @return {HTMLElement/Roo.Element} The new node or Element
4820      */
4821     insertFirst: function(el, values, returnElement){
4822         return this.doInsert('afterBegin', el, values, returnElement);
4823     },
4824
4825     /**
4826      * Applies the supplied values to the template and inserts the new node(s) before el.
4827      * @param {String/HTMLElement/Roo.Element} el The context element
4828      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4829      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4830      * @return {HTMLElement/Roo.Element} The new node or Element
4831      */
4832     insertBefore: function(el, values, returnElement){
4833         return this.doInsert('beforeBegin', el, values, returnElement);
4834     },
4835
4836     /**
4837      * Applies the supplied values to the template and inserts the new node(s) after el.
4838      * @param {String/HTMLElement/Roo.Element} el The context element
4839      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4840      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4841      * @return {HTMLElement/Roo.Element} The new node or Element
4842      */
4843     insertAfter : function(el, values, returnElement){
4844         return this.doInsert('afterEnd', el, values, returnElement);
4845     },
4846     
4847     /**
4848      * Applies the supplied values to the template and appends the new node(s) to el.
4849      * @param {String/HTMLElement/Roo.Element} el The context element
4850      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4851      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4852      * @return {HTMLElement/Roo.Element} The new node or Element
4853      */
4854     append : function(el, values, returnElement){
4855         return this.doInsert('beforeEnd', el, values, returnElement);
4856     },
4857
4858     doInsert : function(where, el, values, returnEl){
4859         el = Roo.getDom(el);
4860         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4861         return returnEl ? Roo.get(newNode, true) : newNode;
4862     },
4863
4864     /**
4865      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4866      * @param {String/HTMLElement/Roo.Element} el The context element
4867      * @param {Object} values The template values. Can be an array if your params are numeric (i.e. {0}) or an object (i.e. {foo: 'bar'})
4868      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4869      * @return {HTMLElement/Roo.Element} The new node or Element
4870      */
4871     overwrite : function(el, values, returnElement){
4872         el = Roo.getDom(el);
4873         el.innerHTML = this.applyTemplate(values);
4874         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4875     }
4876 };
4877 /**
4878  * Alias for {@link #applyTemplate}
4879  * @method
4880  */
4881 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4882
4883 // backwards compat
4884 Roo.DomHelper.Template = Roo.Template;
4885
4886 /**
4887  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4888  * @param {String/HTMLElement} el A DOM element or its id
4889  * @returns {Roo.Template} The created template
4890  * @static
4891  */
4892 Roo.Template.from = function(el){
4893     el = Roo.getDom(el);
4894     return new Roo.Template(el.value || el.innerHTML);
4895 };/*
4896  * Based on:
4897  * Ext JS Library 1.1.1
4898  * Copyright(c) 2006-2007, Ext JS, LLC.
4899  *
4900  * Originally Released Under LGPL - original licence link has changed is not relivant.
4901  *
4902  * Fork - LGPL
4903  * <script type="text/javascript">
4904  */
4905  
4906
4907 /*
4908  * This is code is also distributed under MIT license for use
4909  * with jQuery and prototype JavaScript libraries.
4910  */
4911 /**
4912  * @class Roo.DomQuery
4913 Provides high performance selector/xpath processing by compiling queries into reusable functions. New pseudo classes and matchers can be plugged. It works on HTML and XML documents (if a content node is passed in).
4914 <p>
4915 DomQuery supports most of the <a href="http://www.w3.org/TR/2005/WD-css3-selectors-20051215/">CSS3 selectors spec</a>, along with some custom selectors and basic XPath.</p>
4916
4917 <p>
4918 All selectors, attribute filters and pseudos below can be combined infinitely in any order. For example "div.foo:nth-child(odd)[@foo=bar].bar:first" would be a perfectly valid selector. Node filters are processed in the order in which they appear, which allows you to optimize your queries for your document structure.
4919 </p>
4920 <h4>Element Selectors:</h4>
4921 <ul class="list">
4922     <li> <b>*</b> any element</li>
4923     <li> <b>E</b> an element with the tag E</li>
4924     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4925     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4926     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4927     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4928 </ul>
4929 <h4>Attribute Selectors:</h4>
4930 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4931 <ul class="list">
4932     <li> <b>E[foo]</b> has an attribute "foo"</li>
4933     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4934     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4935     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4936     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4937     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4938     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4939 </ul>
4940 <h4>Pseudo Classes:</h4>
4941 <ul class="list">
4942     <li> <b>E:first-child</b> E is the first child of its parent</li>
4943     <li> <b>E:last-child</b> E is the last child of its parent</li>
4944     <li> <b>E:nth-child(<i>n</i>)</b> E is the <i>n</i>th child of its parent (1 based as per the spec)</li>
4945     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4946     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4947     <li> <b>E:only-child</b> E is the only child of its parent</li>
4948     <li> <b>E:checked</b> E is an element that is has a checked attribute that is true (e.g. a radio or checkbox) </li>
4949     <li> <b>E:first</b> the first E in the resultset</li>
4950     <li> <b>E:last</b> the last E in the resultset</li>
4951     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4952     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4953     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4954     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4955     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4956     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4957     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4958     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4959     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4960 </ul>
4961 <h4>CSS Value Selectors:</h4>
4962 <ul class="list">
4963     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4964     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4965     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4966     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4967     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4968     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4969 </ul>
4970  * @singleton
4971  */
4972 Roo.DomQuery = function(){
4973     var cache = {}, simpleCache = {}, valueCache = {};
4974     var nonSpace = /\S/;
4975     var trimRe = /^\s+|\s+$/g;
4976     var tplRe = /\{(\d+)\}/g;
4977     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4978     var tagTokenRe = /^(#)?([\w-\*]+)/;
4979     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4980
4981     function child(p, index){
4982         var i = 0;
4983         var n = p.firstChild;
4984         while(n){
4985             if(n.nodeType == 1){
4986                if(++i == index){
4987                    return n;
4988                }
4989             }
4990             n = n.nextSibling;
4991         }
4992         return null;
4993     };
4994
4995     function next(n){
4996         while((n = n.nextSibling) && n.nodeType != 1);
4997         return n;
4998     };
4999
5000     function prev(n){
5001         while((n = n.previousSibling) && n.nodeType != 1);
5002         return n;
5003     };
5004
5005     function children(d){
5006         var n = d.firstChild, ni = -1;
5007             while(n){
5008                 var nx = n.nextSibling;
5009                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5010                     d.removeChild(n);
5011                 }else{
5012                     n.nodeIndex = ++ni;
5013                 }
5014                 n = nx;
5015             }
5016             return this;
5017         };
5018
5019     function byClassName(c, a, v){
5020         if(!v){
5021             return c;
5022         }
5023         var r = [], ri = -1, cn;
5024         for(var i = 0, ci; ci = c[i]; i++){
5025             if((' '+ci.className+' ').indexOf(v) != -1){
5026                 r[++ri] = ci;
5027             }
5028         }
5029         return r;
5030     };
5031
5032     function attrValue(n, attr){
5033         if(!n.tagName && typeof n.length != "undefined"){
5034             n = n[0];
5035         }
5036         if(!n){
5037             return null;
5038         }
5039         if(attr == "for"){
5040             return n.htmlFor;
5041         }
5042         if(attr == "class" || attr == "className"){
5043             return n.className;
5044         }
5045         return n.getAttribute(attr) || n[attr];
5046
5047     };
5048
5049     function getNodes(ns, mode, tagName){
5050         var result = [], ri = -1, cs;
5051         if(!ns){
5052             return result;
5053         }
5054         tagName = tagName || "*";
5055         if(typeof ns.getElementsByTagName != "undefined"){
5056             ns = [ns];
5057         }
5058         if(!mode){
5059             for(var i = 0, ni; ni = ns[i]; i++){
5060                 cs = ni.getElementsByTagName(tagName);
5061                 for(var j = 0, ci; ci = cs[j]; j++){
5062                     result[++ri] = ci;
5063                 }
5064             }
5065         }else if(mode == "/" || mode == ">"){
5066             var utag = tagName.toUpperCase();
5067             for(var i = 0, ni, cn; ni = ns[i]; i++){
5068                 cn = ni.children || ni.childNodes;
5069                 for(var j = 0, cj; cj = cn[j]; j++){
5070                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5071                         result[++ri] = cj;
5072                     }
5073                 }
5074             }
5075         }else if(mode == "+"){
5076             var utag = tagName.toUpperCase();
5077             for(var i = 0, n; n = ns[i]; i++){
5078                 while((n = n.nextSibling) && n.nodeType != 1);
5079                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5080                     result[++ri] = n;
5081                 }
5082             }
5083         }else if(mode == "~"){
5084             for(var i = 0, n; n = ns[i]; i++){
5085                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5086                 if(n){
5087                     result[++ri] = n;
5088                 }
5089             }
5090         }
5091         return result;
5092     };
5093
5094     function concat(a, b){
5095         if(b.slice){
5096             return a.concat(b);
5097         }
5098         for(var i = 0, l = b.length; i < l; i++){
5099             a[a.length] = b[i];
5100         }
5101         return a;
5102     }
5103
5104     function byTag(cs, tagName){
5105         if(cs.tagName || cs == document){
5106             cs = [cs];
5107         }
5108         if(!tagName){
5109             return cs;
5110         }
5111         var r = [], ri = -1;
5112         tagName = tagName.toLowerCase();
5113         for(var i = 0, ci; ci = cs[i]; i++){
5114             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5115                 r[++ri] = ci;
5116             }
5117         }
5118         return r;
5119     };
5120
5121     function byId(cs, attr, id){
5122         if(cs.tagName || cs == document){
5123             cs = [cs];
5124         }
5125         if(!id){
5126             return cs;
5127         }
5128         var r = [], ri = -1;
5129         for(var i = 0,ci; ci = cs[i]; i++){
5130             if(ci && ci.id == id){
5131                 r[++ri] = ci;
5132                 return r;
5133             }
5134         }
5135         return r;
5136     };
5137
5138     function byAttribute(cs, attr, value, op, custom){
5139         var r = [], ri = -1, st = custom=="{";
5140         var f = Roo.DomQuery.operators[op];
5141         for(var i = 0, ci; ci = cs[i]; i++){
5142             var a;
5143             if(st){
5144                 a = Roo.DomQuery.getStyle(ci, attr);
5145             }
5146             else if(attr == "class" || attr == "className"){
5147                 a = ci.className;
5148             }else if(attr == "for"){
5149                 a = ci.htmlFor;
5150             }else if(attr == "href"){
5151                 a = ci.getAttribute("href", 2);
5152             }else{
5153                 a = ci.getAttribute(attr);
5154             }
5155             if((f && f(a, value)) || (!f && a)){
5156                 r[++ri] = ci;
5157             }
5158         }
5159         return r;
5160     };
5161
5162     function byPseudo(cs, name, value){
5163         return Roo.DomQuery.pseudos[name](cs, value);
5164     };
5165
5166     // This is for IE MSXML which does not support expandos.
5167     // IE runs the same speed using setAttribute, however FF slows way down
5168     // and Safari completely fails so they need to continue to use expandos.
5169     var isIE = window.ActiveXObject ? true : false;
5170
5171     // this eval is stop the compressor from
5172     // renaming the variable to something shorter
5173     
5174     /** eval:var:batch */
5175     var batch = 30803; 
5176
5177     var key = 30803;
5178
5179     function nodupIEXml(cs){
5180         var d = ++key;
5181         cs[0].setAttribute("_nodup", d);
5182         var r = [cs[0]];
5183         for(var i = 1, len = cs.length; i < len; i++){
5184             var c = cs[i];
5185             if(!c.getAttribute("_nodup") != d){
5186                 c.setAttribute("_nodup", d);
5187                 r[r.length] = c;
5188             }
5189         }
5190         for(var i = 0, len = cs.length; i < len; i++){
5191             cs[i].removeAttribute("_nodup");
5192         }
5193         return r;
5194     }
5195
5196     function nodup(cs){
5197         if(!cs){
5198             return [];
5199         }
5200         var len = cs.length, c, i, r = cs, cj, ri = -1;
5201         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5202             return cs;
5203         }
5204         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5205             return nodupIEXml(cs);
5206         }
5207         var d = ++key;
5208         cs[0]._nodup = d;
5209         for(i = 1; c = cs[i]; i++){
5210             if(c._nodup != d){
5211                 c._nodup = d;
5212             }else{
5213                 r = [];
5214                 for(var j = 0; j < i; j++){
5215                     r[++ri] = cs[j];
5216                 }
5217                 for(j = i+1; cj = cs[j]; j++){
5218                     if(cj._nodup != d){
5219                         cj._nodup = d;
5220                         r[++ri] = cj;
5221                     }
5222                 }
5223                 return r;
5224             }
5225         }
5226         return r;
5227     }
5228
5229     function quickDiffIEXml(c1, c2){
5230         var d = ++key;
5231         for(var i = 0, len = c1.length; i < len; i++){
5232             c1[i].setAttribute("_qdiff", d);
5233         }
5234         var r = [];
5235         for(var i = 0, len = c2.length; i < len; i++){
5236             if(c2[i].getAttribute("_qdiff") != d){
5237                 r[r.length] = c2[i];
5238             }
5239         }
5240         for(var i = 0, len = c1.length; i < len; i++){
5241            c1[i].removeAttribute("_qdiff");
5242         }
5243         return r;
5244     }
5245
5246     function quickDiff(c1, c2){
5247         var len1 = c1.length;
5248         if(!len1){
5249             return c2;
5250         }
5251         if(isIE && c1[0].selectSingleNode){
5252             return quickDiffIEXml(c1, c2);
5253         }
5254         var d = ++key;
5255         for(var i = 0; i < len1; i++){
5256             c1[i]._qdiff = d;
5257         }
5258         var r = [];
5259         for(var i = 0, len = c2.length; i < len; i++){
5260             if(c2[i]._qdiff != d){
5261                 r[r.length] = c2[i];
5262             }
5263         }
5264         return r;
5265     }
5266
5267     function quickId(ns, mode, root, id){
5268         if(ns == root){
5269            var d = root.ownerDocument || root;
5270            return d.getElementById(id);
5271         }
5272         ns = getNodes(ns, mode, "*");
5273         return byId(ns, null, id);
5274     }
5275
5276     return {
5277         getStyle : function(el, name){
5278             return Roo.fly(el).getStyle(name);
5279         },
5280         /**
5281          * Compiles a selector/xpath query into a reusable function. The returned function
5282          * takes one parameter "root" (optional), which is the context node from where the query should start.
5283          * @param {String} selector The selector/xpath query
5284          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5285          * @return {Function}
5286          */
5287         compile : function(path, type){
5288             type = type || "select";
5289             
5290             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5291             var q = path, mode, lq;
5292             var tk = Roo.DomQuery.matchers;
5293             var tklen = tk.length;
5294             var mm;
5295
5296             // accept leading mode switch
5297             var lmode = q.match(modeRe);
5298             if(lmode && lmode[1]){
5299                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5300                 q = q.replace(lmode[1], "");
5301             }
5302             // strip leading slashes
5303             while(path.substr(0, 1)=="/"){
5304                 path = path.substr(1);
5305             }
5306
5307             while(q && lq != q){
5308                 lq = q;
5309                 var tm = q.match(tagTokenRe);
5310                 if(type == "select"){
5311                     if(tm){
5312                         if(tm[1] == "#"){
5313                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5314                         }else{
5315                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5316                         }
5317                         q = q.replace(tm[0], "");
5318                     }else if(q.substr(0, 1) != '@'){
5319                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5320                     }
5321                 }else{
5322                     if(tm){
5323                         if(tm[1] == "#"){
5324                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5325                         }else{
5326                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5327                         }
5328                         q = q.replace(tm[0], "");
5329                     }
5330                 }
5331                 while(!(mm = q.match(modeRe))){
5332                     var matched = false;
5333                     for(var j = 0; j < tklen; j++){
5334                         var t = tk[j];
5335                         var m = q.match(t.re);
5336                         if(m){
5337                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5338                                                     return m[i];
5339                                                 });
5340                             q = q.replace(m[0], "");
5341                             matched = true;
5342                             break;
5343                         }
5344                     }
5345                     // prevent infinite loop on bad selector
5346                     if(!matched){
5347                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5348                     }
5349                 }
5350                 if(mm[1]){
5351                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5352                     q = q.replace(mm[1], "");
5353                 }
5354             }
5355             fn[fn.length] = "return nodup(n);\n}";
5356             
5357              /** 
5358               * list of variables that need from compression as they are used by eval.
5359              *  eval:var:batch 
5360              *  eval:var:nodup
5361              *  eval:var:byTag
5362              *  eval:var:ById
5363              *  eval:var:getNodes
5364              *  eval:var:quickId
5365              *  eval:var:mode
5366              *  eval:var:root
5367              *  eval:var:n
5368              *  eval:var:byClassName
5369              *  eval:var:byPseudo
5370              *  eval:var:byAttribute
5371              *  eval:var:attrValue
5372              * 
5373              **/ 
5374             eval(fn.join(""));
5375             return f;
5376         },
5377
5378         /**
5379          * Selects a group of elements.
5380          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5381          * @param {Node} root (optional) The start of the query (defaults to document).
5382          * @return {Array}
5383          */
5384         select : function(path, root, type){
5385             if(!root || root == document){
5386                 root = document;
5387             }
5388             if(typeof root == "string"){
5389                 root = document.getElementById(root);
5390             }
5391             var paths = path.split(",");
5392             var results = [];
5393             for(var i = 0, len = paths.length; i < len; i++){
5394                 var p = paths[i].replace(trimRe, "");
5395                 if(!cache[p]){
5396                     cache[p] = Roo.DomQuery.compile(p);
5397                     if(!cache[p]){
5398                         throw p + " is not a valid selector";
5399                     }
5400                 }
5401                 var result = cache[p](root);
5402                 if(result && result != document){
5403                     results = results.concat(result);
5404                 }
5405             }
5406             if(paths.length > 1){
5407                 return nodup(results);
5408             }
5409             return results;
5410         },
5411
5412         /**
5413          * Selects a single element.
5414          * @param {String} selector The selector/xpath query
5415          * @param {Node} root (optional) The start of the query (defaults to document).
5416          * @return {Element}
5417          */
5418         selectNode : function(path, root){
5419             return Roo.DomQuery.select(path, root)[0];
5420         },
5421
5422         /**
5423          * Selects the value of a node, optionally replacing null with the defaultValue.
5424          * @param {String} selector The selector/xpath query
5425          * @param {Node} root (optional) The start of the query (defaults to document).
5426          * @param {String} defaultValue
5427          */
5428         selectValue : function(path, root, defaultValue){
5429             path = path.replace(trimRe, "");
5430             if(!valueCache[path]){
5431                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5432             }
5433             var n = valueCache[path](root);
5434             n = n[0] ? n[0] : n;
5435             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5436             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5437         },
5438
5439         /**
5440          * Selects the value of a node, parsing integers and floats.
5441          * @param {String} selector The selector/xpath query
5442          * @param {Node} root (optional) The start of the query (defaults to document).
5443          * @param {Number} defaultValue
5444          * @return {Number}
5445          */
5446         selectNumber : function(path, root, defaultValue){
5447             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5448             return parseFloat(v);
5449         },
5450
5451         /**
5452          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5453          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5454          * @param {String} selector The simple selector to test
5455          * @return {Boolean}
5456          */
5457         is : function(el, ss){
5458             if(typeof el == "string"){
5459                 el = document.getElementById(el);
5460             }
5461             var isArray = (el instanceof Array);
5462             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5463             return isArray ? (result.length == el.length) : (result.length > 0);
5464         },
5465
5466         /**
5467          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5468          * @param {Array} el An array of elements to filter
5469          * @param {String} selector The simple selector to test
5470          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5471          * the selector instead of the ones that match
5472          * @return {Array}
5473          */
5474         filter : function(els, ss, nonMatches){
5475             ss = ss.replace(trimRe, "");
5476             if(!simpleCache[ss]){
5477                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5478             }
5479             var result = simpleCache[ss](els);
5480             return nonMatches ? quickDiff(result, els) : result;
5481         },
5482
5483         /**
5484          * Collection of matching regular expressions and code snippets.
5485          */
5486         matchers : [{
5487                 re: /^\.([\w-]+)/,
5488                 select: 'n = byClassName(n, null, " {1} ");'
5489             }, {
5490                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5491                 select: 'n = byPseudo(n, "{1}", "{2}");'
5492             },{
5493                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5494                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5495             }, {
5496                 re: /^#([\w-]+)/,
5497                 select: 'n = byId(n, null, "{1}");'
5498             },{
5499                 re: /^@([\w-]+)/,
5500                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5501             }
5502         ],
5503
5504         /**
5505          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5506          * New operators can be added as long as the match the format <i>c</i>= where <i>c</i> is any character other than space, &gt; &lt;.
5507          */
5508         operators : {
5509             "=" : function(a, v){
5510                 return a == v;
5511             },
5512             "!=" : function(a, v){
5513                 return a != v;
5514             },
5515             "^=" : function(a, v){
5516                 return a && a.substr(0, v.length) == v;
5517             },
5518             "$=" : function(a, v){
5519                 return a && a.substr(a.length-v.length) == v;
5520             },
5521             "*=" : function(a, v){
5522                 return a && a.indexOf(v) !== -1;
5523             },
5524             "%=" : function(a, v){
5525                 return (a % v) == 0;
5526             },
5527             "|=" : function(a, v){
5528                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5529             },
5530             "~=" : function(a, v){
5531                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5532             }
5533         },
5534
5535         /**
5536          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5537          * and the argument (if any) supplied in the selector.
5538          */
5539         pseudos : {
5540             "first-child" : function(c){
5541                 var r = [], ri = -1, n;
5542                 for(var i = 0, ci; ci = n = c[i]; i++){
5543                     while((n = n.previousSibling) && n.nodeType != 1);
5544                     if(!n){
5545                         r[++ri] = ci;
5546                     }
5547                 }
5548                 return r;
5549             },
5550
5551             "last-child" : function(c){
5552                 var r = [], ri = -1, n;
5553                 for(var i = 0, ci; ci = n = c[i]; i++){
5554                     while((n = n.nextSibling) && n.nodeType != 1);
5555                     if(!n){
5556                         r[++ri] = ci;
5557                     }
5558                 }
5559                 return r;
5560             },
5561
5562             "nth-child" : function(c, a) {
5563                 var r = [], ri = -1;
5564                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5565                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5566                 for(var i = 0, n; n = c[i]; i++){
5567                     var pn = n.parentNode;
5568                     if (batch != pn._batch) {
5569                         var j = 0;
5570                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5571                             if(cn.nodeType == 1){
5572                                cn.nodeIndex = ++j;
5573                             }
5574                         }
5575                         pn._batch = batch;
5576                     }
5577                     if (f == 1) {
5578                         if (l == 0 || n.nodeIndex == l){
5579                             r[++ri] = n;
5580                         }
5581                     } else if ((n.nodeIndex + l) % f == 0){
5582                         r[++ri] = n;
5583                     }
5584                 }
5585
5586                 return r;
5587             },
5588
5589             "only-child" : function(c){
5590                 var r = [], ri = -1;;
5591                 for(var i = 0, ci; ci = c[i]; i++){
5592                     if(!prev(ci) && !next(ci)){
5593                         r[++ri] = ci;
5594                     }
5595                 }
5596                 return r;
5597             },
5598
5599             "empty" : function(c){
5600                 var r = [], ri = -1;
5601                 for(var i = 0, ci; ci = c[i]; i++){
5602                     var cns = ci.childNodes, j = 0, cn, empty = true;
5603                     while(cn = cns[j]){
5604                         ++j;
5605                         if(cn.nodeType == 1 || cn.nodeType == 3){
5606                             empty = false;
5607                             break;
5608                         }
5609                     }
5610                     if(empty){
5611                         r[++ri] = ci;
5612                     }
5613                 }
5614                 return r;
5615             },
5616
5617             "contains" : function(c, v){
5618                 var r = [], ri = -1;
5619                 for(var i = 0, ci; ci = c[i]; i++){
5620                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5621                         r[++ri] = ci;
5622                     }
5623                 }
5624                 return r;
5625             },
5626
5627             "nodeValue" : function(c, v){
5628                 var r = [], ri = -1;
5629                 for(var i = 0, ci; ci = c[i]; i++){
5630                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5631                         r[++ri] = ci;
5632                     }
5633                 }
5634                 return r;
5635             },
5636
5637             "checked" : function(c){
5638                 var r = [], ri = -1;
5639                 for(var i = 0, ci; ci = c[i]; i++){
5640                     if(ci.checked == true){
5641                         r[++ri] = ci;
5642                     }
5643                 }
5644                 return r;
5645             },
5646
5647             "not" : function(c, ss){
5648                 return Roo.DomQuery.filter(c, ss, true);
5649             },
5650
5651             "odd" : function(c){
5652                 return this["nth-child"](c, "odd");
5653             },
5654
5655             "even" : function(c){
5656                 return this["nth-child"](c, "even");
5657             },
5658
5659             "nth" : function(c, a){
5660                 return c[a-1] || [];
5661             },
5662
5663             "first" : function(c){
5664                 return c[0] || [];
5665             },
5666
5667             "last" : function(c){
5668                 return c[c.length-1] || [];
5669             },
5670
5671             "has" : function(c, ss){
5672                 var s = Roo.DomQuery.select;
5673                 var r = [], ri = -1;
5674                 for(var i = 0, ci; ci = c[i]; i++){
5675                     if(s(ss, ci).length > 0){
5676                         r[++ri] = ci;
5677                     }
5678                 }
5679                 return r;
5680             },
5681
5682             "next" : function(c, ss){
5683                 var is = Roo.DomQuery.is;
5684                 var r = [], ri = -1;
5685                 for(var i = 0, ci; ci = c[i]; i++){
5686                     var n = next(ci);
5687                     if(n && is(n, ss)){
5688                         r[++ri] = ci;
5689                     }
5690                 }
5691                 return r;
5692             },
5693
5694             "prev" : function(c, ss){
5695                 var is = Roo.DomQuery.is;
5696                 var r = [], ri = -1;
5697                 for(var i = 0, ci; ci = c[i]; i++){
5698                     var n = prev(ci);
5699                     if(n && is(n, ss)){
5700                         r[++ri] = ci;
5701                     }
5702                 }
5703                 return r;
5704             }
5705         }
5706     };
5707 }();
5708
5709 /**
5710  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5711  * @param {String} path The selector/xpath query
5712  * @param {Node} root (optional) The start of the query (defaults to document).
5713  * @return {Array}
5714  * @member Roo
5715  * @method query
5716  */
5717 Roo.query = Roo.DomQuery.select;
5718 /*
5719  * Based on:
5720  * Ext JS Library 1.1.1
5721  * Copyright(c) 2006-2007, Ext JS, LLC.
5722  *
5723  * Originally Released Under LGPL - original licence link has changed is not relivant.
5724  *
5725  * Fork - LGPL
5726  * <script type="text/javascript">
5727  */
5728
5729 /**
5730  * @class Roo.util.Observable
5731  * Base class that provides a common interface for publishing events. Subclasses are expected to
5732  * to have a property "events" with all the events defined.<br>
5733  * For example:
5734  * <pre><code>
5735  Employee = function(name){
5736     this.name = name;
5737     this.addEvents({
5738         "fired" : true,
5739         "quit" : true
5740     });
5741  }
5742  Roo.extend(Employee, Roo.util.Observable);
5743 </code></pre>
5744  * @param {Object} config properties to use (incuding events / listeners)
5745  */
5746
5747 Roo.util.Observable = function(cfg){
5748     
5749     cfg = cfg|| {};
5750     this.addEvents(cfg.events || {});
5751     if (cfg.events) {
5752         delete cfg.events; // make sure
5753     }
5754      
5755     Roo.apply(this, cfg);
5756     
5757     if(this.listeners){
5758         this.on(this.listeners);
5759         delete this.listeners;
5760     }
5761 };
5762 Roo.util.Observable.prototype = {
5763     /** 
5764  * @cfg {Object} listeners  list of events and functions to call for this object, 
5765  * For example :
5766  * <pre><code>
5767     listeners :  { 
5768        'click' : function(e) {
5769            ..... 
5770         } ,
5771         .... 
5772     } 
5773   </code></pre>
5774  */
5775     
5776     
5777     /**
5778      * Fires the specified event with the passed parameters (minus the event name).
5779      * @param {String} eventName
5780      * @param {Object...} args Variable number of parameters are passed to handlers
5781      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5782      */
5783     fireEvent : function(){
5784         var ce = this.events[arguments[0].toLowerCase()];
5785         if(typeof ce == "object"){
5786             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5787         }else{
5788             return true;
5789         }
5790     },
5791
5792     // private
5793     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5794
5795     /**
5796      * Appends an event handler to this component
5797      * @param {String}   eventName The type of event to listen for
5798      * @param {Function} handler The method the event invokes
5799      * @param {Object}   scope (optional) The scope in which to execute the handler
5800      * function. The handler function's "this" context.
5801      * @param {Object}   options (optional) An object containing handler configuration
5802      * properties. This may contain any of the following properties:<ul>
5803      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5804      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5805      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5806      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5807      * by the specified number of milliseconds. If the event fires again within that time, the original
5808      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5809      * </ul><br>
5810      * <p>
5811      * <b>Combining Options</b><br>
5812      * Using the options argument, it is possible to combine different types of listeners:<br>
5813      * <br>
5814      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5815                 <pre><code>
5816                 el.on('click', this.onClick, this, {
5817                         single: true,
5818                 delay: 100,
5819                 forumId: 4
5820                 });
5821                 </code></pre>
5822      * <p>
5823      * <b>Attaching multiple handlers in 1 call</b><br>
5824      * The method also allows for a single argument to be passed which is a config object containing properties
5825      * which specify multiple handlers.
5826      * <pre><code>
5827                 el.on({
5828                         'click': {
5829                         fn: this.onClick,
5830                         scope: this,
5831                         delay: 100
5832                 }, 
5833                 'mouseover': {
5834                         fn: this.onMouseOver,
5835                         scope: this
5836                 },
5837                 'mouseout': {
5838                         fn: this.onMouseOut,
5839                         scope: this
5840                 }
5841                 });
5842                 </code></pre>
5843      * <p>
5844      * Or a shorthand syntax which passes the same scope object to all handlers:
5845         <pre><code>
5846                 el.on({
5847                         'click': this.onClick,
5848                 'mouseover': this.onMouseOver,
5849                 'mouseout': this.onMouseOut,
5850                 scope: this
5851                 });
5852                 </code></pre>
5853      */
5854     addListener : function(eventName, fn, scope, o){
5855         if(typeof eventName == "object"){
5856             o = eventName;
5857             for(var e in o){
5858                 if(this.filterOptRe.test(e)){
5859                     continue;
5860                 }
5861                 if(typeof o[e] == "function"){
5862                     // shared options
5863                     this.addListener(e, o[e], o.scope,  o);
5864                 }else{
5865                     // individual options
5866                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5867                 }
5868             }
5869             return;
5870         }
5871         o = (!o || typeof o == "boolean") ? {} : o;
5872         eventName = eventName.toLowerCase();
5873         var ce = this.events[eventName] || true;
5874         if(typeof ce == "boolean"){
5875             ce = new Roo.util.Event(this, eventName);
5876             this.events[eventName] = ce;
5877         }
5878         ce.addListener(fn, scope, o);
5879     },
5880
5881     /**
5882      * Removes a listener
5883      * @param {String}   eventName     The type of event to listen for
5884      * @param {Function} handler        The handler to remove
5885      * @param {Object}   scope  (optional) The scope (this object) for the handler
5886      */
5887     removeListener : function(eventName, fn, scope){
5888         var ce = this.events[eventName.toLowerCase()];
5889         if(typeof ce == "object"){
5890             ce.removeListener(fn, scope);
5891         }
5892     },
5893
5894     /**
5895      * Removes all listeners for this object
5896      */
5897     purgeListeners : function(){
5898         for(var evt in this.events){
5899             if(typeof this.events[evt] == "object"){
5900                  this.events[evt].clearListeners();
5901             }
5902         }
5903     },
5904
5905     relayEvents : function(o, events){
5906         var createHandler = function(ename){
5907             return function(){
5908                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5909             };
5910         };
5911         for(var i = 0, len = events.length; i < len; i++){
5912             var ename = events[i];
5913             if(!this.events[ename]){ this.events[ename] = true; };
5914             o.on(ename, createHandler(ename), this);
5915         }
5916     },
5917
5918     /**
5919      * Used to define events on this Observable
5920      * @param {Object} object The object with the events defined
5921      */
5922     addEvents : function(o){
5923         if(!this.events){
5924             this.events = {};
5925         }
5926         Roo.applyIf(this.events, o);
5927     },
5928
5929     /**
5930      * Checks to see if this object has any listeners for a specified event
5931      * @param {String} eventName The name of the event to check for
5932      * @return {Boolean} True if the event is being listened for, else false
5933      */
5934     hasListener : function(eventName){
5935         var e = this.events[eventName];
5936         return typeof e == "object" && e.listeners.length > 0;
5937     }
5938 };
5939 /**
5940  * Appends an event handler to this element (shorthand for addListener)
5941  * @param {String}   eventName     The type of event to listen for
5942  * @param {Function} handler        The method the event invokes
5943  * @param {Object}   scope (optional) The scope in which to execute the handler
5944  * function. The handler function's "this" context.
5945  * @param {Object}   options  (optional)
5946  * @method
5947  */
5948 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5949 /**
5950  * Removes a listener (shorthand for removeListener)
5951  * @param {String}   eventName     The type of event to listen for
5952  * @param {Function} handler        The handler to remove
5953  * @param {Object}   scope  (optional) The scope (this object) for the handler
5954  * @method
5955  */
5956 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5957
5958 /**
5959  * Starts capture on the specified Observable. All events will be passed
5960  * to the supplied function with the event name + standard signature of the event
5961  * <b>before</b> the event is fired. If the supplied function returns false,
5962  * the event will not fire.
5963  * @param {Observable} o The Observable to capture
5964  * @param {Function} fn The function to call
5965  * @param {Object} scope (optional) The scope (this object) for the fn
5966  * @static
5967  */
5968 Roo.util.Observable.capture = function(o, fn, scope){
5969     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5970 };
5971
5972 /**
5973  * Removes <b>all</b> added captures from the Observable.
5974  * @param {Observable} o The Observable to release
5975  * @static
5976  */
5977 Roo.util.Observable.releaseCapture = function(o){
5978     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5979 };
5980
5981 (function(){
5982
5983     var createBuffered = function(h, o, scope){
5984         var task = new Roo.util.DelayedTask();
5985         return function(){
5986             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5987         };
5988     };
5989
5990     var createSingle = function(h, e, fn, scope){
5991         return function(){
5992             e.removeListener(fn, scope);
5993             return h.apply(scope, arguments);
5994         };
5995     };
5996
5997     var createDelayed = function(h, o, scope){
5998         return function(){
5999             var args = Array.prototype.slice.call(arguments, 0);
6000             setTimeout(function(){
6001                 h.apply(scope, args);
6002             }, o.delay || 10);
6003         };
6004     };
6005
6006     Roo.util.Event = function(obj, name){
6007         this.name = name;
6008         this.obj = obj;
6009         this.listeners = [];
6010     };
6011
6012     Roo.util.Event.prototype = {
6013         addListener : function(fn, scope, options){
6014             var o = options || {};
6015             scope = scope || this.obj;
6016             if(!this.isListening(fn, scope)){
6017                 var l = {fn: fn, scope: scope, options: o};
6018                 var h = fn;
6019                 if(o.delay){
6020                     h = createDelayed(h, o, scope);
6021                 }
6022                 if(o.single){
6023                     h = createSingle(h, this, fn, scope);
6024                 }
6025                 if(o.buffer){
6026                     h = createBuffered(h, o, scope);
6027                 }
6028                 l.fireFn = h;
6029                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6030                     this.listeners.push(l);
6031                 }else{
6032                     this.listeners = this.listeners.slice(0);
6033                     this.listeners.push(l);
6034                 }
6035             }
6036         },
6037
6038         findListener : function(fn, scope){
6039             scope = scope || this.obj;
6040             var ls = this.listeners;
6041             for(var i = 0, len = ls.length; i < len; i++){
6042                 var l = ls[i];
6043                 if(l.fn == fn && l.scope == scope){
6044                     return i;
6045                 }
6046             }
6047             return -1;
6048         },
6049
6050         isListening : function(fn, scope){
6051             return this.findListener(fn, scope) != -1;
6052         },
6053
6054         removeListener : function(fn, scope){
6055             var index;
6056             if((index = this.findListener(fn, scope)) != -1){
6057                 if(!this.firing){
6058                     this.listeners.splice(index, 1);
6059                 }else{
6060                     this.listeners = this.listeners.slice(0);
6061                     this.listeners.splice(index, 1);
6062                 }
6063                 return true;
6064             }
6065             return false;
6066         },
6067
6068         clearListeners : function(){
6069             this.listeners = [];
6070         },
6071
6072         fire : function(){
6073             var ls = this.listeners, scope, len = ls.length;
6074             if(len > 0){
6075                 this.firing = true;
6076                 var args = Array.prototype.slice.call(arguments, 0);
6077                 for(var i = 0; i < len; i++){
6078                     var l = ls[i];
6079                     if(l.fireFn.apply(l.scope||this.obj||window, arguments) === false){
6080                         this.firing = false;
6081                         return false;
6082                     }
6083                 }
6084                 this.firing = false;
6085             }
6086             return true;
6087         }
6088     };
6089 })();/*
6090  * RooJS Library 
6091  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6092  *
6093  * Licence LGPL 
6094  *
6095  */
6096  
6097 /**
6098  * @class Roo.Document
6099  * @extends Roo.util.Observable
6100  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6101  * 
6102  * @param {Object} config the methods and properties of the 'base' class for the application.
6103  * 
6104  *  Generic Page handler - implement this to start your app..
6105  * 
6106  * eg.
6107  *  MyProject = new Roo.Document({
6108         events : {
6109             'load' : true // your events..
6110         },
6111         listeners : {
6112             'ready' : function() {
6113                 // fired on Roo.onReady()
6114             }
6115         }
6116  * 
6117  */
6118 Roo.Document = function(cfg) {
6119      
6120     this.addEvents({ 
6121         'ready' : true
6122     });
6123     Roo.util.Observable.call(this,cfg);
6124     
6125     var _this = this;
6126     
6127     Roo.onReady(function() {
6128         _this.fireEvent('ready');
6129     },null,false);
6130     
6131     
6132 }
6133
6134 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6135  * Based on:
6136  * Ext JS Library 1.1.1
6137  * Copyright(c) 2006-2007, Ext JS, LLC.
6138  *
6139  * Originally Released Under LGPL - original licence link has changed is not relivant.
6140  *
6141  * Fork - LGPL
6142  * <script type="text/javascript">
6143  */
6144
6145 /**
6146  * @class Roo.EventManager
6147  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6148  * several useful events directly.
6149  * See {@link Roo.EventObject} for more details on normalized event objects.
6150  * @singleton
6151  */
6152 Roo.EventManager = function(){
6153     var docReadyEvent, docReadyProcId, docReadyState = false;
6154     var resizeEvent, resizeTask, textEvent, textSize;
6155     var E = Roo.lib.Event;
6156     var D = Roo.lib.Dom;
6157
6158     
6159     
6160
6161     var fireDocReady = function(){
6162         if(!docReadyState){
6163             docReadyState = true;
6164             Roo.isReady = true;
6165             if(docReadyProcId){
6166                 clearInterval(docReadyProcId);
6167             }
6168             if(Roo.isGecko || Roo.isOpera) {
6169                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6170             }
6171             if(Roo.isIE){
6172                 var defer = document.getElementById("ie-deferred-loader");
6173                 if(defer){
6174                     defer.onreadystatechange = null;
6175                     defer.parentNode.removeChild(defer);
6176                 }
6177             }
6178             if(docReadyEvent){
6179                 docReadyEvent.fire();
6180                 docReadyEvent.clearListeners();
6181             }
6182         }
6183     };
6184     
6185     var initDocReady = function(){
6186         docReadyEvent = new Roo.util.Event();
6187         if(Roo.isGecko || Roo.isOpera) {
6188             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6189         }else if(Roo.isIE){
6190             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6191             var defer = document.getElementById("ie-deferred-loader");
6192             defer.onreadystatechange = function(){
6193                 if(this.readyState == "complete"){
6194                     fireDocReady();
6195                 }
6196             };
6197         }else if(Roo.isSafari){ 
6198             docReadyProcId = setInterval(function(){
6199                 var rs = document.readyState;
6200                 if(rs == "complete") {
6201                     fireDocReady();     
6202                  }
6203             }, 10);
6204         }
6205         // no matter what, make sure it fires on load
6206         E.on(window, "load", fireDocReady);
6207     };
6208
6209     var createBuffered = function(h, o){
6210         var task = new Roo.util.DelayedTask(h);
6211         return function(e){
6212             // create new event object impl so new events don't wipe out properties
6213             e = new Roo.EventObjectImpl(e);
6214             task.delay(o.buffer, h, null, [e]);
6215         };
6216     };
6217
6218     var createSingle = function(h, el, ename, fn){
6219         return function(e){
6220             Roo.EventManager.removeListener(el, ename, fn);
6221             h(e);
6222         };
6223     };
6224
6225     var createDelayed = function(h, o){
6226         return function(e){
6227             // create new event object impl so new events don't wipe out properties
6228             e = new Roo.EventObjectImpl(e);
6229             setTimeout(function(){
6230                 h(e);
6231             }, o.delay || 10);
6232         };
6233     };
6234     var transitionEndVal = false;
6235     
6236     var transitionEnd = function()
6237     {
6238         if (transitionEndVal) {
6239             return transitionEndVal;
6240         }
6241         var el = document.createElement('div');
6242
6243         var transEndEventNames = {
6244             WebkitTransition : 'webkitTransitionEnd',
6245             MozTransition    : 'transitionend',
6246             OTransition      : 'oTransitionEnd otransitionend',
6247             transition       : 'transitionend'
6248         };
6249     
6250         for (var name in transEndEventNames) {
6251             if (el.style[name] !== undefined) {
6252                 transitionEndVal = transEndEventNames[name];
6253                 return  transitionEndVal ;
6254             }
6255         }
6256     }
6257     
6258
6259     var listen = function(element, ename, opt, fn, scope){
6260         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6261         fn = fn || o.fn; scope = scope || o.scope;
6262         var el = Roo.getDom(element);
6263         
6264         
6265         if(!el){
6266             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6267         }
6268         
6269         if (ename == 'transitionend') {
6270             ename = transitionEnd();
6271         }
6272         var h = function(e){
6273             e = Roo.EventObject.setEvent(e);
6274             var t;
6275             if(o.delegate){
6276                 t = e.getTarget(o.delegate, el);
6277                 if(!t){
6278                     return;
6279                 }
6280             }else{
6281                 t = e.target;
6282             }
6283             if(o.stopEvent === true){
6284                 e.stopEvent();
6285             }
6286             if(o.preventDefault === true){
6287                e.preventDefault();
6288             }
6289             if(o.stopPropagation === true){
6290                 e.stopPropagation();
6291             }
6292
6293             if(o.normalized === false){
6294                 e = e.browserEvent;
6295             }
6296
6297             fn.call(scope || el, e, t, o);
6298         };
6299         if(o.delay){
6300             h = createDelayed(h, o);
6301         }
6302         if(o.single){
6303             h = createSingle(h, el, ename, fn);
6304         }
6305         if(o.buffer){
6306             h = createBuffered(h, o);
6307         }
6308         
6309         fn._handlers = fn._handlers || [];
6310         
6311         
6312         fn._handlers.push([Roo.id(el), ename, h]);
6313         
6314         
6315          
6316         E.on(el, ename, h);
6317         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6318             el.addEventListener("DOMMouseScroll", h, false);
6319             E.on(window, 'unload', function(){
6320                 el.removeEventListener("DOMMouseScroll", h, false);
6321             });
6322         }
6323         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6324             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6325         }
6326         return h;
6327     };
6328
6329     var stopListening = function(el, ename, fn){
6330         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6331         if(hds){
6332             for(var i = 0, len = hds.length; i < len; i++){
6333                 var h = hds[i];
6334                 if(h[0] == id && h[1] == ename){
6335                     hd = h[2];
6336                     hds.splice(i, 1);
6337                     break;
6338                 }
6339             }
6340         }
6341         E.un(el, ename, hd);
6342         el = Roo.getDom(el);
6343         if(ename == "mousewheel" && el.addEventListener){
6344             el.removeEventListener("DOMMouseScroll", hd, false);
6345         }
6346         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6347             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6348         }
6349     };
6350
6351     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6352     
6353     var pub = {
6354         
6355         
6356         /** 
6357          * Fix for doc tools
6358          * @scope Roo.EventManager
6359          */
6360         
6361         
6362         /** 
6363          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6364          * object with a Roo.EventObject
6365          * @param {Function} fn        The method the event invokes
6366          * @param {Object}   scope    An object that becomes the scope of the handler
6367          * @param {boolean}  override If true, the obj passed in becomes
6368          *                             the execution scope of the listener
6369          * @return {Function} The wrapped function
6370          * @deprecated
6371          */
6372         wrap : function(fn, scope, override){
6373             return function(e){
6374                 Roo.EventObject.setEvent(e);
6375                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6376             };
6377         },
6378         
6379         /**
6380      * Appends an event handler to an element (shorthand for addListener)
6381      * @param {String/HTMLElement}   element        The html element or id to assign the
6382      * @param {String}   eventName The type of event to listen for
6383      * @param {Function} handler The method the event invokes
6384      * @param {Object}   scope (optional) The scope in which to execute the handler
6385      * function. The handler function's "this" context.
6386      * @param {Object}   options (optional) An object containing handler configuration
6387      * properties. This may contain any of the following properties:<ul>
6388      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6389      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6390      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6391      * <li>preventDefault {Boolean} True to prevent the default action</li>
6392      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6393      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6394      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6395      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6396      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6397      * by the specified number of milliseconds. If the event fires again within that time, the original
6398      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6399      * </ul><br>
6400      * <p>
6401      * <b>Combining Options</b><br>
6402      * Using the options argument, it is possible to combine different types of listeners:<br>
6403      * <br>
6404      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6405      * Code:<pre><code>
6406 el.on('click', this.onClick, this, {
6407     single: true,
6408     delay: 100,
6409     stopEvent : true,
6410     forumId: 4
6411 });</code></pre>
6412      * <p>
6413      * <b>Attaching multiple handlers in 1 call</b><br>
6414       * The method also allows for a single argument to be passed which is a config object containing properties
6415      * which specify multiple handlers.
6416      * <p>
6417      * Code:<pre><code>
6418 el.on({
6419     'click' : {
6420         fn: this.onClick
6421         scope: this,
6422         delay: 100
6423     },
6424     'mouseover' : {
6425         fn: this.onMouseOver
6426         scope: this
6427     },
6428     'mouseout' : {
6429         fn: this.onMouseOut
6430         scope: this
6431     }
6432 });</code></pre>
6433      * <p>
6434      * Or a shorthand syntax:<br>
6435      * Code:<pre><code>
6436 el.on({
6437     'click' : this.onClick,
6438     'mouseover' : this.onMouseOver,
6439     'mouseout' : this.onMouseOut
6440     scope: this
6441 });</code></pre>
6442      */
6443         addListener : function(element, eventName, fn, scope, options){
6444             if(typeof eventName == "object"){
6445                 var o = eventName;
6446                 for(var e in o){
6447                     if(propRe.test(e)){
6448                         continue;
6449                     }
6450                     if(typeof o[e] == "function"){
6451                         // shared options
6452                         listen(element, e, o, o[e], o.scope);
6453                     }else{
6454                         // individual options
6455                         listen(element, e, o[e]);
6456                     }
6457                 }
6458                 return;
6459             }
6460             return listen(element, eventName, options, fn, scope);
6461         },
6462         
6463         /**
6464          * Removes an event handler
6465          *
6466          * @param {String/HTMLElement}   element        The id or html element to remove the 
6467          *                             event from
6468          * @param {String}   eventName     The type of event
6469          * @param {Function} fn
6470          * @return {Boolean} True if a listener was actually removed
6471          */
6472         removeListener : function(element, eventName, fn){
6473             return stopListening(element, eventName, fn);
6474         },
6475         
6476         /**
6477          * Fires when the document is ready (before onload and before images are loaded). Can be 
6478          * accessed shorthanded Roo.onReady().
6479          * @param {Function} fn        The method the event invokes
6480          * @param {Object}   scope    An  object that becomes the scope of the handler
6481          * @param {boolean}  options
6482          */
6483         onDocumentReady : function(fn, scope, options){
6484             if(docReadyState){ // if it already fired
6485                 docReadyEvent.addListener(fn, scope, options);
6486                 docReadyEvent.fire();
6487                 docReadyEvent.clearListeners();
6488                 return;
6489             }
6490             if(!docReadyEvent){
6491                 initDocReady();
6492             }
6493             docReadyEvent.addListener(fn, scope, options);
6494         },
6495         
6496         /**
6497          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6498          * @param {Function} fn        The method the event invokes
6499          * @param {Object}   scope    An object that becomes the scope of the handler
6500          * @param {boolean}  options
6501          */
6502         onWindowResize : function(fn, scope, options){
6503             if(!resizeEvent){
6504                 resizeEvent = new Roo.util.Event();
6505                 resizeTask = new Roo.util.DelayedTask(function(){
6506                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6507                 });
6508                 E.on(window, "resize", function(){
6509                     if(Roo.isIE){
6510                         resizeTask.delay(50);
6511                     }else{
6512                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6513                     }
6514                 });
6515             }
6516             resizeEvent.addListener(fn, scope, options);
6517         },
6518
6519         /**
6520          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6521          * @param {Function} fn        The method the event invokes
6522          * @param {Object}   scope    An object that becomes the scope of the handler
6523          * @param {boolean}  options
6524          */
6525         onTextResize : function(fn, scope, options){
6526             if(!textEvent){
6527                 textEvent = new Roo.util.Event();
6528                 var textEl = new Roo.Element(document.createElement('div'));
6529                 textEl.dom.className = 'x-text-resize';
6530                 textEl.dom.innerHTML = 'X';
6531                 textEl.appendTo(document.body);
6532                 textSize = textEl.dom.offsetHeight;
6533                 setInterval(function(){
6534                     if(textEl.dom.offsetHeight != textSize){
6535                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6536                     }
6537                 }, this.textResizeInterval);
6538             }
6539             textEvent.addListener(fn, scope, options);
6540         },
6541
6542         /**
6543          * Removes the passed window resize listener.
6544          * @param {Function} fn        The method the event invokes
6545          * @param {Object}   scope    The scope of handler
6546          */
6547         removeResizeListener : function(fn, scope){
6548             if(resizeEvent){
6549                 resizeEvent.removeListener(fn, scope);
6550             }
6551         },
6552
6553         // private
6554         fireResize : function(){
6555             if(resizeEvent){
6556                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6557             }   
6558         },
6559         /**
6560          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6561          */
6562         ieDeferSrc : false,
6563         /**
6564          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6565          */
6566         textResizeInterval : 50
6567     };
6568     
6569     /**
6570      * Fix for doc tools
6571      * @scopeAlias pub=Roo.EventManager
6572      */
6573     
6574      /**
6575      * Appends an event handler to an element (shorthand for addListener)
6576      * @param {String/HTMLElement}   element        The html element or id to assign the
6577      * @param {String}   eventName The type of event to listen for
6578      * @param {Function} handler The method the event invokes
6579      * @param {Object}   scope (optional) The scope in which to execute the handler
6580      * function. The handler function's "this" context.
6581      * @param {Object}   options (optional) An object containing handler configuration
6582      * properties. This may contain any of the following properties:<ul>
6583      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6584      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6585      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6586      * <li>preventDefault {Boolean} True to prevent the default action</li>
6587      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6588      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6589      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6590      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6591      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6592      * by the specified number of milliseconds. If the event fires again within that time, the original
6593      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6594      * </ul><br>
6595      * <p>
6596      * <b>Combining Options</b><br>
6597      * Using the options argument, it is possible to combine different types of listeners:<br>
6598      * <br>
6599      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6600      * Code:<pre><code>
6601 el.on('click', this.onClick, this, {
6602     single: true,
6603     delay: 100,
6604     stopEvent : true,
6605     forumId: 4
6606 });</code></pre>
6607      * <p>
6608      * <b>Attaching multiple handlers in 1 call</b><br>
6609       * The method also allows for a single argument to be passed which is a config object containing properties
6610      * which specify multiple handlers.
6611      * <p>
6612      * Code:<pre><code>
6613 el.on({
6614     'click' : {
6615         fn: this.onClick
6616         scope: this,
6617         delay: 100
6618     },
6619     'mouseover' : {
6620         fn: this.onMouseOver
6621         scope: this
6622     },
6623     'mouseout' : {
6624         fn: this.onMouseOut
6625         scope: this
6626     }
6627 });</code></pre>
6628      * <p>
6629      * Or a shorthand syntax:<br>
6630      * Code:<pre><code>
6631 el.on({
6632     'click' : this.onClick,
6633     'mouseover' : this.onMouseOver,
6634     'mouseout' : this.onMouseOut
6635     scope: this
6636 });</code></pre>
6637      */
6638     pub.on = pub.addListener;
6639     pub.un = pub.removeListener;
6640
6641     pub.stoppedMouseDownEvent = new Roo.util.Event();
6642     return pub;
6643 }();
6644 /**
6645   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6646   * @param {Function} fn        The method the event invokes
6647   * @param {Object}   scope    An  object that becomes the scope of the handler
6648   * @param {boolean}  override If true, the obj passed in becomes
6649   *                             the execution scope of the listener
6650   * @member Roo
6651   * @method onReady
6652  */
6653 Roo.onReady = Roo.EventManager.onDocumentReady;
6654
6655 Roo.onReady(function(){
6656     var bd = Roo.get(document.body);
6657     if(!bd){ return; }
6658
6659     var cls = [
6660             Roo.isIE ? "roo-ie"
6661             : Roo.isIE11 ? "roo-ie11"
6662             : Roo.isEdge ? "roo-edge"
6663             : Roo.isGecko ? "roo-gecko"
6664             : Roo.isOpera ? "roo-opera"
6665             : Roo.isSafari ? "roo-safari" : ""];
6666
6667     if(Roo.isMac){
6668         cls.push("roo-mac");
6669     }
6670     if(Roo.isLinux){
6671         cls.push("roo-linux");
6672     }
6673     if(Roo.isIOS){
6674         cls.push("roo-ios");
6675     }
6676     if(Roo.isTouch){
6677         cls.push("roo-touch");
6678     }
6679     if(Roo.isBorderBox){
6680         cls.push('roo-border-box');
6681     }
6682     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6683         var p = bd.dom.parentNode;
6684         if(p){
6685             p.className += ' roo-strict';
6686         }
6687     }
6688     bd.addClass(cls.join(' '));
6689 });
6690
6691 /**
6692  * @class Roo.EventObject
6693  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6694  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6695  * Example:
6696  * <pre><code>
6697  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6698     e.preventDefault();
6699     var target = e.getTarget();
6700     ...
6701  }
6702  var myDiv = Roo.get("myDiv");
6703  myDiv.on("click", handleClick);
6704  //or
6705  Roo.EventManager.on("myDiv", 'click', handleClick);
6706  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6707  </code></pre>
6708  * @singleton
6709  */
6710 Roo.EventObject = function(){
6711     
6712     var E = Roo.lib.Event;
6713     
6714     // safari keypress events for special keys return bad keycodes
6715     var safariKeys = {
6716         63234 : 37, // left
6717         63235 : 39, // right
6718         63232 : 38, // up
6719         63233 : 40, // down
6720         63276 : 33, // page up
6721         63277 : 34, // page down
6722         63272 : 46, // delete
6723         63273 : 36, // home
6724         63275 : 35  // end
6725     };
6726
6727     // normalize button clicks
6728     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6729                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6730
6731     Roo.EventObjectImpl = function(e){
6732         if(e){
6733             this.setEvent(e.browserEvent || e);
6734         }
6735     };
6736     Roo.EventObjectImpl.prototype = {
6737         /**
6738          * Used to fix doc tools.
6739          * @scope Roo.EventObject.prototype
6740          */
6741             
6742
6743         
6744         
6745         /** The normal browser event */
6746         browserEvent : null,
6747         /** The button pressed in a mouse event */
6748         button : -1,
6749         /** True if the shift key was down during the event */
6750         shiftKey : false,
6751         /** True if the control key was down during the event */
6752         ctrlKey : false,
6753         /** True if the alt key was down during the event */
6754         altKey : false,
6755
6756         /** Key constant 
6757         * @type Number */
6758         BACKSPACE : 8,
6759         /** Key constant 
6760         * @type Number */
6761         TAB : 9,
6762         /** Key constant 
6763         * @type Number */
6764         RETURN : 13,
6765         /** Key constant 
6766         * @type Number */
6767         ENTER : 13,
6768         /** Key constant 
6769         * @type Number */
6770         SHIFT : 16,
6771         /** Key constant 
6772         * @type Number */
6773         CONTROL : 17,
6774         /** Key constant 
6775         * @type Number */
6776         ESC : 27,
6777         /** Key constant 
6778         * @type Number */
6779         SPACE : 32,
6780         /** Key constant 
6781         * @type Number */
6782         PAGEUP : 33,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEDOWN : 34,
6786         /** Key constant 
6787         * @type Number */
6788         END : 35,
6789         /** Key constant 
6790         * @type Number */
6791         HOME : 36,
6792         /** Key constant 
6793         * @type Number */
6794         LEFT : 37,
6795         /** Key constant 
6796         * @type Number */
6797         UP : 38,
6798         /** Key constant 
6799         * @type Number */
6800         RIGHT : 39,
6801         /** Key constant 
6802         * @type Number */
6803         DOWN : 40,
6804         /** Key constant 
6805         * @type Number */
6806         DELETE : 46,
6807         /** Key constant 
6808         * @type Number */
6809         F5 : 116,
6810
6811            /** @private */
6812         setEvent : function(e){
6813             if(e == this || (e && e.browserEvent)){ // already wrapped
6814                 return e;
6815             }
6816             this.browserEvent = e;
6817             if(e){
6818                 // normalize buttons
6819                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6820                 if(e.type == 'click' && this.button == -1){
6821                     this.button = 0;
6822                 }
6823                 this.type = e.type;
6824                 this.shiftKey = e.shiftKey;
6825                 // mac metaKey behaves like ctrlKey
6826                 this.ctrlKey = e.ctrlKey || e.metaKey;
6827                 this.altKey = e.altKey;
6828                 // in getKey these will be normalized for the mac
6829                 this.keyCode = e.keyCode;
6830                 // keyup warnings on firefox.
6831                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6832                 // cache the target for the delayed and or buffered events
6833                 this.target = E.getTarget(e);
6834                 // same for XY
6835                 this.xy = E.getXY(e);
6836             }else{
6837                 this.button = -1;
6838                 this.shiftKey = false;
6839                 this.ctrlKey = false;
6840                 this.altKey = false;
6841                 this.keyCode = 0;
6842                 this.charCode =0;
6843                 this.target = null;
6844                 this.xy = [0, 0];
6845             }
6846             return this;
6847         },
6848
6849         /**
6850          * Stop the event (preventDefault and stopPropagation)
6851          */
6852         stopEvent : function(){
6853             if(this.browserEvent){
6854                 if(this.browserEvent.type == 'mousedown'){
6855                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6856                 }
6857                 E.stopEvent(this.browserEvent);
6858             }
6859         },
6860
6861         /**
6862          * Prevents the browsers default handling of the event.
6863          */
6864         preventDefault : function(){
6865             if(this.browserEvent){
6866                 E.preventDefault(this.browserEvent);
6867             }
6868         },
6869
6870         /** @private */
6871         isNavKeyPress : function(){
6872             var k = this.keyCode;
6873             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6874             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6875         },
6876
6877         isSpecialKey : function(){
6878             var k = this.keyCode;
6879             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6880             (k == 16) || (k == 17) ||
6881             (k >= 18 && k <= 20) ||
6882             (k >= 33 && k <= 35) ||
6883             (k >= 36 && k <= 39) ||
6884             (k >= 44 && k <= 45);
6885         },
6886         /**
6887          * Cancels bubbling of the event.
6888          */
6889         stopPropagation : function(){
6890             if(this.browserEvent){
6891                 if(this.type == 'mousedown'){
6892                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6893                 }
6894                 E.stopPropagation(this.browserEvent);
6895             }
6896         },
6897
6898         /**
6899          * Gets the key code for the event.
6900          * @return {Number}
6901          */
6902         getCharCode : function(){
6903             return this.charCode || this.keyCode;
6904         },
6905
6906         /**
6907          * Returns a normalized keyCode for the event.
6908          * @return {Number} The key code
6909          */
6910         getKey : function(){
6911             var k = this.keyCode || this.charCode;
6912             return Roo.isSafari ? (safariKeys[k] || k) : k;
6913         },
6914
6915         /**
6916          * Gets the x coordinate of the event.
6917          * @return {Number}
6918          */
6919         getPageX : function(){
6920             return this.xy[0];
6921         },
6922
6923         /**
6924          * Gets the y coordinate of the event.
6925          * @return {Number}
6926          */
6927         getPageY : function(){
6928             return this.xy[1];
6929         },
6930
6931         /**
6932          * Gets the time of the event.
6933          * @return {Number}
6934          */
6935         getTime : function(){
6936             if(this.browserEvent){
6937                 return E.getTime(this.browserEvent);
6938             }
6939             return null;
6940         },
6941
6942         /**
6943          * Gets the page coordinates of the event.
6944          * @return {Array} The xy values like [x, y]
6945          */
6946         getXY : function(){
6947             return this.xy;
6948         },
6949
6950         /**
6951          * Gets the target for the event.
6952          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6953          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6954                 search as a number or element (defaults to 10 || document.body)
6955          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6956          * @return {HTMLelement}
6957          */
6958         getTarget : function(selector, maxDepth, returnEl){
6959             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6960         },
6961         /**
6962          * Gets the related target.
6963          * @return {HTMLElement}
6964          */
6965         getRelatedTarget : function(){
6966             if(this.browserEvent){
6967                 return E.getRelatedTarget(this.browserEvent);
6968             }
6969             return null;
6970         },
6971
6972         /**
6973          * Normalizes mouse wheel delta across browsers
6974          * @return {Number} The delta
6975          */
6976         getWheelDelta : function(){
6977             var e = this.browserEvent;
6978             var delta = 0;
6979             if(e.wheelDelta){ /* IE/Opera. */
6980                 delta = e.wheelDelta/120;
6981             }else if(e.detail){ /* Mozilla case. */
6982                 delta = -e.detail/3;
6983             }
6984             return delta;
6985         },
6986
6987         /**
6988          * Returns true if the control, meta, shift or alt key was pressed during this event.
6989          * @return {Boolean}
6990          */
6991         hasModifier : function(){
6992             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6993         },
6994
6995         /**
6996          * Returns true if the target of this event equals el or is a child of el
6997          * @param {String/HTMLElement/Element} el
6998          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
6999          * @return {Boolean}
7000          */
7001         within : function(el, related){
7002             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7003             return t && Roo.fly(el).contains(t);
7004         },
7005
7006         getPoint : function(){
7007             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7008         }
7009     };
7010
7011     return new Roo.EventObjectImpl();
7012 }();
7013             
7014     /*
7015  * Based on:
7016  * Ext JS Library 1.1.1
7017  * Copyright(c) 2006-2007, Ext JS, LLC.
7018  *
7019  * Originally Released Under LGPL - original licence link has changed is not relivant.
7020  *
7021  * Fork - LGPL
7022  * <script type="text/javascript">
7023  */
7024
7025  
7026 // was in Composite Element!??!?!
7027  
7028 (function(){
7029     var D = Roo.lib.Dom;
7030     var E = Roo.lib.Event;
7031     var A = Roo.lib.Anim;
7032
7033     // local style camelizing for speed
7034     var propCache = {};
7035     var camelRe = /(-[a-z])/gi;
7036     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7037     var view = document.defaultView;
7038
7039 /**
7040  * @class Roo.Element
7041  * Represents an Element in the DOM.<br><br>
7042  * Usage:<br>
7043 <pre><code>
7044 var el = Roo.get("my-div");
7045
7046 // or with getEl
7047 var el = getEl("my-div");
7048
7049 // or with a DOM element
7050 var el = Roo.get(myDivElement);
7051 </code></pre>
7052  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7053  * each call instead of constructing a new one.<br><br>
7054  * <b>Animations</b><br />
7055  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7056  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7057 <pre>
7058 Option    Default   Description
7059 --------- --------  ---------------------------------------------
7060 duration  .35       The duration of the animation in seconds
7061 easing    easeOut   The YUI easing method
7062 callback  none      A function to execute when the anim completes
7063 scope     this      The scope (this) of the callback function
7064 </pre>
7065 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7066 * manipulate the animation. Here's an example:
7067 <pre><code>
7068 var el = Roo.get("my-div");
7069
7070 // no animation
7071 el.setWidth(100);
7072
7073 // default animation
7074 el.setWidth(100, true);
7075
7076 // animation with some options set
7077 el.setWidth(100, {
7078     duration: 1,
7079     callback: this.foo,
7080     scope: this
7081 });
7082
7083 // using the "anim" property to get the Anim object
7084 var opt = {
7085     duration: 1,
7086     callback: this.foo,
7087     scope: this
7088 };
7089 el.setWidth(100, opt);
7090 ...
7091 if(opt.anim.isAnimated()){
7092     opt.anim.stop();
7093 }
7094 </code></pre>
7095 * <b> Composite (Collections of) Elements</b><br />
7096  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7097  * @constructor Create a new Element directly.
7098  * @param {String/HTMLElement} element
7099  * @param {Boolean} forceNew (optional) By default the constructor checks to see if there is already an instance of this element in the cache and if there is it returns the same instance. This will skip that check (useful for extending this class).
7100  */
7101     Roo.Element = function(element, forceNew){
7102         var dom = typeof element == "string" ?
7103                 document.getElementById(element) : element;
7104         if(!dom){ // invalid id/element
7105             return null;
7106         }
7107         var id = dom.id;
7108         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7109             return Roo.Element.cache[id];
7110         }
7111
7112         /**
7113          * The DOM element
7114          * @type HTMLElement
7115          */
7116         this.dom = dom;
7117
7118         /**
7119          * The DOM element ID
7120          * @type String
7121          */
7122         this.id = id || Roo.id(dom);
7123     };
7124
7125     var El = Roo.Element;
7126
7127     El.prototype = {
7128         /**
7129          * The element's default display mode  (defaults to "")
7130          * @type String
7131          */
7132         originalDisplay : "",
7133
7134         visibilityMode : 1,
7135         /**
7136          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7137          * @type String
7138          */
7139         defaultUnit : "px",
7140         
7141         /**
7142          * Sets the element's visibility mode. When setVisible() is called it
7143          * will use this to determine whether to set the visibility or the display property.
7144          * @param visMode Element.VISIBILITY or Element.DISPLAY
7145          * @return {Roo.Element} this
7146          */
7147         setVisibilityMode : function(visMode){
7148             this.visibilityMode = visMode;
7149             return this;
7150         },
7151         /**
7152          * Convenience method for setVisibilityMode(Element.DISPLAY)
7153          * @param {String} display (optional) What to set display to when visible
7154          * @return {Roo.Element} this
7155          */
7156         enableDisplayMode : function(display){
7157             this.setVisibilityMode(El.DISPLAY);
7158             if(typeof display != "undefined") { this.originalDisplay = display; }
7159             return this;
7160         },
7161
7162         /**
7163          * Looks at this node and then at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7164          * @param {String} selector The simple selector to test
7165          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7166                 search as a number or element (defaults to 10 || document.body)
7167          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7168          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7169          */
7170         findParent : function(simpleSelector, maxDepth, returnEl){
7171             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7172             maxDepth = maxDepth || 50;
7173             if(typeof maxDepth != "number"){
7174                 stopEl = Roo.getDom(maxDepth);
7175                 maxDepth = 10;
7176             }
7177             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7178                 if(dq.is(p, simpleSelector)){
7179                     return returnEl ? Roo.get(p) : p;
7180                 }
7181                 depth++;
7182                 p = p.parentNode;
7183             }
7184             return null;
7185         },
7186
7187
7188         /**
7189          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7190          * @param {String} selector The simple selector to test
7191          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7192                 search as a number or element (defaults to 10 || document.body)
7193          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7194          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7195          */
7196         findParentNode : function(simpleSelector, maxDepth, returnEl){
7197             var p = Roo.fly(this.dom.parentNode, '_internal');
7198             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7199         },
7200         
7201         /**
7202          * Looks at  the scrollable parent element
7203          */
7204         findScrollableParent : function()
7205         {
7206             var overflowRegex = /(auto|scroll)/;
7207             
7208             if(this.getStyle('position') === 'fixed'){
7209                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7210             }
7211             
7212             var excludeStaticParent = this.getStyle('position') === "absolute";
7213             
7214             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7215                 
7216                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7217                     continue;
7218                 }
7219                 
7220                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7221                     return parent;
7222                 }
7223                 
7224                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7225                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7226                 }
7227             }
7228             
7229             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7230         },
7231
7232         /**
7233          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7234          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7235          * @param {String} selector The simple selector to test
7236          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7237                 search as a number or element (defaults to 10 || document.body)
7238          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7239          */
7240         up : function(simpleSelector, maxDepth){
7241             return this.findParentNode(simpleSelector, maxDepth, true);
7242         },
7243
7244
7245
7246         /**
7247          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7248          * @param {String} selector The simple selector to test
7249          * @return {Boolean} True if this element matches the selector, else false
7250          */
7251         is : function(simpleSelector){
7252             return Roo.DomQuery.is(this.dom, simpleSelector);
7253         },
7254
7255         /**
7256          * Perform animation on this element.
7257          * @param {Object} args The YUI animation control args
7258          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7259          * @param {Function} onComplete (optional) Function to call when animation completes
7260          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7261          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7262          * @return {Roo.Element} this
7263          */
7264         animate : function(args, duration, onComplete, easing, animType){
7265             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7266             return this;
7267         },
7268
7269         /*
7270          * @private Internal animation call
7271          */
7272         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7273             animType = animType || 'run';
7274             opt = opt || {};
7275             var anim = Roo.lib.Anim[animType](
7276                 this.dom, args,
7277                 (opt.duration || defaultDur) || .35,
7278                 (opt.easing || defaultEase) || 'easeOut',
7279                 function(){
7280                     Roo.callback(cb, this);
7281                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7282                 },
7283                 this
7284             );
7285             opt.anim = anim;
7286             return anim;
7287         },
7288
7289         // private legacy anim prep
7290         preanim : function(a, i){
7291             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7292         },
7293
7294         /**
7295          * Removes worthless text nodes
7296          * @param {Boolean} forceReclean (optional) By default the element
7297          * keeps track if it has been cleaned already so
7298          * you can call this over and over. However, if you update the element and
7299          * need to force a reclean, you can pass true.
7300          */
7301         clean : function(forceReclean){
7302             if(this.isCleaned && forceReclean !== true){
7303                 return this;
7304             }
7305             var ns = /\S/;
7306             var d = this.dom, n = d.firstChild, ni = -1;
7307             while(n){
7308                 var nx = n.nextSibling;
7309                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7310                     d.removeChild(n);
7311                 }else{
7312                     n.nodeIndex = ++ni;
7313                 }
7314                 n = nx;
7315             }
7316             this.isCleaned = true;
7317             return this;
7318         },
7319
7320         // private
7321         calcOffsetsTo : function(el){
7322             el = Roo.get(el);
7323             var d = el.dom;
7324             var restorePos = false;
7325             if(el.getStyle('position') == 'static'){
7326                 el.position('relative');
7327                 restorePos = true;
7328             }
7329             var x = 0, y =0;
7330             var op = this.dom;
7331             while(op && op != d && op.tagName != 'HTML'){
7332                 x+= op.offsetLeft;
7333                 y+= op.offsetTop;
7334                 op = op.offsetParent;
7335             }
7336             if(restorePos){
7337                 el.position('static');
7338             }
7339             return [x, y];
7340         },
7341
7342         /**
7343          * Scrolls this element into view within the passed container.
7344          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7345          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7346          * @return {Roo.Element} this
7347          */
7348         scrollIntoView : function(container, hscroll){
7349             var c = Roo.getDom(container) || document.body;
7350             var el = this.dom;
7351
7352             var o = this.calcOffsetsTo(c),
7353                 l = o[0],
7354                 t = o[1],
7355                 b = t+el.offsetHeight,
7356                 r = l+el.offsetWidth;
7357
7358             var ch = c.clientHeight;
7359             var ct = parseInt(c.scrollTop, 10);
7360             var cl = parseInt(c.scrollLeft, 10);
7361             var cb = ct + ch;
7362             var cr = cl + c.clientWidth;
7363
7364             if(t < ct){
7365                 c.scrollTop = t;
7366             }else if(b > cb){
7367                 c.scrollTop = b-ch;
7368             }
7369
7370             if(hscroll !== false){
7371                 if(l < cl){
7372                     c.scrollLeft = l;
7373                 }else if(r > cr){
7374                     c.scrollLeft = r-c.clientWidth;
7375                 }
7376             }
7377             return this;
7378         },
7379
7380         // private
7381         scrollChildIntoView : function(child, hscroll){
7382             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7383         },
7384
7385         /**
7386          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7387          * the new height may not be available immediately.
7388          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7389          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7390          * @param {Function} onComplete (optional) Function to call when animation completes
7391          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7392          * @return {Roo.Element} this
7393          */
7394         autoHeight : function(animate, duration, onComplete, easing){
7395             var oldHeight = this.getHeight();
7396             this.clip();
7397             this.setHeight(1); // force clipping
7398             setTimeout(function(){
7399                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7400                 if(!animate){
7401                     this.setHeight(height);
7402                     this.unclip();
7403                     if(typeof onComplete == "function"){
7404                         onComplete();
7405                     }
7406                 }else{
7407                     this.setHeight(oldHeight); // restore original height
7408                     this.setHeight(height, animate, duration, function(){
7409                         this.unclip();
7410                         if(typeof onComplete == "function") { onComplete(); }
7411                     }.createDelegate(this), easing);
7412                 }
7413             }.createDelegate(this), 0);
7414             return this;
7415         },
7416
7417         /**
7418          * Returns true if this element is an ancestor of the passed element
7419          * @param {HTMLElement/String} el The element to check
7420          * @return {Boolean} True if this element is an ancestor of el, else false
7421          */
7422         contains : function(el){
7423             if(!el){return false;}
7424             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7425         },
7426
7427         /**
7428          * Checks whether the element is currently visible using both visibility and display properties.
7429          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7430          * @return {Boolean} True if the element is currently visible, else false
7431          */
7432         isVisible : function(deep) {
7433             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7434             if(deep !== true || !vis){
7435                 return vis;
7436             }
7437             var p = this.dom.parentNode;
7438             while(p && p.tagName.toLowerCase() != "body"){
7439                 if(!Roo.fly(p, '_isVisible').isVisible()){
7440                     return false;
7441                 }
7442                 p = p.parentNode;
7443             }
7444             return true;
7445         },
7446
7447         /**
7448          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7449          * @param {String} selector The CSS selector
7450          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7451          * @return {CompositeElement/CompositeElementLite} The composite element
7452          */
7453         select : function(selector, unique){
7454             return El.select(selector, unique, this.dom);
7455         },
7456
7457         /**
7458          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7459          * @param {String} selector The CSS selector
7460          * @return {Array} An array of the matched nodes
7461          */
7462         query : function(selector, unique){
7463             return Roo.DomQuery.select(selector, this.dom);
7464         },
7465
7466         /**
7467          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7468          * @param {String} selector The CSS selector
7469          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7470          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7471          */
7472         child : function(selector, returnDom){
7473             var n = Roo.DomQuery.selectNode(selector, this.dom);
7474             return returnDom ? n : Roo.get(n);
7475         },
7476
7477         /**
7478          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7479          * @param {String} selector The CSS selector
7480          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7481          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7482          */
7483         down : function(selector, returnDom){
7484             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7485             return returnDom ? n : Roo.get(n);
7486         },
7487
7488         /**
7489          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7490          * @param {String} group The group the DD object is member of
7491          * @param {Object} config The DD config object
7492          * @param {Object} overrides An object containing methods to override/implement on the DD object
7493          * @return {Roo.dd.DD} The DD object
7494          */
7495         initDD : function(group, config, overrides){
7496             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7497             return Roo.apply(dd, overrides);
7498         },
7499
7500         /**
7501          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7502          * @param {String} group The group the DDProxy object is member of
7503          * @param {Object} config The DDProxy config object
7504          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7505          * @return {Roo.dd.DDProxy} The DDProxy object
7506          */
7507         initDDProxy : function(group, config, overrides){
7508             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7509             return Roo.apply(dd, overrides);
7510         },
7511
7512         /**
7513          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7514          * @param {String} group The group the DDTarget object is member of
7515          * @param {Object} config The DDTarget config object
7516          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7517          * @return {Roo.dd.DDTarget} The DDTarget object
7518          */
7519         initDDTarget : function(group, config, overrides){
7520             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7521             return Roo.apply(dd, overrides);
7522         },
7523
7524         /**
7525          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7526          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7527          * @param {Boolean} visible Whether the element is visible
7528          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7529          * @return {Roo.Element} this
7530          */
7531          setVisible : function(visible, animate){
7532             if(!animate || !A){
7533                 if(this.visibilityMode == El.DISPLAY){
7534                     this.setDisplayed(visible);
7535                 }else{
7536                     this.fixDisplay();
7537                     this.dom.style.visibility = visible ? "visible" : "hidden";
7538                 }
7539             }else{
7540                 // closure for composites
7541                 var dom = this.dom;
7542                 var visMode = this.visibilityMode;
7543                 if(visible){
7544                     this.setOpacity(.01);
7545                     this.setVisible(true);
7546                 }
7547                 this.anim({opacity: { to: (visible?1:0) }},
7548                       this.preanim(arguments, 1),
7549                       null, .35, 'easeIn', function(){
7550                          if(!visible){
7551                              if(visMode == El.DISPLAY){
7552                                  dom.style.display = "none";
7553                              }else{
7554                                  dom.style.visibility = "hidden";
7555                              }
7556                              Roo.get(dom).setOpacity(1);
7557                          }
7558                      });
7559             }
7560             return this;
7561         },
7562
7563         /**
7564          * Returns true if display is not "none"
7565          * @return {Boolean}
7566          */
7567         isDisplayed : function() {
7568             return this.getStyle("display") != "none";
7569         },
7570
7571         /**
7572          * Toggles the element's visibility or display, depending on visibility mode.
7573          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7574          * @return {Roo.Element} this
7575          */
7576         toggle : function(animate){
7577             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7578             return this;
7579         },
7580
7581         /**
7582          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7583          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7584          * @return {Roo.Element} this
7585          */
7586         setDisplayed : function(value) {
7587             if(typeof value == "boolean"){
7588                value = value ? this.originalDisplay : "none";
7589             }
7590             this.setStyle("display", value);
7591             return this;
7592         },
7593
7594         /**
7595          * Tries to focus the element. Any exceptions are caught and ignored.
7596          * @return {Roo.Element} this
7597          */
7598         focus : function() {
7599             try{
7600                 this.dom.focus();
7601             }catch(e){}
7602             return this;
7603         },
7604
7605         /**
7606          * Tries to blur the element. Any exceptions are caught and ignored.
7607          * @return {Roo.Element} this
7608          */
7609         blur : function() {
7610             try{
7611                 this.dom.blur();
7612             }catch(e){}
7613             return this;
7614         },
7615
7616         /**
7617          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7618          * @param {String/Array} className The CSS class to add, or an array of classes
7619          * @return {Roo.Element} this
7620          */
7621         addClass : function(className){
7622             if(className instanceof Array){
7623                 for(var i = 0, len = className.length; i < len; i++) {
7624                     this.addClass(className[i]);
7625                 }
7626             }else{
7627                 if(className && !this.hasClass(className)){
7628                     this.dom.className = this.dom.className + " " + className;
7629                 }
7630             }
7631             return this;
7632         },
7633
7634         /**
7635          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7636          * @param {String/Array} className The CSS class to add, or an array of classes
7637          * @return {Roo.Element} this
7638          */
7639         radioClass : function(className){
7640             var siblings = this.dom.parentNode.childNodes;
7641             for(var i = 0; i < siblings.length; i++) {
7642                 var s = siblings[i];
7643                 if(s.nodeType == 1){
7644                     Roo.get(s).removeClass(className);
7645                 }
7646             }
7647             this.addClass(className);
7648             return this;
7649         },
7650
7651         /**
7652          * Removes one or more CSS classes from the element.
7653          * @param {String/Array} className The CSS class to remove, or an array of classes
7654          * @return {Roo.Element} this
7655          */
7656         removeClass : function(className){
7657             if(!className || !this.dom.className){
7658                 return this;
7659             }
7660             if(className instanceof Array){
7661                 for(var i = 0, len = className.length; i < len; i++) {
7662                     this.removeClass(className[i]);
7663                 }
7664             }else{
7665                 if(this.hasClass(className)){
7666                     var re = this.classReCache[className];
7667                     if (!re) {
7668                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7669                        this.classReCache[className] = re;
7670                     }
7671                     this.dom.className =
7672                         this.dom.className.replace(re, " ");
7673                 }
7674             }
7675             return this;
7676         },
7677
7678         // private
7679         classReCache: {},
7680
7681         /**
7682          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7683          * @param {String} className The CSS class to toggle
7684          * @return {Roo.Element} this
7685          */
7686         toggleClass : function(className){
7687             if(this.hasClass(className)){
7688                 this.removeClass(className);
7689             }else{
7690                 this.addClass(className);
7691             }
7692             return this;
7693         },
7694
7695         /**
7696          * Checks if the specified CSS class exists on this element's DOM node.
7697          * @param {String} className The CSS class to check for
7698          * @return {Boolean} True if the class exists, else false
7699          */
7700         hasClass : function(className){
7701             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7702         },
7703
7704         /**
7705          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7706          * @param {String} oldClassName The CSS class to replace
7707          * @param {String} newClassName The replacement CSS class
7708          * @return {Roo.Element} this
7709          */
7710         replaceClass : function(oldClassName, newClassName){
7711             this.removeClass(oldClassName);
7712             this.addClass(newClassName);
7713             return this;
7714         },
7715
7716         /**
7717          * Returns an object with properties matching the styles requested.
7718          * For example, el.getStyles('color', 'font-size', 'width') might return
7719          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7720          * @param {String} style1 A style name
7721          * @param {String} style2 A style name
7722          * @param {String} etc.
7723          * @return {Object} The style object
7724          */
7725         getStyles : function(){
7726             var a = arguments, len = a.length, r = {};
7727             for(var i = 0; i < len; i++){
7728                 r[a[i]] = this.getStyle(a[i]);
7729             }
7730             return r;
7731         },
7732
7733         /**
7734          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7735          * @param {String} property The style property whose value is returned.
7736          * @return {String} The current value of the style property for this element.
7737          */
7738         getStyle : function(){
7739             return view && view.getComputedStyle ?
7740                 function(prop){
7741                     var el = this.dom, v, cs, camel;
7742                     if(prop == 'float'){
7743                         prop = "cssFloat";
7744                     }
7745                     if(el.style && (v = el.style[prop])){
7746                         return v;
7747                     }
7748                     if(cs = view.getComputedStyle(el, "")){
7749                         if(!(camel = propCache[prop])){
7750                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7751                         }
7752                         return cs[camel];
7753                     }
7754                     return null;
7755                 } :
7756                 function(prop){
7757                     var el = this.dom, v, cs, camel;
7758                     if(prop == 'opacity'){
7759                         if(typeof el.style.filter == 'string'){
7760                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7761                             if(m){
7762                                 var fv = parseFloat(m[1]);
7763                                 if(!isNaN(fv)){
7764                                     return fv ? fv / 100 : 0;
7765                                 }
7766                             }
7767                         }
7768                         return 1;
7769                     }else if(prop == 'float'){
7770                         prop = "styleFloat";
7771                     }
7772                     if(!(camel = propCache[prop])){
7773                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7774                     }
7775                     if(v = el.style[camel]){
7776                         return v;
7777                     }
7778                     if(cs = el.currentStyle){
7779                         return cs[camel];
7780                     }
7781                     return null;
7782                 };
7783         }(),
7784
7785         /**
7786          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7787          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7788          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7789          * @return {Roo.Element} this
7790          */
7791         setStyle : function(prop, value){
7792             if(typeof prop == "string"){
7793                 
7794                 if (prop == 'float') {
7795                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7796                     return this;
7797                 }
7798                 
7799                 var camel;
7800                 if(!(camel = propCache[prop])){
7801                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7802                 }
7803                 
7804                 if(camel == 'opacity') {
7805                     this.setOpacity(value);
7806                 }else{
7807                     this.dom.style[camel] = value;
7808                 }
7809             }else{
7810                 for(var style in prop){
7811                     if(typeof prop[style] != "function"){
7812                        this.setStyle(style, prop[style]);
7813                     }
7814                 }
7815             }
7816             return this;
7817         },
7818
7819         /**
7820          * More flexible version of {@link #setStyle} for setting style properties.
7821          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7822          * a function which returns such a specification.
7823          * @return {Roo.Element} this
7824          */
7825         applyStyles : function(style){
7826             Roo.DomHelper.applyStyles(this.dom, style);
7827             return this;
7828         },
7829
7830         /**
7831           * Gets the current X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7832           * @return {Number} The X position of the element
7833           */
7834         getX : function(){
7835             return D.getX(this.dom);
7836         },
7837
7838         /**
7839           * Gets the current Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7840           * @return {Number} The Y position of the element
7841           */
7842         getY : function(){
7843             return D.getY(this.dom);
7844         },
7845
7846         /**
7847           * Gets the current position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7848           * @return {Array} The XY position of the element
7849           */
7850         getXY : function(){
7851             return D.getXY(this.dom);
7852         },
7853
7854         /**
7855          * Sets the X position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7856          * @param {Number} The X position of the element
7857          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7858          * @return {Roo.Element} this
7859          */
7860         setX : function(x, animate){
7861             if(!animate || !A){
7862                 D.setX(this.dom, x);
7863             }else{
7864                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7865             }
7866             return this;
7867         },
7868
7869         /**
7870          * Sets the Y position of the element based on page coordinates.  Element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7871          * @param {Number} The Y position of the element
7872          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7873          * @return {Roo.Element} this
7874          */
7875         setY : function(y, animate){
7876             if(!animate || !A){
7877                 D.setY(this.dom, y);
7878             }else{
7879                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7880             }
7881             return this;
7882         },
7883
7884         /**
7885          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7886          * @param {String} left The left CSS property value
7887          * @return {Roo.Element} this
7888          */
7889         setLeft : function(left){
7890             this.setStyle("left", this.addUnits(left));
7891             return this;
7892         },
7893
7894         /**
7895          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7896          * @param {String} top The top CSS property value
7897          * @return {Roo.Element} this
7898          */
7899         setTop : function(top){
7900             this.setStyle("top", this.addUnits(top));
7901             return this;
7902         },
7903
7904         /**
7905          * Sets the element's CSS right style.
7906          * @param {String} right The right CSS property value
7907          * @return {Roo.Element} this
7908          */
7909         setRight : function(right){
7910             this.setStyle("right", this.addUnits(right));
7911             return this;
7912         },
7913
7914         /**
7915          * Sets the element's CSS bottom style.
7916          * @param {String} bottom The bottom CSS property value
7917          * @return {Roo.Element} this
7918          */
7919         setBottom : function(bottom){
7920             this.setStyle("bottom", this.addUnits(bottom));
7921             return this;
7922         },
7923
7924         /**
7925          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7926          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7927          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7928          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7929          * @return {Roo.Element} this
7930          */
7931         setXY : function(pos, animate){
7932             if(!animate || !A){
7933                 D.setXY(this.dom, pos);
7934             }else{
7935                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7936             }
7937             return this;
7938         },
7939
7940         /**
7941          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7942          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7943          * @param {Number} x X value for new position (coordinates are page-based)
7944          * @param {Number} y Y value for new position (coordinates are page-based)
7945          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7946          * @return {Roo.Element} this
7947          */
7948         setLocation : function(x, y, animate){
7949             this.setXY([x, y], this.preanim(arguments, 2));
7950             return this;
7951         },
7952
7953         /**
7954          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7955          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7956          * @param {Number} x X value for new position (coordinates are page-based)
7957          * @param {Number} y Y value for new position (coordinates are page-based)
7958          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7959          * @return {Roo.Element} this
7960          */
7961         moveTo : function(x, y, animate){
7962             this.setXY([x, y], this.preanim(arguments, 2));
7963             return this;
7964         },
7965
7966         /**
7967          * Returns the region of the given element.
7968          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7969          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7970          */
7971         getRegion : function(){
7972             return D.getRegion(this.dom);
7973         },
7974
7975         /**
7976          * Returns the offset height of the element
7977          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7978          * @return {Number} The element's height
7979          */
7980         getHeight : function(contentHeight){
7981             var h = this.dom.offsetHeight || 0;
7982             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7983         },
7984
7985         /**
7986          * Returns the offset width of the element
7987          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7988          * @return {Number} The element's width
7989          */
7990         getWidth : function(contentWidth){
7991             var w = this.dom.offsetWidth || 0;
7992             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7993         },
7994
7995         /**
7996          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
7997          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
7998          * if a height has not been set using CSS.
7999          * @return {Number}
8000          */
8001         getComputedHeight : function(){
8002             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8003             if(!h){
8004                 h = parseInt(this.getStyle('height'), 10) || 0;
8005                 if(!this.isBorderBox()){
8006                     h += this.getFrameWidth('tb');
8007                 }
8008             }
8009             return h;
8010         },
8011
8012         /**
8013          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8014          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8015          * if a width has not been set using CSS.
8016          * @return {Number}
8017          */
8018         getComputedWidth : function(){
8019             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8020             if(!w){
8021                 w = parseInt(this.getStyle('width'), 10) || 0;
8022                 if(!this.isBorderBox()){
8023                     w += this.getFrameWidth('lr');
8024                 }
8025             }
8026             return w;
8027         },
8028
8029         /**
8030          * Returns the size of the element.
8031          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8032          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8033          */
8034         getSize : function(contentSize){
8035             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8036         },
8037
8038         /**
8039          * Returns the width and height of the viewport.
8040          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8041          */
8042         getViewSize : function(){
8043             var d = this.dom, doc = document, aw = 0, ah = 0;
8044             if(d == doc || d == doc.body){
8045                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8046             }else{
8047                 return {
8048                     width : d.clientWidth,
8049                     height: d.clientHeight
8050                 };
8051             }
8052         },
8053
8054         /**
8055          * Returns the value of the "value" attribute
8056          * @param {Boolean} asNumber true to parse the value as a number
8057          * @return {String/Number}
8058          */
8059         getValue : function(asNumber){
8060             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8061         },
8062
8063         // private
8064         adjustWidth : function(width){
8065             if(typeof width == "number"){
8066                 if(this.autoBoxAdjust && !this.isBorderBox()){
8067                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8068                 }
8069                 if(width < 0){
8070                     width = 0;
8071                 }
8072             }
8073             return width;
8074         },
8075
8076         // private
8077         adjustHeight : function(height){
8078             if(typeof height == "number"){
8079                if(this.autoBoxAdjust && !this.isBorderBox()){
8080                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8081                }
8082                if(height < 0){
8083                    height = 0;
8084                }
8085             }
8086             return height;
8087         },
8088
8089         /**
8090          * Set the width of the element
8091          * @param {Number} width The new width
8092          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8093          * @return {Roo.Element} this
8094          */
8095         setWidth : function(width, animate){
8096             width = this.adjustWidth(width);
8097             if(!animate || !A){
8098                 this.dom.style.width = this.addUnits(width);
8099             }else{
8100                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8101             }
8102             return this;
8103         },
8104
8105         /**
8106          * Set the height of the element
8107          * @param {Number} height The new height
8108          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8109          * @return {Roo.Element} this
8110          */
8111          setHeight : function(height, animate){
8112             height = this.adjustHeight(height);
8113             if(!animate || !A){
8114                 this.dom.style.height = this.addUnits(height);
8115             }else{
8116                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8117             }
8118             return this;
8119         },
8120
8121         /**
8122          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8123          * @param {Number} width The new width
8124          * @param {Number} height The new height
8125          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8126          * @return {Roo.Element} this
8127          */
8128          setSize : function(width, height, animate){
8129             if(typeof width == "object"){ // in case of object from getSize()
8130                 height = width.height; width = width.width;
8131             }
8132             width = this.adjustWidth(width); height = this.adjustHeight(height);
8133             if(!animate || !A){
8134                 this.dom.style.width = this.addUnits(width);
8135                 this.dom.style.height = this.addUnits(height);
8136             }else{
8137                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8138             }
8139             return this;
8140         },
8141
8142         /**
8143          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8144          * @param {Number} x X value for new position (coordinates are page-based)
8145          * @param {Number} y Y value for new position (coordinates are page-based)
8146          * @param {Number} width The new width
8147          * @param {Number} height The new height
8148          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8149          * @return {Roo.Element} this
8150          */
8151         setBounds : function(x, y, width, height, animate){
8152             if(!animate || !A){
8153                 this.setSize(width, height);
8154                 this.setLocation(x, y);
8155             }else{
8156                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8157                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8158                               this.preanim(arguments, 4), 'motion');
8159             }
8160             return this;
8161         },
8162
8163         /**
8164          * Sets the element's position and size the the specified region. If animation is true then width, height, x and y will be animated concurrently.
8165          * @param {Roo.lib.Region} region The region to fill
8166          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8167          * @return {Roo.Element} this
8168          */
8169         setRegion : function(region, animate){
8170             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8171             return this;
8172         },
8173
8174         /**
8175          * Appends an event handler
8176          *
8177          * @param {String}   eventName     The type of event to append
8178          * @param {Function} fn        The method the event invokes
8179          * @param {Object} scope       (optional) The scope (this object) of the fn
8180          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8181          */
8182         addListener : function(eventName, fn, scope, options){
8183             if (this.dom) {
8184                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8185             }
8186         },
8187
8188         /**
8189          * Removes an event handler from this element
8190          * @param {String} eventName the type of event to remove
8191          * @param {Function} fn the method the event invokes
8192          * @return {Roo.Element} this
8193          */
8194         removeListener : function(eventName, fn){
8195             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8196             return this;
8197         },
8198
8199         /**
8200          * Removes all previous added listeners from this element
8201          * @return {Roo.Element} this
8202          */
8203         removeAllListeners : function(){
8204             E.purgeElement(this.dom);
8205             return this;
8206         },
8207
8208         relayEvent : function(eventName, observable){
8209             this.on(eventName, function(e){
8210                 observable.fireEvent(eventName, e);
8211             });
8212         },
8213
8214         /**
8215          * Set the opacity of the element
8216          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8217          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8218          * @return {Roo.Element} this
8219          */
8220          setOpacity : function(opacity, animate){
8221             if(!animate || !A){
8222                 var s = this.dom.style;
8223                 if(Roo.isIE){
8224                     s.zoom = 1;
8225                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8226                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8227                 }else{
8228                     s.opacity = opacity;
8229                 }
8230             }else{
8231                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8232             }
8233             return this;
8234         },
8235
8236         /**
8237          * Gets the left X coordinate
8238          * @param {Boolean} local True to get the local css position instead of page coordinate
8239          * @return {Number}
8240          */
8241         getLeft : function(local){
8242             if(!local){
8243                 return this.getX();
8244             }else{
8245                 return parseInt(this.getStyle("left"), 10) || 0;
8246             }
8247         },
8248
8249         /**
8250          * Gets the right X coordinate of the element (element X position + element width)
8251          * @param {Boolean} local True to get the local css position instead of page coordinate
8252          * @return {Number}
8253          */
8254         getRight : function(local){
8255             if(!local){
8256                 return this.getX() + this.getWidth();
8257             }else{
8258                 return (this.getLeft(true) + this.getWidth()) || 0;
8259             }
8260         },
8261
8262         /**
8263          * Gets the top Y coordinate
8264          * @param {Boolean} local True to get the local css position instead of page coordinate
8265          * @return {Number}
8266          */
8267         getTop : function(local) {
8268             if(!local){
8269                 return this.getY();
8270             }else{
8271                 return parseInt(this.getStyle("top"), 10) || 0;
8272             }
8273         },
8274
8275         /**
8276          * Gets the bottom Y coordinate of the element (element Y position + element height)
8277          * @param {Boolean} local True to get the local css position instead of page coordinate
8278          * @return {Number}
8279          */
8280         getBottom : function(local){
8281             if(!local){
8282                 return this.getY() + this.getHeight();
8283             }else{
8284                 return (this.getTop(true) + this.getHeight()) || 0;
8285             }
8286         },
8287
8288         /**
8289         * Initializes positioning on this element. If a desired position is not passed, it will make the
8290         * the element positioned relative IF it is not already positioned.
8291         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8292         * @param {Number} zIndex (optional) The zIndex to apply
8293         * @param {Number} x (optional) Set the page X position
8294         * @param {Number} y (optional) Set the page Y position
8295         */
8296         position : function(pos, zIndex, x, y){
8297             if(!pos){
8298                if(this.getStyle('position') == 'static'){
8299                    this.setStyle('position', 'relative');
8300                }
8301             }else{
8302                 this.setStyle("position", pos);
8303             }
8304             if(zIndex){
8305                 this.setStyle("z-index", zIndex);
8306             }
8307             if(x !== undefined && y !== undefined){
8308                 this.setXY([x, y]);
8309             }else if(x !== undefined){
8310                 this.setX(x);
8311             }else if(y !== undefined){
8312                 this.setY(y);
8313             }
8314         },
8315
8316         /**
8317         * Clear positioning back to the default when the document was loaded
8318         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8319         * @return {Roo.Element} this
8320          */
8321         clearPositioning : function(value){
8322             value = value ||'';
8323             this.setStyle({
8324                 "left": value,
8325                 "right": value,
8326                 "top": value,
8327                 "bottom": value,
8328                 "z-index": "",
8329                 "position" : "static"
8330             });
8331             return this;
8332         },
8333
8334         /**
8335         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8336         * snapshot before performing an update and then restoring the element.
8337         * @return {Object}
8338         */
8339         getPositioning : function(){
8340             var l = this.getStyle("left");
8341             var t = this.getStyle("top");
8342             return {
8343                 "position" : this.getStyle("position"),
8344                 "left" : l,
8345                 "right" : l ? "" : this.getStyle("right"),
8346                 "top" : t,
8347                 "bottom" : t ? "" : this.getStyle("bottom"),
8348                 "z-index" : this.getStyle("z-index")
8349             };
8350         },
8351
8352         /**
8353          * Gets the width of the border(s) for the specified side(s)
8354          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8355          * passing lr would get the border (l)eft width + the border (r)ight width.
8356          * @return {Number} The width of the sides passed added together
8357          */
8358         getBorderWidth : function(side){
8359             return this.addStyles(side, El.borders);
8360         },
8361
8362         /**
8363          * Gets the width of the padding(s) for the specified side(s)
8364          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8365          * passing lr would get the padding (l)eft + the padding (r)ight.
8366          * @return {Number} The padding of the sides passed added together
8367          */
8368         getPadding : function(side){
8369             return this.addStyles(side, El.paddings);
8370         },
8371
8372         /**
8373         * Set positioning with an object returned by getPositioning().
8374         * @param {Object} posCfg
8375         * @return {Roo.Element} this
8376          */
8377         setPositioning : function(pc){
8378             this.applyStyles(pc);
8379             if(pc.right == "auto"){
8380                 this.dom.style.right = "";
8381             }
8382             if(pc.bottom == "auto"){
8383                 this.dom.style.bottom = "";
8384             }
8385             return this;
8386         },
8387
8388         // private
8389         fixDisplay : function(){
8390             if(this.getStyle("display") == "none"){
8391                 this.setStyle("visibility", "hidden");
8392                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8393                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8394                     this.setStyle("display", "block");
8395                 }
8396             }
8397         },
8398
8399         /**
8400          * Quick set left and top adding default units
8401          * @param {String} left The left CSS property value
8402          * @param {String} top The top CSS property value
8403          * @return {Roo.Element} this
8404          */
8405          setLeftTop : function(left, top){
8406             this.dom.style.left = this.addUnits(left);
8407             this.dom.style.top = this.addUnits(top);
8408             return this;
8409         },
8410
8411         /**
8412          * Move this element relative to its current position.
8413          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8414          * @param {Number} distance How far to move the element in pixels
8415          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8416          * @return {Roo.Element} this
8417          */
8418          move : function(direction, distance, animate){
8419             var xy = this.getXY();
8420             direction = direction.toLowerCase();
8421             switch(direction){
8422                 case "l":
8423                 case "left":
8424                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8425                     break;
8426                case "r":
8427                case "right":
8428                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8429                     break;
8430                case "t":
8431                case "top":
8432                case "up":
8433                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8434                     break;
8435                case "b":
8436                case "bottom":
8437                case "down":
8438                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8439                     break;
8440             }
8441             return this;
8442         },
8443
8444         /**
8445          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8446          * @return {Roo.Element} this
8447          */
8448         clip : function(){
8449             if(!this.isClipped){
8450                this.isClipped = true;
8451                this.originalClip = {
8452                    "o": this.getStyle("overflow"),
8453                    "x": this.getStyle("overflow-x"),
8454                    "y": this.getStyle("overflow-y")
8455                };
8456                this.setStyle("overflow", "hidden");
8457                this.setStyle("overflow-x", "hidden");
8458                this.setStyle("overflow-y", "hidden");
8459             }
8460             return this;
8461         },
8462
8463         /**
8464          *  Return clipping (overflow) to original clipping before clip() was called
8465          * @return {Roo.Element} this
8466          */
8467         unclip : function(){
8468             if(this.isClipped){
8469                 this.isClipped = false;
8470                 var o = this.originalClip;
8471                 if(o.o){this.setStyle("overflow", o.o);}
8472                 if(o.x){this.setStyle("overflow-x", o.x);}
8473                 if(o.y){this.setStyle("overflow-y", o.y);}
8474             }
8475             return this;
8476         },
8477
8478
8479         /**
8480          * Gets the x,y coordinates specified by the anchor position on the element.
8481          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8482          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8483          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8484          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8485          * @return {Array} [x, y] An array containing the element's x and y coordinates
8486          */
8487         getAnchorXY : function(anchor, local, s){
8488             //Passing a different size is useful for pre-calculating anchors,
8489             //especially for anchored animations that change the el size.
8490
8491             var w, h, vp = false;
8492             if(!s){
8493                 var d = this.dom;
8494                 if(d == document.body || d == document){
8495                     vp = true;
8496                     w = D.getViewWidth(); h = D.getViewHeight();
8497                 }else{
8498                     w = this.getWidth(); h = this.getHeight();
8499                 }
8500             }else{
8501                 w = s.width;  h = s.height;
8502             }
8503             var x = 0, y = 0, r = Math.round;
8504             switch((anchor || "tl").toLowerCase()){
8505                 case "c":
8506                     x = r(w*.5);
8507                     y = r(h*.5);
8508                 break;
8509                 case "t":
8510                     x = r(w*.5);
8511                     y = 0;
8512                 break;
8513                 case "l":
8514                     x = 0;
8515                     y = r(h*.5);
8516                 break;
8517                 case "r":
8518                     x = w;
8519                     y = r(h*.5);
8520                 break;
8521                 case "b":
8522                     x = r(w*.5);
8523                     y = h;
8524                 break;
8525                 case "tl":
8526                     x = 0;
8527                     y = 0;
8528                 break;
8529                 case "bl":
8530                     x = 0;
8531                     y = h;
8532                 break;
8533                 case "br":
8534                     x = w;
8535                     y = h;
8536                 break;
8537                 case "tr":
8538                     x = w;
8539                     y = 0;
8540                 break;
8541             }
8542             if(local === true){
8543                 return [x, y];
8544             }
8545             if(vp){
8546                 var sc = this.getScroll();
8547                 return [x + sc.left, y + sc.top];
8548             }
8549             //Add the element's offset xy
8550             var o = this.getXY();
8551             return [x+o[0], y+o[1]];
8552         },
8553
8554         /**
8555          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8556          * supported position values.
8557          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8558          * @param {String} position The position to align to.
8559          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8560          * @return {Array} [x, y]
8561          */
8562         getAlignToXY : function(el, p, o){
8563             el = Roo.get(el);
8564             var d = this.dom;
8565             if(!el.dom){
8566                 throw "Element.alignTo with an element that doesn't exist";
8567             }
8568             var c = false; //constrain to viewport
8569             var p1 = "", p2 = "";
8570             o = o || [0,0];
8571
8572             if(!p){
8573                 p = "tl-bl";
8574             }else if(p == "?"){
8575                 p = "tl-bl?";
8576             }else if(p.indexOf("-") == -1){
8577                 p = "tl-" + p;
8578             }
8579             p = p.toLowerCase();
8580             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8581             if(!m){
8582                throw "Element.alignTo with an invalid alignment " + p;
8583             }
8584             p1 = m[1]; p2 = m[2]; c = !!m[3];
8585
8586             //Subtract the aligned el's internal xy from the target's offset xy
8587             //plus custom offset to get the aligned el's new offset xy
8588             var a1 = this.getAnchorXY(p1, true);
8589             var a2 = el.getAnchorXY(p2, false);
8590             var x = a2[0] - a1[0] + o[0];
8591             var y = a2[1] - a1[1] + o[1];
8592             if(c){
8593                 //constrain the aligned el to viewport if necessary
8594                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8595                 // 5px of margin for ie
8596                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8597
8598                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8599                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8600                 //otherwise swap the aligned el to the opposite border of the target.
8601                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8602                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8603                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8604                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8605
8606                var doc = document;
8607                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8608                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8609
8610                if((x+w) > dw + scrollX){
8611                     x = swapX ? r.left-w : dw+scrollX-w;
8612                 }
8613                if(x < scrollX){
8614                    x = swapX ? r.right : scrollX;
8615                }
8616                if((y+h) > dh + scrollY){
8617                     y = swapY ? r.top-h : dh+scrollY-h;
8618                 }
8619                if (y < scrollY){
8620                    y = swapY ? r.bottom : scrollY;
8621                }
8622             }
8623             return [x,y];
8624         },
8625
8626         // private
8627         getConstrainToXY : function(){
8628             var os = {top:0, left:0, bottom:0, right: 0};
8629
8630             return function(el, local, offsets, proposedXY){
8631                 el = Roo.get(el);
8632                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8633
8634                 var vw, vh, vx = 0, vy = 0;
8635                 if(el.dom == document.body || el.dom == document){
8636                     vw = Roo.lib.Dom.getViewWidth();
8637                     vh = Roo.lib.Dom.getViewHeight();
8638                 }else{
8639                     vw = el.dom.clientWidth;
8640                     vh = el.dom.clientHeight;
8641                     if(!local){
8642                         var vxy = el.getXY();
8643                         vx = vxy[0];
8644                         vy = vxy[1];
8645                     }
8646                 }
8647
8648                 var s = el.getScroll();
8649
8650                 vx += offsets.left + s.left;
8651                 vy += offsets.top + s.top;
8652
8653                 vw -= offsets.right;
8654                 vh -= offsets.bottom;
8655
8656                 var vr = vx+vw;
8657                 var vb = vy+vh;
8658
8659                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8660                 var x = xy[0], y = xy[1];
8661                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8662
8663                 // only move it if it needs it
8664                 var moved = false;
8665
8666                 // first validate right/bottom
8667                 if((x + w) > vr){
8668                     x = vr - w;
8669                     moved = true;
8670                 }
8671                 if((y + h) > vb){
8672                     y = vb - h;
8673                     moved = true;
8674                 }
8675                 // then make sure top/left isn't negative
8676                 if(x < vx){
8677                     x = vx;
8678                     moved = true;
8679                 }
8680                 if(y < vy){
8681                     y = vy;
8682                     moved = true;
8683                 }
8684                 return moved ? [x, y] : false;
8685             };
8686         }(),
8687
8688         // private
8689         adjustForConstraints : function(xy, parent, offsets){
8690             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8691         },
8692
8693         /**
8694          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8695          * document it aligns it to the viewport.
8696          * The position parameter is optional, and can be specified in any one of the following formats:
8697          * <ul>
8698          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8699          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8700          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8701          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8702          *   <li><b>Two anchors</b>: If two values from the table below are passed separated by a dash, the first value is used as the
8703          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8704          * </ul>
8705          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8706          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8707          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8708          * that specified in order to enforce the viewport constraints.
8709          * Following are all of the supported anchor positions:
8710     <pre>
8711     Value  Description
8712     -----  -----------------------------
8713     tl     The top left corner (default)
8714     t      The center of the top edge
8715     tr     The top right corner
8716     l      The center of the left edge
8717     c      In the center of the element
8718     r      The center of the right edge
8719     bl     The bottom left corner
8720     b      The center of the bottom edge
8721     br     The bottom right corner
8722     </pre>
8723     Example Usage:
8724     <pre><code>
8725     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8726     el.alignTo("other-el");
8727
8728     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8729     el.alignTo("other-el", "tr?");
8730
8731     // align the bottom right corner of el with the center left edge of other-el
8732     el.alignTo("other-el", "br-l?");
8733
8734     // align the center of el with the bottom left corner of other-el and
8735     // adjust the x position by -6 pixels (and the y position by 0)
8736     el.alignTo("other-el", "c-bl", [-6, 0]);
8737     </code></pre>
8738          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8739          * @param {String} position The position to align to.
8740          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8741          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8742          * @return {Roo.Element} this
8743          */
8744         alignTo : function(element, position, offsets, animate){
8745             var xy = this.getAlignToXY(element, position, offsets);
8746             this.setXY(xy, this.preanim(arguments, 3));
8747             return this;
8748         },
8749
8750         /**
8751          * Anchors an element to another element and realigns it when the window is resized.
8752          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8753          * @param {String} position The position to align to.
8754          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8755          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8756          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8757          * is a number, it is used as the buffer delay (defaults to 50ms).
8758          * @param {Function} callback The function to call after the animation finishes
8759          * @return {Roo.Element} this
8760          */
8761         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8762             var action = function(){
8763                 this.alignTo(el, alignment, offsets, animate);
8764                 Roo.callback(callback, this);
8765             };
8766             Roo.EventManager.onWindowResize(action, this);
8767             var tm = typeof monitorScroll;
8768             if(tm != 'undefined'){
8769                 Roo.EventManager.on(window, 'scroll', action, this,
8770                     {buffer: tm == 'number' ? monitorScroll : 50});
8771             }
8772             action.call(this); // align immediately
8773             return this;
8774         },
8775         /**
8776          * Clears any opacity settings from this element. Required in some cases for IE.
8777          * @return {Roo.Element} this
8778          */
8779         clearOpacity : function(){
8780             if (window.ActiveXObject) {
8781                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8782                     this.dom.style.filter = "";
8783                 }
8784             } else {
8785                 this.dom.style.opacity = "";
8786                 this.dom.style["-moz-opacity"] = "";
8787                 this.dom.style["-khtml-opacity"] = "";
8788             }
8789             return this;
8790         },
8791
8792         /**
8793          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8794          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8795          * @return {Roo.Element} this
8796          */
8797         hide : function(animate){
8798             this.setVisible(false, this.preanim(arguments, 0));
8799             return this;
8800         },
8801
8802         /**
8803         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8804         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8805          * @return {Roo.Element} this
8806          */
8807         show : function(animate){
8808             this.setVisible(true, this.preanim(arguments, 0));
8809             return this;
8810         },
8811
8812         /**
8813          * @private Test if size has a unit, otherwise appends the default
8814          */
8815         addUnits : function(size){
8816             return Roo.Element.addUnits(size, this.defaultUnit);
8817         },
8818
8819         /**
8820          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8821          * @return {Roo.Element} this
8822          */
8823         beginMeasure : function(){
8824             var el = this.dom;
8825             if(el.offsetWidth || el.offsetHeight){
8826                 return this; // offsets work already
8827             }
8828             var changed = [];
8829             var p = this.dom, b = document.body; // start with this element
8830             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8831                 var pe = Roo.get(p);
8832                 if(pe.getStyle('display') == 'none'){
8833                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8834                     p.style.visibility = "hidden";
8835                     p.style.display = "block";
8836                 }
8837                 p = p.parentNode;
8838             }
8839             this._measureChanged = changed;
8840             return this;
8841
8842         },
8843
8844         /**
8845          * Restores displays to before beginMeasure was called
8846          * @return {Roo.Element} this
8847          */
8848         endMeasure : function(){
8849             var changed = this._measureChanged;
8850             if(changed){
8851                 for(var i = 0, len = changed.length; i < len; i++) {
8852                     var r = changed[i];
8853                     r.el.style.visibility = r.visibility;
8854                     r.el.style.display = "none";
8855                 }
8856                 this._measureChanged = null;
8857             }
8858             return this;
8859         },
8860
8861         /**
8862         * Update the innerHTML of this element, optionally searching for and processing scripts
8863         * @param {String} html The new HTML
8864         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8865         * @param {Function} callback For async script loading you can be noticed when the update completes
8866         * @return {Roo.Element} this
8867          */
8868         update : function(html, loadScripts, callback){
8869             if(typeof html == "undefined"){
8870                 html = "";
8871             }
8872             if(loadScripts !== true){
8873                 this.dom.innerHTML = html;
8874                 if(typeof callback == "function"){
8875                     callback();
8876                 }
8877                 return this;
8878             }
8879             var id = Roo.id();
8880             var dom = this.dom;
8881
8882             html += '<span id="' + id + '"></span>';
8883
8884             E.onAvailable(id, function(){
8885                 var hd = document.getElementsByTagName("head")[0];
8886                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8887                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8888                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8889
8890                 var match;
8891                 while(match = re.exec(html)){
8892                     var attrs = match[1];
8893                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8894                     if(srcMatch && srcMatch[2]){
8895                        var s = document.createElement("script");
8896                        s.src = srcMatch[2];
8897                        var typeMatch = attrs.match(typeRe);
8898                        if(typeMatch && typeMatch[2]){
8899                            s.type = typeMatch[2];
8900                        }
8901                        hd.appendChild(s);
8902                     }else if(match[2] && match[2].length > 0){
8903                         if(window.execScript) {
8904                            window.execScript(match[2]);
8905                         } else {
8906                             /**
8907                              * eval:var:id
8908                              * eval:var:dom
8909                              * eval:var:html
8910                              * 
8911                              */
8912                            window.eval(match[2]);
8913                         }
8914                     }
8915                 }
8916                 var el = document.getElementById(id);
8917                 if(el){el.parentNode.removeChild(el);}
8918                 if(typeof callback == "function"){
8919                     callback();
8920                 }
8921             });
8922             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8923             return this;
8924         },
8925
8926         /**
8927          * Direct access to the UpdateManager update() method (takes the same parameters).
8928          * @param {String/Function} url The url for this request or a function to call to get the url
8929          * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
8930          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8931          * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
8932          * @return {Roo.Element} this
8933          */
8934         load : function(){
8935             var um = this.getUpdateManager();
8936             um.update.apply(um, arguments);
8937             return this;
8938         },
8939
8940         /**
8941         * Gets this element's UpdateManager
8942         * @return {Roo.UpdateManager} The UpdateManager
8943         */
8944         getUpdateManager : function(){
8945             if(!this.updateManager){
8946                 this.updateManager = new Roo.UpdateManager(this);
8947             }
8948             return this.updateManager;
8949         },
8950
8951         /**
8952          * Disables text selection for this element (normalized across browsers)
8953          * @return {Roo.Element} this
8954          */
8955         unselectable : function(){
8956             this.dom.unselectable = "on";
8957             this.swallowEvent("selectstart", true);
8958             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8959             this.addClass("x-unselectable");
8960             return this;
8961         },
8962
8963         /**
8964         * Calculates the x, y to center this element on the screen
8965         * @return {Array} The x, y values [x, y]
8966         */
8967         getCenterXY : function(){
8968             return this.getAlignToXY(document, 'c-c');
8969         },
8970
8971         /**
8972         * Centers the Element in either the viewport, or another Element.
8973         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8974         */
8975         center : function(centerIn){
8976             this.alignTo(centerIn || document, 'c-c');
8977             return this;
8978         },
8979
8980         /**
8981          * Tests various css rules/browsers to determine if this element uses a border box
8982          * @return {Boolean}
8983          */
8984         isBorderBox : function(){
8985             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8986         },
8987
8988         /**
8989          * Return a box {x, y, width, height} that can be used to set another elements
8990          * size/location to match this element.
8991          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8992          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8993          * @return {Object} box An object in the format {x, y, width, height}
8994          */
8995         getBox : function(contentBox, local){
8996             var xy;
8997             if(!local){
8998                 xy = this.getXY();
8999             }else{
9000                 var left = parseInt(this.getStyle("left"), 10) || 0;
9001                 var top = parseInt(this.getStyle("top"), 10) || 0;
9002                 xy = [left, top];
9003             }
9004             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9005             if(!contentBox){
9006                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9007             }else{
9008                 var l = this.getBorderWidth("l")+this.getPadding("l");
9009                 var r = this.getBorderWidth("r")+this.getPadding("r");
9010                 var t = this.getBorderWidth("t")+this.getPadding("t");
9011                 var b = this.getBorderWidth("b")+this.getPadding("b");
9012                 bx = {x: xy[0]+l, y: xy[1]+t, 0: xy[0]+l, 1: xy[1]+t, width: w-(l+r), height: h-(t+b)};
9013             }
9014             bx.right = bx.x + bx.width;
9015             bx.bottom = bx.y + bx.height;
9016             return bx;
9017         },
9018
9019         /**
9020          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9021          for more information about the sides.
9022          * @param {String} sides
9023          * @return {Number}
9024          */
9025         getFrameWidth : function(sides, onlyContentBox){
9026             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9027         },
9028
9029         /**
9030          * Sets the element's box. Use getBox() on another element to get a box obj. If animate is true then width, height, x and y will be animated concurrently.
9031          * @param {Object} box The box to fill {x, y, width, height}
9032          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9033          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9034          * @return {Roo.Element} this
9035          */
9036         setBox : function(box, adjust, animate){
9037             var w = box.width, h = box.height;
9038             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9039                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9040                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9041             }
9042             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9043             return this;
9044         },
9045
9046         /**
9047          * Forces the browser to repaint this element
9048          * @return {Roo.Element} this
9049          */
9050          repaint : function(){
9051             var dom = this.dom;
9052             this.addClass("x-repaint");
9053             setTimeout(function(){
9054                 Roo.get(dom).removeClass("x-repaint");
9055             }, 1);
9056             return this;
9057         },
9058
9059         /**
9060          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9061          * then it returns the calculated width of the sides (see getPadding)
9062          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9063          * @return {Object/Number}
9064          */
9065         getMargins : function(side){
9066             if(!side){
9067                 return {
9068                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9069                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9070                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9071                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9072                 };
9073             }else{
9074                 return this.addStyles(side, El.margins);
9075              }
9076         },
9077
9078         // private
9079         addStyles : function(sides, styles){
9080             var val = 0, v, w;
9081             for(var i = 0, len = sides.length; i < len; i++){
9082                 v = this.getStyle(styles[sides.charAt(i)]);
9083                 if(v){
9084                      w = parseInt(v, 10);
9085                      if(w){ val += w; }
9086                 }
9087             }
9088             return val;
9089         },
9090
9091         /**
9092          * Creates a proxy element of this element
9093          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9094          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9095          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9096          * @return {Roo.Element} The new proxy element
9097          */
9098         createProxy : function(config, renderTo, matchBox){
9099             if(renderTo){
9100                 renderTo = Roo.getDom(renderTo);
9101             }else{
9102                 renderTo = document.body;
9103             }
9104             config = typeof config == "object" ?
9105                 config : {tag : "div", cls: config};
9106             var proxy = Roo.DomHelper.append(renderTo, config, true);
9107             if(matchBox){
9108                proxy.setBox(this.getBox());
9109             }
9110             return proxy;
9111         },
9112
9113         /**
9114          * Puts a mask over this element to disable user interaction. Requires core.css.
9115          * This method can only be applied to elements which accept child nodes.
9116          * @param {String} msg (optional) A message to display in the mask
9117          * @param {String} msgCls (optional) A css class to apply to the msg element
9118          * @return {Element} The mask  element
9119          */
9120         mask : function(msg, msgCls)
9121         {
9122             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9123                 this.setStyle("position", "relative");
9124             }
9125             if(!this._mask){
9126                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9127             }
9128             
9129             this.addClass("x-masked");
9130             this._mask.setDisplayed(true);
9131             
9132             // we wander
9133             var z = 0;
9134             var dom = this.dom;
9135             while (dom && dom.style) {
9136                 if (!isNaN(parseInt(dom.style.zIndex))) {
9137                     z = Math.max(z, parseInt(dom.style.zIndex));
9138                 }
9139                 dom = dom.parentNode;
9140             }
9141             // if we are masking the body - then it hides everything..
9142             if (this.dom == document.body) {
9143                 z = 1000000;
9144                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9145                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9146             }
9147            
9148             if(typeof msg == 'string'){
9149                 if(!this._maskMsg){
9150                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9151                         cls: "roo-el-mask-msg", 
9152                         cn: [
9153                             {
9154                                 tag: 'i',
9155                                 cls: 'fa fa-spinner fa-spin'
9156                             },
9157                             {
9158                                 tag: 'div'
9159                             }   
9160                         ]
9161                     }, true);
9162                 }
9163                 var mm = this._maskMsg;
9164                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9165                 if (mm.dom.lastChild) { // weird IE issue?
9166                     mm.dom.lastChild.innerHTML = msg;
9167                 }
9168                 mm.setDisplayed(true);
9169                 mm.center(this);
9170                 mm.setStyle('z-index', z + 102);
9171             }
9172             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9173                 this._mask.setHeight(this.getHeight());
9174             }
9175             this._mask.setStyle('z-index', z + 100);
9176             
9177             return this._mask;
9178         },
9179
9180         /**
9181          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9182          * it is cached for reuse.
9183          */
9184         unmask : function(removeEl){
9185             if(this._mask){
9186                 if(removeEl === true){
9187                     this._mask.remove();
9188                     delete this._mask;
9189                     if(this._maskMsg){
9190                         this._maskMsg.remove();
9191                         delete this._maskMsg;
9192                     }
9193                 }else{
9194                     this._mask.setDisplayed(false);
9195                     if(this._maskMsg){
9196                         this._maskMsg.setDisplayed(false);
9197                     }
9198                 }
9199             }
9200             this.removeClass("x-masked");
9201         },
9202
9203         /**
9204          * Returns true if this element is masked
9205          * @return {Boolean}
9206          */
9207         isMasked : function(){
9208             return this._mask && this._mask.isVisible();
9209         },
9210
9211         /**
9212          * Creates an iframe shim for this element to keep selects and other windowed objects from
9213          * showing through.
9214          * @return {Roo.Element} The new shim element
9215          */
9216         createShim : function(){
9217             var el = document.createElement('iframe');
9218             el.frameBorder = 'no';
9219             el.className = 'roo-shim';
9220             if(Roo.isIE && Roo.isSecure){
9221                 el.src = Roo.SSL_SECURE_URL;
9222             }
9223             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9224             shim.autoBoxAdjust = false;
9225             return shim;
9226         },
9227
9228         /**
9229          * Removes this element from the DOM and deletes it from the cache
9230          */
9231         remove : function(){
9232             if(this.dom.parentNode){
9233                 this.dom.parentNode.removeChild(this.dom);
9234             }
9235             delete El.cache[this.dom.id];
9236         },
9237
9238         /**
9239          * Sets up event handlers to add and remove a css class when the mouse is over this element
9240          * @param {String} className
9241          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9242          * mouseout events for children elements
9243          * @return {Roo.Element} this
9244          */
9245         addClassOnOver : function(className, preventFlicker){
9246             this.on("mouseover", function(){
9247                 Roo.fly(this, '_internal').addClass(className);
9248             }, this.dom);
9249             var removeFn = function(e){
9250                 if(preventFlicker !== true || !e.within(this, true)){
9251                     Roo.fly(this, '_internal').removeClass(className);
9252                 }
9253             };
9254             this.on("mouseout", removeFn, this.dom);
9255             return this;
9256         },
9257
9258         /**
9259          * Sets up event handlers to add and remove a css class when this element has the focus
9260          * @param {String} className
9261          * @return {Roo.Element} this
9262          */
9263         addClassOnFocus : function(className){
9264             this.on("focus", function(){
9265                 Roo.fly(this, '_internal').addClass(className);
9266             }, this.dom);
9267             this.on("blur", function(){
9268                 Roo.fly(this, '_internal').removeClass(className);
9269             }, this.dom);
9270             return this;
9271         },
9272         /**
9273          * Sets up event handlers to add and remove a css class when the mouse is down and then up on this element (a click effect)
9274          * @param {String} className
9275          * @return {Roo.Element} this
9276          */
9277         addClassOnClick : function(className){
9278             var dom = this.dom;
9279             this.on("mousedown", function(){
9280                 Roo.fly(dom, '_internal').addClass(className);
9281                 var d = Roo.get(document);
9282                 var fn = function(){
9283                     Roo.fly(dom, '_internal').removeClass(className);
9284                     d.removeListener("mouseup", fn);
9285                 };
9286                 d.on("mouseup", fn);
9287             });
9288             return this;
9289         },
9290
9291         /**
9292          * Stops the specified event from bubbling and optionally prevents the default action
9293          * @param {String} eventName
9294          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9295          * @return {Roo.Element} this
9296          */
9297         swallowEvent : function(eventName, preventDefault){
9298             var fn = function(e){
9299                 e.stopPropagation();
9300                 if(preventDefault){
9301                     e.preventDefault();
9302                 }
9303             };
9304             if(eventName instanceof Array){
9305                 for(var i = 0, len = eventName.length; i < len; i++){
9306                      this.on(eventName[i], fn);
9307                 }
9308                 return this;
9309             }
9310             this.on(eventName, fn);
9311             return this;
9312         },
9313
9314         /**
9315          * @private
9316          */
9317       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9318
9319         /**
9320          * Sizes this element to its parent element's dimensions performing
9321          * neccessary box adjustments.
9322          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9323          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9324          * @return {Roo.Element} this
9325          */
9326         fitToParent : function(monitorResize, targetParent) {
9327           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9328           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9329           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9330             return;
9331           }
9332           var p = Roo.get(targetParent || this.dom.parentNode);
9333           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9334           if (monitorResize === true) {
9335             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9336             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9337           }
9338           return this;
9339         },
9340
9341         /**
9342          * Gets the next sibling, skipping text nodes
9343          * @return {HTMLElement} The next sibling or null
9344          */
9345         getNextSibling : function(){
9346             var n = this.dom.nextSibling;
9347             while(n && n.nodeType != 1){
9348                 n = n.nextSibling;
9349             }
9350             return n;
9351         },
9352
9353         /**
9354          * Gets the previous sibling, skipping text nodes
9355          * @return {HTMLElement} The previous sibling or null
9356          */
9357         getPrevSibling : function(){
9358             var n = this.dom.previousSibling;
9359             while(n && n.nodeType != 1){
9360                 n = n.previousSibling;
9361             }
9362             return n;
9363         },
9364
9365
9366         /**
9367          * Appends the passed element(s) to this element
9368          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9369          * @return {Roo.Element} this
9370          */
9371         appendChild: function(el){
9372             el = Roo.get(el);
9373             el.appendTo(this);
9374             return this;
9375         },
9376
9377         /**
9378          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9379          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9380          * automatically generated with the specified attributes.
9381          * @param {HTMLElement} insertBefore (optional) a child element of this element
9382          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9383          * @return {Roo.Element} The new child element
9384          */
9385         createChild: function(config, insertBefore, returnDom){
9386             config = config || {tag:'div'};
9387             if(insertBefore){
9388                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9389             }
9390             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9391         },
9392
9393         /**
9394          * Appends this element to the passed element
9395          * @param {String/HTMLElement/Element} el The new parent element
9396          * @return {Roo.Element} this
9397          */
9398         appendTo: function(el){
9399             el = Roo.getDom(el);
9400             el.appendChild(this.dom);
9401             return this;
9402         },
9403
9404         /**
9405          * Inserts this element before the passed element in the DOM
9406          * @param {String/HTMLElement/Element} el The element to insert before
9407          * @return {Roo.Element} this
9408          */
9409         insertBefore: function(el){
9410             el = Roo.getDom(el);
9411             el.parentNode.insertBefore(this.dom, el);
9412             return this;
9413         },
9414
9415         /**
9416          * Inserts this element after the passed element in the DOM
9417          * @param {String/HTMLElement/Element} el The element to insert after
9418          * @return {Roo.Element} this
9419          */
9420         insertAfter: function(el){
9421             el = Roo.getDom(el);
9422             el.parentNode.insertBefore(this.dom, el.nextSibling);
9423             return this;
9424         },
9425
9426         /**
9427          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9428          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9429          * @return {Roo.Element} The new child
9430          */
9431         insertFirst: function(el, returnDom){
9432             el = el || {};
9433             if(typeof el == 'object' && !el.nodeType){ // dh config
9434                 return this.createChild(el, this.dom.firstChild, returnDom);
9435             }else{
9436                 el = Roo.getDom(el);
9437                 this.dom.insertBefore(el, this.dom.firstChild);
9438                 return !returnDom ? Roo.get(el) : el;
9439             }
9440         },
9441
9442         /**
9443          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9444          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9445          * @param {String} where (optional) 'before' or 'after' defaults to before
9446          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9447          * @return {Roo.Element} the inserted Element
9448          */
9449         insertSibling: function(el, where, returnDom){
9450             where = where ? where.toLowerCase() : 'before';
9451             el = el || {};
9452             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9453
9454             if(typeof el == 'object' && !el.nodeType){ // dh config
9455                 if(where == 'after' && !this.dom.nextSibling){
9456                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9457                 }else{
9458                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9459                 }
9460
9461             }else{
9462                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9463                             where == 'before' ? this.dom : this.dom.nextSibling);
9464                 if(!returnDom){
9465                     rt = Roo.get(rt);
9466                 }
9467             }
9468             return rt;
9469         },
9470
9471         /**
9472          * Creates and wraps this element with another element
9473          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9474          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9475          * @return {HTMLElement/Element} The newly created wrapper element
9476          */
9477         wrap: function(config, returnDom){
9478             if(!config){
9479                 config = {tag: "div"};
9480             }
9481             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9482             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9483             return newEl;
9484         },
9485
9486         /**
9487          * Replaces the passed element with this element
9488          * @param {String/HTMLElement/Element} el The element to replace
9489          * @return {Roo.Element} this
9490          */
9491         replace: function(el){
9492             el = Roo.get(el);
9493             this.insertBefore(el);
9494             el.remove();
9495             return this;
9496         },
9497
9498         /**
9499          * Inserts an html fragment into this element
9500          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9501          * @param {String} html The HTML fragment
9502          * @param {Boolean} returnEl True to return an Roo.Element
9503          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9504          */
9505         insertHtml : function(where, html, returnEl){
9506             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9507             return returnEl ? Roo.get(el) : el;
9508         },
9509
9510         /**
9511          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9512          * @param {Object} o The object with the attributes
9513          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9514          * @return {Roo.Element} this
9515          */
9516         set : function(o, useSet){
9517             var el = this.dom;
9518             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9519             for(var attr in o){
9520                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9521                 if(attr=="cls"){
9522                     el.className = o["cls"];
9523                 }else{
9524                     if(useSet) {
9525                         el.setAttribute(attr, o[attr]);
9526                     } else {
9527                         el[attr] = o[attr];
9528                     }
9529                 }
9530             }
9531             if(o.style){
9532                 Roo.DomHelper.applyStyles(el, o.style);
9533             }
9534             return this;
9535         },
9536
9537         /**
9538          * Convenience method for constructing a KeyMap
9539          * @param {Number/Array/Object/String} key Either a string with the keys to listen for, the numeric key code, array of key codes or an object with the following options:
9540          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9541          * @param {Function} fn The function to call
9542          * @param {Object} scope (optional) The scope of the function
9543          * @return {Roo.KeyMap} The KeyMap created
9544          */
9545         addKeyListener : function(key, fn, scope){
9546             var config;
9547             if(typeof key != "object" || key instanceof Array){
9548                 config = {
9549                     key: key,
9550                     fn: fn,
9551                     scope: scope
9552                 };
9553             }else{
9554                 config = {
9555                     key : key.key,
9556                     shift : key.shift,
9557                     ctrl : key.ctrl,
9558                     alt : key.alt,
9559                     fn: fn,
9560                     scope: scope
9561                 };
9562             }
9563             return new Roo.KeyMap(this, config);
9564         },
9565
9566         /**
9567          * Creates a KeyMap for this element
9568          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9569          * @return {Roo.KeyMap} The KeyMap created
9570          */
9571         addKeyMap : function(config){
9572             return new Roo.KeyMap(this, config);
9573         },
9574
9575         /**
9576          * Returns true if this element is scrollable.
9577          * @return {Boolean}
9578          */
9579          isScrollable : function(){
9580             var dom = this.dom;
9581             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9582         },
9583
9584         /**
9585          * Scrolls this element the specified scroll point. It does NOT do bounds checking so if you scroll to a weird value it will try to do it. For auto bounds checking, use scroll().
9586          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9587          * @param {Number} value The new scroll value
9588          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9589          * @return {Element} this
9590          */
9591
9592         scrollTo : function(side, value, animate){
9593             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9594             if(!animate || !A){
9595                 this.dom[prop] = value;
9596             }else{
9597                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9598                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9599             }
9600             return this;
9601         },
9602
9603         /**
9604          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9605          * within this element's scrollable range.
9606          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9607          * @param {Number} distance How far to scroll the element in pixels
9608          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9609          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9610          * was scrolled as far as it could go.
9611          */
9612          scroll : function(direction, distance, animate){
9613              if(!this.isScrollable()){
9614                  return;
9615              }
9616              var el = this.dom;
9617              var l = el.scrollLeft, t = el.scrollTop;
9618              var w = el.scrollWidth, h = el.scrollHeight;
9619              var cw = el.clientWidth, ch = el.clientHeight;
9620              direction = direction.toLowerCase();
9621              var scrolled = false;
9622              var a = this.preanim(arguments, 2);
9623              switch(direction){
9624                  case "l":
9625                  case "left":
9626                      if(w - l > cw){
9627                          var v = Math.min(l + distance, w-cw);
9628                          this.scrollTo("left", v, a);
9629                          scrolled = true;
9630                      }
9631                      break;
9632                 case "r":
9633                 case "right":
9634                      if(l > 0){
9635                          var v = Math.max(l - distance, 0);
9636                          this.scrollTo("left", v, a);
9637                          scrolled = true;
9638                      }
9639                      break;
9640                 case "t":
9641                 case "top":
9642                 case "up":
9643                      if(t > 0){
9644                          var v = Math.max(t - distance, 0);
9645                          this.scrollTo("top", v, a);
9646                          scrolled = true;
9647                      }
9648                      break;
9649                 case "b":
9650                 case "bottom":
9651                 case "down":
9652                      if(h - t > ch){
9653                          var v = Math.min(t + distance, h-ch);
9654                          this.scrollTo("top", v, a);
9655                          scrolled = true;
9656                      }
9657                      break;
9658              }
9659              return scrolled;
9660         },
9661
9662         /**
9663          * Translates the passed page coordinates into left/top css values for this element
9664          * @param {Number/Array} x The page x or an array containing [x, y]
9665          * @param {Number} y The page y
9666          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9667          */
9668         translatePoints : function(x, y){
9669             if(typeof x == 'object' || x instanceof Array){
9670                 y = x[1]; x = x[0];
9671             }
9672             var p = this.getStyle('position');
9673             var o = this.getXY();
9674
9675             var l = parseInt(this.getStyle('left'), 10);
9676             var t = parseInt(this.getStyle('top'), 10);
9677
9678             if(isNaN(l)){
9679                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9680             }
9681             if(isNaN(t)){
9682                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9683             }
9684
9685             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9686         },
9687
9688         /**
9689          * Returns the current scroll position of the element.
9690          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9691          */
9692         getScroll : function(){
9693             var d = this.dom, doc = document;
9694             if(d == doc || d == doc.body){
9695                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9696                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9697                 return {left: l, top: t};
9698             }else{
9699                 return {left: d.scrollLeft, top: d.scrollTop};
9700             }
9701         },
9702
9703         /**
9704          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9705          * are convert to standard 6 digit hex color.
9706          * @param {String} attr The css attribute
9707          * @param {String} defaultValue The default value to use when a valid color isn't found
9708          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9709          * YUI color anims.
9710          */
9711         getColor : function(attr, defaultValue, prefix){
9712             var v = this.getStyle(attr);
9713             if(!v || v == "transparent" || v == "inherit") {
9714                 return defaultValue;
9715             }
9716             var color = typeof prefix == "undefined" ? "#" : prefix;
9717             if(v.substr(0, 4) == "rgb("){
9718                 var rvs = v.slice(4, v.length -1).split(",");
9719                 for(var i = 0; i < 3; i++){
9720                     var h = parseInt(rvs[i]).toString(16);
9721                     if(h < 16){
9722                         h = "0" + h;
9723                     }
9724                     color += h;
9725                 }
9726             } else {
9727                 if(v.substr(0, 1) == "#"){
9728                     if(v.length == 4) {
9729                         for(var i = 1; i < 4; i++){
9730                             var c = v.charAt(i);
9731                             color +=  c + c;
9732                         }
9733                     }else if(v.length == 7){
9734                         color += v.substr(1);
9735                     }
9736                 }
9737             }
9738             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9739         },
9740
9741         /**
9742          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9743          * gradient background, rounded corners and a 4-way shadow.
9744          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9745          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9746          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9747          * @return {Roo.Element} this
9748          */
9749         boxWrap : function(cls){
9750             cls = cls || 'x-box';
9751             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9752             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9753             return el;
9754         },
9755
9756         /**
9757          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9758          * @param {String} namespace The namespace in which to look for the attribute
9759          * @param {String} name The attribute name
9760          * @return {String} The attribute value
9761          */
9762         getAttributeNS : Roo.isIE ? function(ns, name){
9763             var d = this.dom;
9764             var type = typeof d[ns+":"+name];
9765             if(type != 'undefined' && type != 'unknown'){
9766                 return d[ns+":"+name];
9767             }
9768             return d[name];
9769         } : function(ns, name){
9770             var d = this.dom;
9771             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9772         },
9773         
9774         
9775         /**
9776          * Sets or Returns the value the dom attribute value
9777          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9778          * @param {String} value (optional) The value to set the attribute to
9779          * @return {String} The attribute value
9780          */
9781         attr : function(name){
9782             if (arguments.length > 1) {
9783                 this.dom.setAttribute(name, arguments[1]);
9784                 return arguments[1];
9785             }
9786             if (typeof(name) == 'object') {
9787                 for(var i in name) {
9788                     this.attr(i, name[i]);
9789                 }
9790                 return name;
9791             }
9792             
9793             
9794             if (!this.dom.hasAttribute(name)) {
9795                 return undefined;
9796             }
9797             return this.dom.getAttribute(name);
9798         }
9799         
9800         
9801         
9802     };
9803
9804     var ep = El.prototype;
9805
9806     /**
9807      * Appends an event handler (Shorthand for addListener)
9808      * @param {String}   eventName     The type of event to append
9809      * @param {Function} fn        The method the event invokes
9810      * @param {Object} scope       (optional) The scope (this object) of the fn
9811      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9812      * @method
9813      */
9814     ep.on = ep.addListener;
9815         // backwards compat
9816     ep.mon = ep.addListener;
9817
9818     /**
9819      * Removes an event handler from this element (shorthand for removeListener)
9820      * @param {String} eventName the type of event to remove
9821      * @param {Function} fn the method the event invokes
9822      * @return {Roo.Element} this
9823      * @method
9824      */
9825     ep.un = ep.removeListener;
9826
9827     /**
9828      * true to automatically adjust width and height settings for box-model issues (default to true)
9829      */
9830     ep.autoBoxAdjust = true;
9831
9832     // private
9833     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9834
9835     // private
9836     El.addUnits = function(v, defaultUnit){
9837         if(v === "" || v == "auto"){
9838             return v;
9839         }
9840         if(v === undefined){
9841             return '';
9842         }
9843         if(typeof v == "number" || !El.unitPattern.test(v)){
9844             return v + (defaultUnit || 'px');
9845         }
9846         return v;
9847     };
9848
9849     // special markup used throughout Roo when box wrapping elements
9850     El.boxMarkup = '<div class="{0}-tl"><div class="{0}-tr"><div class="{0}-tc"></div></div></div><div class="{0}-ml"><div class="{0}-mr"><div class="{0}-mc"></div></div></div><div class="{0}-bl"><div class="{0}-br"><div class="{0}-bc"></div></div></div>';
9851     /**
9852      * Visibility mode constant - Use visibility to hide element
9853      * @static
9854      * @type Number
9855      */
9856     El.VISIBILITY = 1;
9857     /**
9858      * Visibility mode constant - Use display to hide element
9859      * @static
9860      * @type Number
9861      */
9862     El.DISPLAY = 2;
9863
9864     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9865     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9866     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9867
9868
9869
9870     /**
9871      * @private
9872      */
9873     El.cache = {};
9874
9875     var docEl;
9876
9877     /**
9878      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9879      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9880      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9881      * @return {Element} The Element object
9882      * @static
9883      */
9884     El.get = function(el){
9885         var ex, elm, id;
9886         if(!el){ return null; }
9887         if(typeof el == "string"){ // element id
9888             if(!(elm = document.getElementById(el))){
9889                 return null;
9890             }
9891             if(ex = El.cache[el]){
9892                 ex.dom = elm;
9893             }else{
9894                 ex = El.cache[el] = new El(elm);
9895             }
9896             return ex;
9897         }else if(el.tagName){ // dom element
9898             if(!(id = el.id)){
9899                 id = Roo.id(el);
9900             }
9901             if(ex = El.cache[id]){
9902                 ex.dom = el;
9903             }else{
9904                 ex = El.cache[id] = new El(el);
9905             }
9906             return ex;
9907         }else if(el instanceof El){
9908             if(el != docEl){
9909                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9910                                                               // catch case where it hasn't been appended
9911                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9912             }
9913             return el;
9914         }else if(el.isComposite){
9915             return el;
9916         }else if(el instanceof Array){
9917             return El.select(el);
9918         }else if(el == document){
9919             // create a bogus element object representing the document object
9920             if(!docEl){
9921                 var f = function(){};
9922                 f.prototype = El.prototype;
9923                 docEl = new f();
9924                 docEl.dom = document;
9925             }
9926             return docEl;
9927         }
9928         return null;
9929     };
9930
9931     // private
9932     El.uncache = function(el){
9933         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9934             if(a[i]){
9935                 delete El.cache[a[i].id || a[i]];
9936             }
9937         }
9938     };
9939
9940     // private
9941     // Garbage collection - uncache elements/purge listeners on orphaned elements
9942     // so we don't hold a reference and cause the browser to retain them
9943     El.garbageCollect = function(){
9944         if(!Roo.enableGarbageCollector){
9945             clearInterval(El.collectorThread);
9946             return;
9947         }
9948         for(var eid in El.cache){
9949             var el = El.cache[eid], d = el.dom;
9950             // -------------------------------------------------------
9951             // Determining what is garbage:
9952             // -------------------------------------------------------
9953             // !d
9954             // dom node is null, definitely garbage
9955             // -------------------------------------------------------
9956             // !d.parentNode
9957             // no parentNode == direct orphan, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.offsetParent && !document.getElementById(eid)
9960             // display none elements have no offsetParent so we will
9961             // also try to look it up by it's id. However, check
9962             // offsetParent first so we don't do unneeded lookups.
9963             // This enables collection of elements that are not orphans
9964             // directly, but somewhere up the line they have an orphan
9965             // parent.
9966             // -------------------------------------------------------
9967             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9968                 delete El.cache[eid];
9969                 if(d && Roo.enableListenerCollection){
9970                     E.purgeElement(d);
9971                 }
9972             }
9973         }
9974     }
9975     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9976
9977
9978     // dom is optional
9979     El.Flyweight = function(dom){
9980         this.dom = dom;
9981     };
9982     El.Flyweight.prototype = El.prototype;
9983
9984     El._flyweights = {};
9985     /**
9986      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9987      * the dom node can be overwritten by other code.
9988      * @param {String/HTMLElement} el The dom node or id
9989      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9990      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9991      * @static
9992      * @return {Element} The shared Element object
9993      */
9994     El.fly = function(el, named){
9995         named = named || '_global';
9996         el = Roo.getDom(el);
9997         if(!el){
9998             return null;
9999         }
10000         if(!El._flyweights[named]){
10001             El._flyweights[named] = new El.Flyweight();
10002         }
10003         El._flyweights[named].dom = el;
10004         return El._flyweights[named];
10005     };
10006
10007     /**
10008      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10009      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10010      * Shorthand of {@link Roo.Element#get}
10011      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10012      * @return {Element} The Element object
10013      * @member Roo
10014      * @method get
10015      */
10016     Roo.get = El.get;
10017     /**
10018      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10019      * the dom node can be overwritten by other code.
10020      * Shorthand of {@link Roo.Element#fly}
10021      * @param {String/HTMLElement} el The dom node or id
10022      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10023      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10024      * @static
10025      * @return {Element} The shared Element object
10026      * @member Roo
10027      * @method fly
10028      */
10029     Roo.fly = El.fly;
10030
10031     // speedy lookup for elements never to box adjust
10032     var noBoxAdjust = Roo.isStrict ? {
10033         select:1
10034     } : {
10035         input:1, select:1, textarea:1
10036     };
10037     if(Roo.isIE || Roo.isGecko){
10038         noBoxAdjust['button'] = 1;
10039     }
10040
10041
10042     Roo.EventManager.on(window, 'unload', function(){
10043         delete El.cache;
10044         delete El._flyweights;
10045     });
10046 })();
10047
10048
10049
10050
10051 if(Roo.DomQuery){
10052     Roo.Element.selectorFunction = Roo.DomQuery.select;
10053 }
10054
10055 Roo.Element.select = function(selector, unique, root){
10056     var els;
10057     if(typeof selector == "string"){
10058         els = Roo.Element.selectorFunction(selector, root);
10059     }else if(selector.length !== undefined){
10060         els = selector;
10061     }else{
10062         throw "Invalid selector";
10063     }
10064     if(unique === true){
10065         return new Roo.CompositeElement(els);
10066     }else{
10067         return new Roo.CompositeElementLite(els);
10068     }
10069 };
10070 /**
10071  * Selects elements based on the passed CSS selector to enable working on them as 1.
10072  * @param {String/Array} selector The CSS selector or an array of elements
10073  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10074  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10075  * @return {CompositeElementLite/CompositeElement}
10076  * @member Roo
10077  * @method select
10078  */
10079 Roo.select = Roo.Element.select;
10080
10081
10082
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094 /*
10095  * Based on:
10096  * Ext JS Library 1.1.1
10097  * Copyright(c) 2006-2007, Ext JS, LLC.
10098  *
10099  * Originally Released Under LGPL - original licence link has changed is not relivant.
10100  *
10101  * Fork - LGPL
10102  * <script type="text/javascript">
10103  */
10104
10105
10106
10107 //Notifies Element that fx methods are available
10108 Roo.enableFx = true;
10109
10110 /**
10111  * @class Roo.Fx
10112  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10113  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10114  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10115  * Element effects to work.</p><br/>
10116  *
10117  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10118  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10119  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10120  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10121  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10122  * expected results and should be done with care.</p><br/>
10123  *
10124  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10125  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10126 <pre>
10127 Value  Description
10128 -----  -----------------------------
10129 tl     The top left corner
10130 t      The center of the top edge
10131 tr     The top right corner
10132 l      The center of the left edge
10133 r      The center of the right edge
10134 bl     The bottom left corner
10135 b      The center of the bottom edge
10136 br     The bottom right corner
10137 </pre>
10138  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10139  * below are common options that can be passed to any Fx method.</b>
10140  * @cfg {Function} callback A function called when the effect is finished
10141  * @cfg {Object} scope The scope of the effect function
10142  * @cfg {String} easing A valid Easing value for the effect
10143  * @cfg {String} afterCls A css class to apply after the effect
10144  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10145  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10146  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10147  * effects that end with the element being visually hidden, ignored otherwise)
10148  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10149  * a function which returns such a specification that will be applied to the Element after the effect finishes
10150  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10151  * @cfg {Boolean} concurrent Whether to allow subsequently-queued effects to run at the same time as the current effect, or to ensure that they run in sequence
10152  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10153  */
10154 Roo.Fx = {
10155         /**
10156          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10157          * origin for the slide effect.  This function automatically handles wrapping the element with
10158          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10159          * Usage:
10160          *<pre><code>
10161 // default: slide the element in from the top
10162 el.slideIn();
10163
10164 // custom: slide the element in from the right with a 2-second duration
10165 el.slideIn('r', { duration: 2 });
10166
10167 // common config options shown with default values
10168 el.slideIn('t', {
10169     easing: 'easeOut',
10170     duration: .5
10171 });
10172 </code></pre>
10173          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10174          * @param {Object} options (optional) Object literal with any of the Fx config options
10175          * @return {Roo.Element} The Element
10176          */
10177     slideIn : function(anchor, o){
10178         var el = this.getFxEl();
10179         o = o || {};
10180
10181         el.queueFx(o, function(){
10182
10183             anchor = anchor || "t";
10184
10185             // fix display to visibility
10186             this.fixDisplay();
10187
10188             // restore values after effect
10189             var r = this.getFxRestore();
10190             var b = this.getBox();
10191             // fixed size for slide
10192             this.setSize(b);
10193
10194             // wrap if needed
10195             var wrap = this.fxWrap(r.pos, o, "hidden");
10196
10197             var st = this.dom.style;
10198             st.visibility = "visible";
10199             st.position = "absolute";
10200
10201             // clear out temp styles after slide and unwrap
10202             var after = function(){
10203                 el.fxUnwrap(wrap, r.pos, o);
10204                 st.width = r.width;
10205                 st.height = r.height;
10206                 el.afterFx(o);
10207             };
10208             // time to calc the positions
10209             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10210
10211             switch(anchor.toLowerCase()){
10212                 case "t":
10213                     wrap.setSize(b.width, 0);
10214                     st.left = st.bottom = "0";
10215                     a = {height: bh};
10216                 break;
10217                 case "l":
10218                     wrap.setSize(0, b.height);
10219                     st.right = st.top = "0";
10220                     a = {width: bw};
10221                 break;
10222                 case "r":
10223                     wrap.setSize(0, b.height);
10224                     wrap.setX(b.right);
10225                     st.left = st.top = "0";
10226                     a = {width: bw, points: pt};
10227                 break;
10228                 case "b":
10229                     wrap.setSize(b.width, 0);
10230                     wrap.setY(b.bottom);
10231                     st.left = st.top = "0";
10232                     a = {height: bh, points: pt};
10233                 break;
10234                 case "tl":
10235                     wrap.setSize(0, 0);
10236                     st.right = st.bottom = "0";
10237                     a = {width: bw, height: bh};
10238                 break;
10239                 case "bl":
10240                     wrap.setSize(0, 0);
10241                     wrap.setY(b.y+b.height);
10242                     st.right = st.top = "0";
10243                     a = {width: bw, height: bh, points: pt};
10244                 break;
10245                 case "br":
10246                     wrap.setSize(0, 0);
10247                     wrap.setXY([b.right, b.bottom]);
10248                     st.left = st.top = "0";
10249                     a = {width: bw, height: bh, points: pt};
10250                 break;
10251                 case "tr":
10252                     wrap.setSize(0, 0);
10253                     wrap.setX(b.x+b.width);
10254                     st.left = st.bottom = "0";
10255                     a = {width: bw, height: bh, points: pt};
10256                 break;
10257             }
10258             this.dom.style.visibility = "visible";
10259             wrap.show();
10260
10261             arguments.callee.anim = wrap.fxanim(a,
10262                 o,
10263                 'motion',
10264                 .5,
10265                 'easeOut', after);
10266         });
10267         return this;
10268     },
10269     
10270         /**
10271          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10272          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10273          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10274          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10275          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10276          * Usage:
10277          *<pre><code>
10278 // default: slide the element out to the top
10279 el.slideOut();
10280
10281 // custom: slide the element out to the right with a 2-second duration
10282 el.slideOut('r', { duration: 2 });
10283
10284 // common config options shown with default values
10285 el.slideOut('t', {
10286     easing: 'easeOut',
10287     duration: .5,
10288     remove: false,
10289     useDisplay: false
10290 });
10291 </code></pre>
10292          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10293          * @param {Object} options (optional) Object literal with any of the Fx config options
10294          * @return {Roo.Element} The Element
10295          */
10296     slideOut : function(anchor, o){
10297         var el = this.getFxEl();
10298         o = o || {};
10299
10300         el.queueFx(o, function(){
10301
10302             anchor = anchor || "t";
10303
10304             // restore values after effect
10305             var r = this.getFxRestore();
10306             
10307             var b = this.getBox();
10308             // fixed size for slide
10309             this.setSize(b);
10310
10311             // wrap if needed
10312             var wrap = this.fxWrap(r.pos, o, "visible");
10313
10314             var st = this.dom.style;
10315             st.visibility = "visible";
10316             st.position = "absolute";
10317
10318             wrap.setSize(b);
10319
10320             var after = function(){
10321                 if(o.useDisplay){
10322                     el.setDisplayed(false);
10323                 }else{
10324                     el.hide();
10325                 }
10326
10327                 el.fxUnwrap(wrap, r.pos, o);
10328
10329                 st.width = r.width;
10330                 st.height = r.height;
10331
10332                 el.afterFx(o);
10333             };
10334
10335             var a, zero = {to: 0};
10336             switch(anchor.toLowerCase()){
10337                 case "t":
10338                     st.left = st.bottom = "0";
10339                     a = {height: zero};
10340                 break;
10341                 case "l":
10342                     st.right = st.top = "0";
10343                     a = {width: zero};
10344                 break;
10345                 case "r":
10346                     st.left = st.top = "0";
10347                     a = {width: zero, points: {to:[b.right, b.y]}};
10348                 break;
10349                 case "b":
10350                     st.left = st.top = "0";
10351                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10352                 break;
10353                 case "tl":
10354                     st.right = st.bottom = "0";
10355                     a = {width: zero, height: zero};
10356                 break;
10357                 case "bl":
10358                     st.right = st.top = "0";
10359                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10360                 break;
10361                 case "br":
10362                     st.left = st.top = "0";
10363                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10364                 break;
10365                 case "tr":
10366                     st.left = st.bottom = "0";
10367                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10368                 break;
10369             }
10370
10371             arguments.callee.anim = wrap.fxanim(a,
10372                 o,
10373                 'motion',
10374                 .5,
10375                 "easeOut", after);
10376         });
10377         return this;
10378     },
10379
10380         /**
10381          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10382          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10383          * The element must be removed from the DOM using the 'remove' config option if desired.
10384          * Usage:
10385          *<pre><code>
10386 // default
10387 el.puff();
10388
10389 // common config options shown with default values
10390 el.puff({
10391     easing: 'easeOut',
10392     duration: .5,
10393     remove: false,
10394     useDisplay: false
10395 });
10396 </code></pre>
10397          * @param {Object} options (optional) Object literal with any of the Fx config options
10398          * @return {Roo.Element} The Element
10399          */
10400     puff : function(o){
10401         var el = this.getFxEl();
10402         o = o || {};
10403
10404         el.queueFx(o, function(){
10405             this.clearOpacity();
10406             this.show();
10407
10408             // restore values after effect
10409             var r = this.getFxRestore();
10410             var st = this.dom.style;
10411
10412             var after = function(){
10413                 if(o.useDisplay){
10414                     el.setDisplayed(false);
10415                 }else{
10416                     el.hide();
10417                 }
10418
10419                 el.clearOpacity();
10420
10421                 el.setPositioning(r.pos);
10422                 st.width = r.width;
10423                 st.height = r.height;
10424                 st.fontSize = '';
10425                 el.afterFx(o);
10426             };
10427
10428             var width = this.getWidth();
10429             var height = this.getHeight();
10430
10431             arguments.callee.anim = this.fxanim({
10432                     width : {to: this.adjustWidth(width * 2)},
10433                     height : {to: this.adjustHeight(height * 2)},
10434                     points : {by: [-(width * .5), -(height * .5)]},
10435                     opacity : {to: 0},
10436                     fontSize: {to:200, unit: "%"}
10437                 },
10438                 o,
10439                 'motion',
10440                 .5,
10441                 "easeOut", after);
10442         });
10443         return this;
10444     },
10445
10446         /**
10447          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10448          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10449          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10450          * Usage:
10451          *<pre><code>
10452 // default
10453 el.switchOff();
10454
10455 // all config options shown with default values
10456 el.switchOff({
10457     easing: 'easeIn',
10458     duration: .3,
10459     remove: false,
10460     useDisplay: false
10461 });
10462 </code></pre>
10463          * @param {Object} options (optional) Object literal with any of the Fx config options
10464          * @return {Roo.Element} The Element
10465          */
10466     switchOff : function(o){
10467         var el = this.getFxEl();
10468         o = o || {};
10469
10470         el.queueFx(o, function(){
10471             this.clearOpacity();
10472             this.clip();
10473
10474             // restore values after effect
10475             var r = this.getFxRestore();
10476             var st = this.dom.style;
10477
10478             var after = function(){
10479                 if(o.useDisplay){
10480                     el.setDisplayed(false);
10481                 }else{
10482                     el.hide();
10483                 }
10484
10485                 el.clearOpacity();
10486                 el.setPositioning(r.pos);
10487                 st.width = r.width;
10488                 st.height = r.height;
10489
10490                 el.afterFx(o);
10491             };
10492
10493             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10494                 this.clearOpacity();
10495                 (function(){
10496                     this.fxanim({
10497                         height:{to:1},
10498                         points:{by:[0, this.getHeight() * .5]}
10499                     }, o, 'motion', 0.3, 'easeIn', after);
10500                 }).defer(100, this);
10501             });
10502         });
10503         return this;
10504     },
10505
10506     /**
10507      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10508      * changed using the "attr" config option) and then fading back to the original color. If no original
10509      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10510      * Usage:
10511 <pre><code>
10512 // default: highlight background to yellow
10513 el.highlight();
10514
10515 // custom: highlight foreground text to blue for 2 seconds
10516 el.highlight("0000ff", { attr: 'color', duration: 2 });
10517
10518 // common config options shown with default values
10519 el.highlight("ffff9c", {
10520     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10521     endColor: (current color) or "ffffff",
10522     easing: 'easeIn',
10523     duration: 1
10524 });
10525 </code></pre>
10526      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10527      * @param {Object} options (optional) Object literal with any of the Fx config options
10528      * @return {Roo.Element} The Element
10529      */ 
10530     highlight : function(color, o){
10531         var el = this.getFxEl();
10532         o = o || {};
10533
10534         el.queueFx(o, function(){
10535             color = color || "ffff9c";
10536             attr = o.attr || "backgroundColor";
10537
10538             this.clearOpacity();
10539             this.show();
10540
10541             var origColor = this.getColor(attr);
10542             var restoreColor = this.dom.style[attr];
10543             endColor = (o.endColor || origColor) || "ffffff";
10544
10545             var after = function(){
10546                 el.dom.style[attr] = restoreColor;
10547                 el.afterFx(o);
10548             };
10549
10550             var a = {};
10551             a[attr] = {from: color, to: endColor};
10552             arguments.callee.anim = this.fxanim(a,
10553                 o,
10554                 'color',
10555                 1,
10556                 'easeIn', after);
10557         });
10558         return this;
10559     },
10560
10561    /**
10562     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10563     * Usage:
10564 <pre><code>
10565 // default: a single light blue ripple
10566 el.frame();
10567
10568 // custom: 3 red ripples lasting 3 seconds total
10569 el.frame("ff0000", 3, { duration: 3 });
10570
10571 // common config options shown with default values
10572 el.frame("C3DAF9", 1, {
10573     duration: 1 //duration of entire animation (not each individual ripple)
10574     // Note: Easing is not configurable and will be ignored if included
10575 });
10576 </code></pre>
10577     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10578     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10579     * @param {Object} options (optional) Object literal with any of the Fx config options
10580     * @return {Roo.Element} The Element
10581     */
10582     frame : function(color, count, o){
10583         var el = this.getFxEl();
10584         o = o || {};
10585
10586         el.queueFx(o, function(){
10587             color = color || "#C3DAF9";
10588             if(color.length == 6){
10589                 color = "#" + color;
10590             }
10591             count = count || 1;
10592             duration = o.duration || 1;
10593             this.show();
10594
10595             var b = this.getBox();
10596             var animFn = function(){
10597                 var proxy = this.createProxy({
10598
10599                      style:{
10600                         visbility:"hidden",
10601                         position:"absolute",
10602                         "z-index":"35000", // yee haw
10603                         border:"0px solid " + color
10604                      }
10605                   });
10606                 var scale = Roo.isBorderBox ? 2 : 1;
10607                 proxy.animate({
10608                     top:{from:b.y, to:b.y - 20},
10609                     left:{from:b.x, to:b.x - 20},
10610                     borderWidth:{from:0, to:10},
10611                     opacity:{from:1, to:0},
10612                     height:{from:b.height, to:(b.height + (20*scale))},
10613                     width:{from:b.width, to:(b.width + (20*scale))}
10614                 }, duration, function(){
10615                     proxy.remove();
10616                 });
10617                 if(--count > 0){
10618                      animFn.defer((duration/2)*1000, this);
10619                 }else{
10620                     el.afterFx(o);
10621                 }
10622             };
10623             animFn.call(this);
10624         });
10625         return this;
10626     },
10627
10628    /**
10629     * Creates a pause before any subsequent queued effects begin.  If there are
10630     * no effects queued after the pause it will have no effect.
10631     * Usage:
10632 <pre><code>
10633 el.pause(1);
10634 </code></pre>
10635     * @param {Number} seconds The length of time to pause (in seconds)
10636     * @return {Roo.Element} The Element
10637     */
10638     pause : function(seconds){
10639         var el = this.getFxEl();
10640         var o = {};
10641
10642         el.queueFx(o, function(){
10643             setTimeout(function(){
10644                 el.afterFx(o);
10645             }, seconds * 1000);
10646         });
10647         return this;
10648     },
10649
10650    /**
10651     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10652     * using the "endOpacity" config option.
10653     * Usage:
10654 <pre><code>
10655 // default: fade in from opacity 0 to 100%
10656 el.fadeIn();
10657
10658 // custom: fade in from opacity 0 to 75% over 2 seconds
10659 el.fadeIn({ endOpacity: .75, duration: 2});
10660
10661 // common config options shown with default values
10662 el.fadeIn({
10663     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10664     easing: 'easeOut',
10665     duration: .5
10666 });
10667 </code></pre>
10668     * @param {Object} options (optional) Object literal with any of the Fx config options
10669     * @return {Roo.Element} The Element
10670     */
10671     fadeIn : function(o){
10672         var el = this.getFxEl();
10673         o = o || {};
10674         el.queueFx(o, function(){
10675             this.setOpacity(0);
10676             this.fixDisplay();
10677             this.dom.style.visibility = 'visible';
10678             var to = o.endOpacity || 1;
10679             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10680                 o, null, .5, "easeOut", function(){
10681                 if(to == 1){
10682                     this.clearOpacity();
10683                 }
10684                 el.afterFx(o);
10685             });
10686         });
10687         return this;
10688     },
10689
10690    /**
10691     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10692     * using the "endOpacity" config option.
10693     * Usage:
10694 <pre><code>
10695 // default: fade out from the element's current opacity to 0
10696 el.fadeOut();
10697
10698 // custom: fade out from the element's current opacity to 25% over 2 seconds
10699 el.fadeOut({ endOpacity: .25, duration: 2});
10700
10701 // common config options shown with default values
10702 el.fadeOut({
10703     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10704     easing: 'easeOut',
10705     duration: .5
10706     remove: false,
10707     useDisplay: false
10708 });
10709 </code></pre>
10710     * @param {Object} options (optional) Object literal with any of the Fx config options
10711     * @return {Roo.Element} The Element
10712     */
10713     fadeOut : function(o){
10714         var el = this.getFxEl();
10715         o = o || {};
10716         el.queueFx(o, function(){
10717             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10718                 o, null, .5, "easeOut", function(){
10719                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10720                      this.dom.style.display = "none";
10721                 }else{
10722                      this.dom.style.visibility = "hidden";
10723                 }
10724                 this.clearOpacity();
10725                 el.afterFx(o);
10726             });
10727         });
10728         return this;
10729     },
10730
10731    /**
10732     * Animates the transition of an element's dimensions from a starting height/width
10733     * to an ending height/width.
10734     * Usage:
10735 <pre><code>
10736 // change height and width to 100x100 pixels
10737 el.scale(100, 100);
10738
10739 // common config options shown with default values.  The height and width will default to
10740 // the element's existing values if passed as null.
10741 el.scale(
10742     [element's width],
10743     [element's height], {
10744     easing: 'easeOut',
10745     duration: .35
10746 });
10747 </code></pre>
10748     * @param {Number} width  The new width (pass undefined to keep the original width)
10749     * @param {Number} height  The new height (pass undefined to keep the original height)
10750     * @param {Object} options (optional) Object literal with any of the Fx config options
10751     * @return {Roo.Element} The Element
10752     */
10753     scale : function(w, h, o){
10754         this.shift(Roo.apply({}, o, {
10755             width: w,
10756             height: h
10757         }));
10758         return this;
10759     },
10760
10761    /**
10762     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10763     * Any of these properties not specified in the config object will not be changed.  This effect 
10764     * requires that at least one new dimension, position or opacity setting must be passed in on
10765     * the config object in order for the function to have any effect.
10766     * Usage:
10767 <pre><code>
10768 // slide the element horizontally to x position 200 while changing the height and opacity
10769 el.shift({ x: 200, height: 50, opacity: .8 });
10770
10771 // common config options shown with default values.
10772 el.shift({
10773     width: [element's width],
10774     height: [element's height],
10775     x: [element's x position],
10776     y: [element's y position],
10777     opacity: [element's opacity],
10778     easing: 'easeOut',
10779     duration: .35
10780 });
10781 </code></pre>
10782     * @param {Object} options  Object literal with any of the Fx config options
10783     * @return {Roo.Element} The Element
10784     */
10785     shift : function(o){
10786         var el = this.getFxEl();
10787         o = o || {};
10788         el.queueFx(o, function(){
10789             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10790             if(w !== undefined){
10791                 a.width = {to: this.adjustWidth(w)};
10792             }
10793             if(h !== undefined){
10794                 a.height = {to: this.adjustHeight(h)};
10795             }
10796             if(x !== undefined || y !== undefined){
10797                 a.points = {to: [
10798                     x !== undefined ? x : this.getX(),
10799                     y !== undefined ? y : this.getY()
10800                 ]};
10801             }
10802             if(op !== undefined){
10803                 a.opacity = {to: op};
10804             }
10805             if(o.xy !== undefined){
10806                 a.points = {to: o.xy};
10807             }
10808             arguments.callee.anim = this.fxanim(a,
10809                 o, 'motion', .35, "easeOut", function(){
10810                 el.afterFx(o);
10811             });
10812         });
10813         return this;
10814     },
10815
10816         /**
10817          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10818          * ending point of the effect.
10819          * Usage:
10820          *<pre><code>
10821 // default: slide the element downward while fading out
10822 el.ghost();
10823
10824 // custom: slide the element out to the right with a 2-second duration
10825 el.ghost('r', { duration: 2 });
10826
10827 // common config options shown with default values
10828 el.ghost('b', {
10829     easing: 'easeOut',
10830     duration: .5
10831     remove: false,
10832     useDisplay: false
10833 });
10834 </code></pre>
10835          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10836          * @param {Object} options (optional) Object literal with any of the Fx config options
10837          * @return {Roo.Element} The Element
10838          */
10839     ghost : function(anchor, o){
10840         var el = this.getFxEl();
10841         o = o || {};
10842
10843         el.queueFx(o, function(){
10844             anchor = anchor || "b";
10845
10846             // restore values after effect
10847             var r = this.getFxRestore();
10848             var w = this.getWidth(),
10849                 h = this.getHeight();
10850
10851             var st = this.dom.style;
10852
10853             var after = function(){
10854                 if(o.useDisplay){
10855                     el.setDisplayed(false);
10856                 }else{
10857                     el.hide();
10858                 }
10859
10860                 el.clearOpacity();
10861                 el.setPositioning(r.pos);
10862                 st.width = r.width;
10863                 st.height = r.height;
10864
10865                 el.afterFx(o);
10866             };
10867
10868             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10869             switch(anchor.toLowerCase()){
10870                 case "t":
10871                     pt.by = [0, -h];
10872                 break;
10873                 case "l":
10874                     pt.by = [-w, 0];
10875                 break;
10876                 case "r":
10877                     pt.by = [w, 0];
10878                 break;
10879                 case "b":
10880                     pt.by = [0, h];
10881                 break;
10882                 case "tl":
10883                     pt.by = [-w, -h];
10884                 break;
10885                 case "bl":
10886                     pt.by = [-w, h];
10887                 break;
10888                 case "br":
10889                     pt.by = [w, h];
10890                 break;
10891                 case "tr":
10892                     pt.by = [w, -h];
10893                 break;
10894             }
10895
10896             arguments.callee.anim = this.fxanim(a,
10897                 o,
10898                 'motion',
10899                 .5,
10900                 "easeOut", after);
10901         });
10902         return this;
10903     },
10904
10905         /**
10906          * Ensures that all effects queued after syncFx is called on the element are
10907          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10908          * @return {Roo.Element} The Element
10909          */
10910     syncFx : function(){
10911         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10912             block : false,
10913             concurrent : true,
10914             stopFx : false
10915         });
10916         return this;
10917     },
10918
10919         /**
10920          * Ensures that all effects queued after sequenceFx is called on the element are
10921          * run in sequence.  This is the opposite of {@link #syncFx}.
10922          * @return {Roo.Element} The Element
10923          */
10924     sequenceFx : function(){
10925         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10926             block : false,
10927             concurrent : false,
10928             stopFx : false
10929         });
10930         return this;
10931     },
10932
10933         /* @private */
10934     nextFx : function(){
10935         var ef = this.fxQueue[0];
10936         if(ef){
10937             ef.call(this);
10938         }
10939     },
10940
10941         /**
10942          * Returns true if the element has any effects actively running or queued, else returns false.
10943          * @return {Boolean} True if element has active effects, else false
10944          */
10945     hasActiveFx : function(){
10946         return this.fxQueue && this.fxQueue[0];
10947     },
10948
10949         /**
10950          * Stops any running effects and clears the element's internal effects queue if it contains
10951          * any additional effects that haven't started yet.
10952          * @return {Roo.Element} The Element
10953          */
10954     stopFx : function(){
10955         if(this.hasActiveFx()){
10956             var cur = this.fxQueue[0];
10957             if(cur && cur.anim && cur.anim.isAnimated()){
10958                 this.fxQueue = [cur]; // clear out others
10959                 cur.anim.stop(true);
10960             }
10961         }
10962         return this;
10963     },
10964
10965         /* @private */
10966     beforeFx : function(o){
10967         if(this.hasActiveFx() && !o.concurrent){
10968            if(o.stopFx){
10969                this.stopFx();
10970                return true;
10971            }
10972            return false;
10973         }
10974         return true;
10975     },
10976
10977         /**
10978          * Returns true if the element is currently blocking so that no other effect can be queued
10979          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10980          * used to ensure that an effect initiated by a user action runs to completion prior to the
10981          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10982          * @return {Boolean} True if blocking, else false
10983          */
10984     hasFxBlock : function(){
10985         var q = this.fxQueue;
10986         return q && q[0] && q[0].block;
10987     },
10988
10989         /* @private */
10990     queueFx : function(o, fn){
10991         if(!this.fxQueue){
10992             this.fxQueue = [];
10993         }
10994         if(!this.hasFxBlock()){
10995             Roo.applyIf(o, this.fxDefaults);
10996             if(!o.concurrent){
10997                 var run = this.beforeFx(o);
10998                 fn.block = o.block;
10999                 this.fxQueue.push(fn);
11000                 if(run){
11001                     this.nextFx();
11002                 }
11003             }else{
11004                 fn.call(this);
11005             }
11006         }
11007         return this;
11008     },
11009
11010         /* @private */
11011     fxWrap : function(pos, o, vis){
11012         var wrap;
11013         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11014             var wrapXY;
11015             if(o.fixPosition){
11016                 wrapXY = this.getXY();
11017             }
11018             var div = document.createElement("div");
11019             div.style.visibility = vis;
11020             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11021             wrap.setPositioning(pos);
11022             if(wrap.getStyle("position") == "static"){
11023                 wrap.position("relative");
11024             }
11025             this.clearPositioning('auto');
11026             wrap.clip();
11027             wrap.dom.appendChild(this.dom);
11028             if(wrapXY){
11029                 wrap.setXY(wrapXY);
11030             }
11031         }
11032         return wrap;
11033     },
11034
11035         /* @private */
11036     fxUnwrap : function(wrap, pos, o){
11037         this.clearPositioning();
11038         this.setPositioning(pos);
11039         if(!o.wrap){
11040             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11041             wrap.remove();
11042         }
11043     },
11044
11045         /* @private */
11046     getFxRestore : function(){
11047         var st = this.dom.style;
11048         return {pos: this.getPositioning(), width: st.width, height : st.height};
11049     },
11050
11051         /* @private */
11052     afterFx : function(o){
11053         if(o.afterStyle){
11054             this.applyStyles(o.afterStyle);
11055         }
11056         if(o.afterCls){
11057             this.addClass(o.afterCls);
11058         }
11059         if(o.remove === true){
11060             this.remove();
11061         }
11062         Roo.callback(o.callback, o.scope, [this]);
11063         if(!o.concurrent){
11064             this.fxQueue.shift();
11065             this.nextFx();
11066         }
11067     },
11068
11069         /* @private */
11070     getFxEl : function(){ // support for composite element fx
11071         return Roo.get(this.dom);
11072     },
11073
11074         /* @private */
11075     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11076         animType = animType || 'run';
11077         opt = opt || {};
11078         var anim = Roo.lib.Anim[animType](
11079             this.dom, args,
11080             (opt.duration || defaultDur) || .35,
11081             (opt.easing || defaultEase) || 'easeOut',
11082             function(){
11083                 Roo.callback(cb, this);
11084             },
11085             this
11086         );
11087         opt.anim = anim;
11088         return anim;
11089     }
11090 };
11091
11092 // backwords compat
11093 Roo.Fx.resize = Roo.Fx.scale;
11094
11095 //When included, Roo.Fx is automatically applied to Element so that all basic
11096 //effects are available directly via the Element API
11097 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11098  * Based on:
11099  * Ext JS Library 1.1.1
11100  * Copyright(c) 2006-2007, Ext JS, LLC.
11101  *
11102  * Originally Released Under LGPL - original licence link has changed is not relivant.
11103  *
11104  * Fork - LGPL
11105  * <script type="text/javascript">
11106  */
11107
11108
11109 /**
11110  * @class Roo.CompositeElement
11111  * Standard composite class. Creates a Roo.Element for every element in the collection.
11112  * <br><br>
11113  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11114  * actions will be performed on all the elements in this collection.</b>
11115  * <br><br>
11116  * All methods return <i>this</i> and can be chained.
11117  <pre><code>
11118  var els = Roo.select("#some-el div.some-class", true);
11119  // or select directly from an existing element
11120  var el = Roo.get('some-el');
11121  el.select('div.some-class', true);
11122
11123  els.setWidth(100); // all elements become 100 width
11124  els.hide(true); // all elements fade out and hide
11125  // or
11126  els.setWidth(100).hide(true);
11127  </code></pre>
11128  */
11129 Roo.CompositeElement = function(els){
11130     this.elements = [];
11131     this.addElements(els);
11132 };
11133 Roo.CompositeElement.prototype = {
11134     isComposite: true,
11135     addElements : function(els){
11136         if(!els) {
11137             return this;
11138         }
11139         if(typeof els == "string"){
11140             els = Roo.Element.selectorFunction(els);
11141         }
11142         var yels = this.elements;
11143         var index = yels.length-1;
11144         for(var i = 0, len = els.length; i < len; i++) {
11145                 yels[++index] = Roo.get(els[i]);
11146         }
11147         return this;
11148     },
11149
11150     /**
11151     * Clears this composite and adds the elements returned by the passed selector.
11152     * @param {String/Array} els A string CSS selector, an array of elements or an element
11153     * @return {CompositeElement} this
11154     */
11155     fill : function(els){
11156         this.elements = [];
11157         this.add(els);
11158         return this;
11159     },
11160
11161     /**
11162     * Filters this composite to only elements that match the passed selector.
11163     * @param {String} selector A string CSS selector
11164     * @param {Boolean} inverse return inverse filter (not matches)
11165     * @return {CompositeElement} this
11166     */
11167     filter : function(selector, inverse){
11168         var els = [];
11169         inverse = inverse || false;
11170         this.each(function(el){
11171             var match = inverse ? !el.is(selector) : el.is(selector);
11172             if(match){
11173                 els[els.length] = el.dom;
11174             }
11175         });
11176         this.fill(els);
11177         return this;
11178     },
11179
11180     invoke : function(fn, args){
11181         var els = this.elements;
11182         for(var i = 0, len = els.length; i < len; i++) {
11183                 Roo.Element.prototype[fn].apply(els[i], args);
11184         }
11185         return this;
11186     },
11187     /**
11188     * Adds elements to this composite.
11189     * @param {String/Array} els A string CSS selector, an array of elements or an element
11190     * @return {CompositeElement} this
11191     */
11192     add : function(els){
11193         if(typeof els == "string"){
11194             this.addElements(Roo.Element.selectorFunction(els));
11195         }else if(els.length !== undefined){
11196             this.addElements(els);
11197         }else{
11198             this.addElements([els]);
11199         }
11200         return this;
11201     },
11202     /**
11203     * Calls the passed function passing (el, this, index) for each element in this composite.
11204     * @param {Function} fn The function to call
11205     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11206     * @return {CompositeElement} this
11207     */
11208     each : function(fn, scope){
11209         var els = this.elements;
11210         for(var i = 0, len = els.length; i < len; i++){
11211             if(fn.call(scope || els[i], els[i], this, i) === false) {
11212                 break;
11213             }
11214         }
11215         return this;
11216     },
11217
11218     /**
11219      * Returns the Element object at the specified index
11220      * @param {Number} index
11221      * @return {Roo.Element}
11222      */
11223     item : function(index){
11224         return this.elements[index] || null;
11225     },
11226
11227     /**
11228      * Returns the first Element
11229      * @return {Roo.Element}
11230      */
11231     first : function(){
11232         return this.item(0);
11233     },
11234
11235     /**
11236      * Returns the last Element
11237      * @return {Roo.Element}
11238      */
11239     last : function(){
11240         return this.item(this.elements.length-1);
11241     },
11242
11243     /**
11244      * Returns the number of elements in this composite
11245      * @return Number
11246      */
11247     getCount : function(){
11248         return this.elements.length;
11249     },
11250
11251     /**
11252      * Returns true if this composite contains the passed element
11253      * @return Boolean
11254      */
11255     contains : function(el){
11256         return this.indexOf(el) !== -1;
11257     },
11258
11259     /**
11260      * Returns true if this composite contains the passed element
11261      * @return Boolean
11262      */
11263     indexOf : function(el){
11264         return this.elements.indexOf(Roo.get(el));
11265     },
11266
11267
11268     /**
11269     * Removes the specified element(s).
11270     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11271     * or an array of any of those.
11272     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11273     * @return {CompositeElement} this
11274     */
11275     removeElement : function(el, removeDom){
11276         if(el instanceof Array){
11277             for(var i = 0, len = el.length; i < len; i++){
11278                 this.removeElement(el[i]);
11279             }
11280             return this;
11281         }
11282         var index = typeof el == 'number' ? el : this.indexOf(el);
11283         if(index !== -1){
11284             if(removeDom){
11285                 var d = this.elements[index];
11286                 if(d.dom){
11287                     d.remove();
11288                 }else{
11289                     d.parentNode.removeChild(d);
11290                 }
11291             }
11292             this.elements.splice(index, 1);
11293         }
11294         return this;
11295     },
11296
11297     /**
11298     * Replaces the specified element with the passed element.
11299     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11300     * to replace.
11301     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11302     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11303     * @return {CompositeElement} this
11304     */
11305     replaceElement : function(el, replacement, domReplace){
11306         var index = typeof el == 'number' ? el : this.indexOf(el);
11307         if(index !== -1){
11308             if(domReplace){
11309                 this.elements[index].replaceWith(replacement);
11310             }else{
11311                 this.elements.splice(index, 1, Roo.get(replacement))
11312             }
11313         }
11314         return this;
11315     },
11316
11317     /**
11318      * Removes all elements.
11319      */
11320     clear : function(){
11321         this.elements = [];
11322     }
11323 };
11324 (function(){
11325     Roo.CompositeElement.createCall = function(proto, fnName){
11326         if(!proto[fnName]){
11327             proto[fnName] = function(){
11328                 return this.invoke(fnName, arguments);
11329             };
11330         }
11331     };
11332     for(var fnName in Roo.Element.prototype){
11333         if(typeof Roo.Element.prototype[fnName] == "function"){
11334             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11335         }
11336     };
11337 })();
11338 /*
11339  * Based on:
11340  * Ext JS Library 1.1.1
11341  * Copyright(c) 2006-2007, Ext JS, LLC.
11342  *
11343  * Originally Released Under LGPL - original licence link has changed is not relivant.
11344  *
11345  * Fork - LGPL
11346  * <script type="text/javascript">
11347  */
11348
11349 /**
11350  * @class Roo.CompositeElementLite
11351  * @extends Roo.CompositeElement
11352  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11353  <pre><code>
11354  var els = Roo.select("#some-el div.some-class");
11355  // or select directly from an existing element
11356  var el = Roo.get('some-el');
11357  el.select('div.some-class');
11358
11359  els.setWidth(100); // all elements become 100 width
11360  els.hide(true); // all elements fade out and hide
11361  // or
11362  els.setWidth(100).hide(true);
11363  </code></pre><br><br>
11364  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11365  * actions will be performed on all the elements in this collection.</b>
11366  */
11367 Roo.CompositeElementLite = function(els){
11368     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11369     this.el = new Roo.Element.Flyweight();
11370 };
11371 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11372     addElements : function(els){
11373         if(els){
11374             if(els instanceof Array){
11375                 this.elements = this.elements.concat(els);
11376             }else{
11377                 var yels = this.elements;
11378                 var index = yels.length-1;
11379                 for(var i = 0, len = els.length; i < len; i++) {
11380                     yels[++index] = els[i];
11381                 }
11382             }
11383         }
11384         return this;
11385     },
11386     invoke : function(fn, args){
11387         var els = this.elements;
11388         var el = this.el;
11389         for(var i = 0, len = els.length; i < len; i++) {
11390             el.dom = els[i];
11391                 Roo.Element.prototype[fn].apply(el, args);
11392         }
11393         return this;
11394     },
11395     /**
11396      * Returns a flyweight Element of the dom element object at the specified index
11397      * @param {Number} index
11398      * @return {Roo.Element}
11399      */
11400     item : function(index){
11401         if(!this.elements[index]){
11402             return null;
11403         }
11404         this.el.dom = this.elements[index];
11405         return this.el;
11406     },
11407
11408     // fixes scope with flyweight
11409     addListener : function(eventName, handler, scope, opt){
11410         var els = this.elements;
11411         for(var i = 0, len = els.length; i < len; i++) {
11412             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11413         }
11414         return this;
11415     },
11416
11417     /**
11418     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11419     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11420     * a reference to the dom node, use el.dom.</b>
11421     * @param {Function} fn The function to call
11422     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11423     * @return {CompositeElement} this
11424     */
11425     each : function(fn, scope){
11426         var els = this.elements;
11427         var el = this.el;
11428         for(var i = 0, len = els.length; i < len; i++){
11429             el.dom = els[i];
11430                 if(fn.call(scope || el, el, this, i) === false){
11431                 break;
11432             }
11433         }
11434         return this;
11435     },
11436
11437     indexOf : function(el){
11438         return this.elements.indexOf(Roo.getDom(el));
11439     },
11440
11441     replaceElement : function(el, replacement, domReplace){
11442         var index = typeof el == 'number' ? el : this.indexOf(el);
11443         if(index !== -1){
11444             replacement = Roo.getDom(replacement);
11445             if(domReplace){
11446                 var d = this.elements[index];
11447                 d.parentNode.insertBefore(replacement, d);
11448                 d.parentNode.removeChild(d);
11449             }
11450             this.elements.splice(index, 1, replacement);
11451         }
11452         return this;
11453     }
11454 });
11455 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11456
11457 /*
11458  * Based on:
11459  * Ext JS Library 1.1.1
11460  * Copyright(c) 2006-2007, Ext JS, LLC.
11461  *
11462  * Originally Released Under LGPL - original licence link has changed is not relivant.
11463  *
11464  * Fork - LGPL
11465  * <script type="text/javascript">
11466  */
11467
11468  
11469
11470 /**
11471  * @class Roo.data.Connection
11472  * @extends Roo.util.Observable
11473  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11474  * either to a configured URL, or to a URL specified at request time.<br><br>
11475  * <p>
11476  * Requests made by this class are asynchronous, and will return immediately. No data from
11477  * the server will be available to the statement immediately following the {@link #request} call.
11478  * To process returned data, use a callback in the request options object, or an event listener.</p><br>
11479  * <p>
11480  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11481  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11482  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11483  * property and, if present, the IFRAME's XML document as the responseXML property.</p><br>
11484  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11485  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11486  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11487  * standard DOM methods.
11488  * @constructor
11489  * @param {Object} config a configuration object.
11490  */
11491 Roo.data.Connection = function(config){
11492     Roo.apply(this, config);
11493     this.addEvents({
11494         /**
11495          * @event beforerequest
11496          * Fires before a network request is made to retrieve a data object.
11497          * @param {Connection} conn This Connection object.
11498          * @param {Object} options The options config object passed to the {@link #request} method.
11499          */
11500         "beforerequest" : true,
11501         /**
11502          * @event requestcomplete
11503          * Fires if the request was successfully completed.
11504          * @param {Connection} conn This Connection object.
11505          * @param {Object} response The XHR object containing the response data.
11506          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11507          * @param {Object} options The options config object passed to the {@link #request} method.
11508          */
11509         "requestcomplete" : true,
11510         /**
11511          * @event requestexception
11512          * Fires if an error HTTP status was returned from the server.
11513          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11514          * @param {Connection} conn This Connection object.
11515          * @param {Object} response The XHR object containing the response data.
11516          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11517          * @param {Object} options The options config object passed to the {@link #request} method.
11518          */
11519         "requestexception" : true
11520     });
11521     Roo.data.Connection.superclass.constructor.call(this);
11522 };
11523
11524 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11525     /**
11526      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11527      */
11528     /**
11529      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11530      * extra parameters to each request made by this object. (defaults to undefined)
11531      */
11532     /**
11533      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11534      *  to each request made by this object. (defaults to undefined)
11535      */
11536     /**
11537      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11538      */
11539     /**
11540      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11541      */
11542     timeout : 30000,
11543     /**
11544      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11545      * @type Boolean
11546      */
11547     autoAbort:false,
11548
11549     /**
11550      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11551      * @type Boolean
11552      */
11553     disableCaching: true,
11554
11555     /**
11556      * Sends an HTTP request to a remote server.
11557      * @param {Object} options An object which may contain the following properties:<ul>
11558      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11559      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11560      * request, a url encoded string or a function to call to get either.</li>
11561      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11562      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11563      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11564      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11565      * <li>options {Object} The parameter to the request call.</li>
11566      * <li>success {Boolean} True if the request succeeded.</li>
11567      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11568      * </ul></li>
11569      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11570      * The callback is passed the following parameters:<ul>
11571      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11572      * <li>options {Object} The parameter to the request call.</li>
11573      * </ul></li>
11574      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11575      * The callback is passed the following parameters:<ul>
11576      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11577      * <li>options {Object} The parameter to the request call.</li>
11578      * </ul></li>
11579      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11580      * for the callback function. Defaults to the browser window.</li>
11581      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11582      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11583      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11584      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11585      * params for the post data. Any params will be appended to the URL.</li>
11586      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11587      * </ul>
11588      * @return {Number} transactionId
11589      */
11590     request : function(o){
11591         if(this.fireEvent("beforerequest", this, o) !== false){
11592             var p = o.params;
11593
11594             if(typeof p == "function"){
11595                 p = p.call(o.scope||window, o);
11596             }
11597             if(typeof p == "object"){
11598                 p = Roo.urlEncode(o.params);
11599             }
11600             if(this.extraParams){
11601                 var extras = Roo.urlEncode(this.extraParams);
11602                 p = p ? (p + '&' + extras) : extras;
11603             }
11604
11605             var url = o.url || this.url;
11606             if(typeof url == 'function'){
11607                 url = url.call(o.scope||window, o);
11608             }
11609
11610             if(o.form){
11611                 var form = Roo.getDom(o.form);
11612                 url = url || form.action;
11613
11614                 var enctype = form.getAttribute("enctype");
11615                 
11616                 if (o.formData) {
11617                     return this.doFormDataUpload(o,p,url);
11618                 }
11619                 
11620                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11621                     return this.doFormUpload(o, p, url);
11622                 }
11623                 var f = Roo.lib.Ajax.serializeForm(form);
11624                 p = p ? (p + '&' + f) : f;
11625             }
11626
11627             var hs = o.headers;
11628             if(this.defaultHeaders){
11629                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11630                 if(!o.headers){
11631                     o.headers = hs;
11632                 }
11633             }
11634
11635             var cb = {
11636                 success: this.handleResponse,
11637                 failure: this.handleFailure,
11638                 scope: this,
11639                 argument: {options: o},
11640                 timeout : o.timeout || this.timeout
11641             };
11642
11643             var method = o.method||this.method||(p ? "POST" : "GET");
11644
11645             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11646                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11647             }
11648
11649             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11650                 if(o.autoAbort){
11651                     this.abort();
11652                 }
11653             }else if(this.autoAbort !== false){
11654                 this.abort();
11655             }
11656
11657             if((method == 'GET' && p) || o.xmlData){
11658                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11659                 p = '';
11660             }
11661             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11662             return this.transId;
11663         }else{
11664             Roo.callback(o.callback, o.scope, [o, null, null]);
11665             return null;
11666         }
11667     },
11668
11669     /**
11670      * Determine whether this object has a request outstanding.
11671      * @param {Number} transactionId (Optional) defaults to the last transaction
11672      * @return {Boolean} True if there is an outstanding request.
11673      */
11674     isLoading : function(transId){
11675         if(transId){
11676             return Roo.lib.Ajax.isCallInProgress(transId);
11677         }else{
11678             return this.transId ? true : false;
11679         }
11680     },
11681
11682     /**
11683      * Aborts any outstanding request.
11684      * @param {Number} transactionId (Optional) defaults to the last transaction
11685      */
11686     abort : function(transId){
11687         if(transId || this.isLoading()){
11688             Roo.lib.Ajax.abort(transId || this.transId);
11689         }
11690     },
11691
11692     // private
11693     handleResponse : function(response){
11694         this.transId = false;
11695         var options = response.argument.options;
11696         response.argument = options ? options.argument : null;
11697         this.fireEvent("requestcomplete", this, response, options);
11698         Roo.callback(options.success, options.scope, [response, options]);
11699         Roo.callback(options.callback, options.scope, [options, true, response]);
11700     },
11701
11702     // private
11703     handleFailure : function(response, e){
11704         this.transId = false;
11705         var options = response.argument.options;
11706         response.argument = options ? options.argument : null;
11707         this.fireEvent("requestexception", this, response, options, e);
11708         Roo.callback(options.failure, options.scope, [response, options]);
11709         Roo.callback(options.callback, options.scope, [options, false, response]);
11710     },
11711
11712     // private
11713     doFormUpload : function(o, ps, url){
11714         var id = Roo.id();
11715         var frame = document.createElement('iframe');
11716         frame.id = id;
11717         frame.name = id;
11718         frame.className = 'x-hidden';
11719         if(Roo.isIE){
11720             frame.src = Roo.SSL_SECURE_URL;
11721         }
11722         document.body.appendChild(frame);
11723
11724         if(Roo.isIE){
11725            document.frames[id].name = id;
11726         }
11727
11728         var form = Roo.getDom(o.form);
11729         form.target = id;
11730         form.method = 'POST';
11731         form.enctype = form.encoding = 'multipart/form-data';
11732         if(url){
11733             form.action = url;
11734         }
11735
11736         var hiddens, hd;
11737         if(ps){ // add dynamic params
11738             hiddens = [];
11739             ps = Roo.urlDecode(ps, false);
11740             for(var k in ps){
11741                 if(ps.hasOwnProperty(k)){
11742                     hd = document.createElement('input');
11743                     hd.type = 'hidden';
11744                     hd.name = k;
11745                     hd.value = ps[k];
11746                     form.appendChild(hd);
11747                     hiddens.push(hd);
11748                 }
11749             }
11750         }
11751
11752         function cb(){
11753             var r = {  // bogus response object
11754                 responseText : '',
11755                 responseXML : null
11756             };
11757
11758             r.argument = o ? o.argument : null;
11759
11760             try { //
11761                 var doc;
11762                 if(Roo.isIE){
11763                     doc = frame.contentWindow.document;
11764                 }else {
11765                     doc = (frame.contentDocument || window.frames[id].document);
11766                 }
11767                 if(doc && doc.body){
11768                     r.responseText = doc.body.innerHTML;
11769                 }
11770                 if(doc && doc.XMLDocument){
11771                     r.responseXML = doc.XMLDocument;
11772                 }else {
11773                     r.responseXML = doc;
11774                 }
11775             }
11776             catch(e) {
11777                 // ignore
11778             }
11779
11780             Roo.EventManager.removeListener(frame, 'load', cb, this);
11781
11782             this.fireEvent("requestcomplete", this, r, o);
11783             Roo.callback(o.success, o.scope, [r, o]);
11784             Roo.callback(o.callback, o.scope, [o, true, r]);
11785
11786             setTimeout(function(){document.body.removeChild(frame);}, 100);
11787         }
11788
11789         Roo.EventManager.on(frame, 'load', cb, this);
11790         form.submit();
11791
11792         if(hiddens){ // remove dynamic params
11793             for(var i = 0, len = hiddens.length; i < len; i++){
11794                 form.removeChild(hiddens[i]);
11795             }
11796         }
11797     },
11798     // this is a 'formdata version???'
11799     
11800     
11801     doFormDataUpload : function(o, ps, url)
11802     {
11803         var form = Roo.getDom(o.form);
11804         form.enctype = form.encoding = 'multipart/form-data';
11805         var formData = o.formData === true ? new FormData(form) : o.formData;
11806       
11807         var cb = {
11808             success: this.handleResponse,
11809             failure: this.handleFailure,
11810             scope: this,
11811             argument: {options: o},
11812             timeout : o.timeout || this.timeout
11813         };
11814  
11815         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11816             if(o.autoAbort){
11817                 this.abort();
11818             }
11819         }else if(this.autoAbort !== false){
11820             this.abort();
11821         }
11822
11823         //Roo.lib.Ajax.defaultPostHeader = null;
11824         Roo.lib.Ajax.useDefaultHeader = false;
11825         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11826         Roo.lib.Ajax.useDefaultHeader = true;
11827  
11828          
11829     }
11830     
11831 });
11832 /*
11833  * Based on:
11834  * Ext JS Library 1.1.1
11835  * Copyright(c) 2006-2007, Ext JS, LLC.
11836  *
11837  * Originally Released Under LGPL - original licence link has changed is not relivant.
11838  *
11839  * Fork - LGPL
11840  * <script type="text/javascript">
11841  */
11842  
11843 /**
11844  * Global Ajax request class.
11845  * 
11846  * @class Roo.Ajax
11847  * @extends Roo.data.Connection
11848  * @static
11849  * 
11850  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11851  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11852  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11853  * @cfg {String} method (Optional)  The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
11854  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11855  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11856  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11857  */
11858 Roo.Ajax = new Roo.data.Connection({
11859     // fix up the docs
11860     /**
11861      * @scope Roo.Ajax
11862      * @type {Boolear} 
11863      */
11864     autoAbort : false,
11865
11866     /**
11867      * Serialize the passed form into a url encoded string
11868      * @scope Roo.Ajax
11869      * @param {String/HTMLElement} form
11870      * @return {String}
11871      */
11872     serializeForm : function(form){
11873         return Roo.lib.Ajax.serializeForm(form);
11874     }
11875 });/*
11876  * Based on:
11877  * Ext JS Library 1.1.1
11878  * Copyright(c) 2006-2007, Ext JS, LLC.
11879  *
11880  * Originally Released Under LGPL - original licence link has changed is not relivant.
11881  *
11882  * Fork - LGPL
11883  * <script type="text/javascript">
11884  */
11885
11886  
11887 /**
11888  * @class Roo.UpdateManager
11889  * @extends Roo.util.Observable
11890  * Provides AJAX-style update for Element object.<br><br>
11891  * Usage:<br>
11892  * <pre><code>
11893  * // Get it from a Roo.Element object
11894  * var el = Roo.get("foo");
11895  * var mgr = el.getUpdateManager();
11896  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11897  * ...
11898  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11899  * <br>
11900  * // or directly (returns the same UpdateManager instance)
11901  * var mgr = new Roo.UpdateManager("myElementId");
11902  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11903  * mgr.on("update", myFcnNeedsToKnow);
11904  * <br>
11905    // short handed call directly from the element object
11906    Roo.get("foo").load({
11907         url: "bar.php",
11908         scripts:true,
11909         params: "for=bar",
11910         text: "Loading Foo..."
11911    });
11912  * </code></pre>
11913  * @constructor
11914  * Create new UpdateManager directly.
11915  * @param {String/HTMLElement/Roo.Element} el The element to update
11916  * @param {Boolean} forceNew (optional) By default the constructor checks to see if the passed element already has an UpdateManager and if it does it returns the same instance. This will skip that check (useful for extending this class).
11917  */
11918 Roo.UpdateManager = function(el, forceNew){
11919     el = Roo.get(el);
11920     if(!forceNew && el.updateManager){
11921         return el.updateManager;
11922     }
11923     /**
11924      * The Element object
11925      * @type Roo.Element
11926      */
11927     this.el = el;
11928     /**
11929      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11930      * @type String
11931      */
11932     this.defaultUrl = null;
11933
11934     this.addEvents({
11935         /**
11936          * @event beforeupdate
11937          * Fired before an update is made, return false from your handler and the update is cancelled.
11938          * @param {Roo.Element} el
11939          * @param {String/Object/Function} url
11940          * @param {String/Object} params
11941          */
11942         "beforeupdate": true,
11943         /**
11944          * @event update
11945          * Fired after successful update is made.
11946          * @param {Roo.Element} el
11947          * @param {Object} oResponseObject The response Object
11948          */
11949         "update": true,
11950         /**
11951          * @event failure
11952          * Fired on update failure.
11953          * @param {Roo.Element} el
11954          * @param {Object} oResponseObject The response Object
11955          */
11956         "failure": true
11957     });
11958     var d = Roo.UpdateManager.defaults;
11959     /**
11960      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11961      * @type String
11962      */
11963     this.sslBlankUrl = d.sslBlankUrl;
11964     /**
11965      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11966      * @type Boolean
11967      */
11968     this.disableCaching = d.disableCaching;
11969     /**
11970      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11971      * @type String
11972      */
11973     this.indicatorText = d.indicatorText;
11974     /**
11975      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11976      * @type String
11977      */
11978     this.showLoadIndicator = d.showLoadIndicator;
11979     /**
11980      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11981      * @type Number
11982      */
11983     this.timeout = d.timeout;
11984
11985     /**
11986      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11987      * @type Boolean
11988      */
11989     this.loadScripts = d.loadScripts;
11990
11991     /**
11992      * Transaction object of current executing transaction
11993      */
11994     this.transaction = null;
11995
11996     /**
11997      * @private
11998      */
11999     this.autoRefreshProcId = null;
12000     /**
12001      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12002      * @type Function
12003      */
12004     this.refreshDelegate = this.refresh.createDelegate(this);
12005     /**
12006      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12007      * @type Function
12008      */
12009     this.updateDelegate = this.update.createDelegate(this);
12010     /**
12011      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12012      * @type Function
12013      */
12014     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12015     /**
12016      * @private
12017      */
12018     this.successDelegate = this.processSuccess.createDelegate(this);
12019     /**
12020      * @private
12021      */
12022     this.failureDelegate = this.processFailure.createDelegate(this);
12023
12024     if(!this.renderer){
12025      /**
12026       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12027       */
12028     this.renderer = new Roo.UpdateManager.BasicRenderer();
12029     }
12030     
12031     Roo.UpdateManager.superclass.constructor.call(this);
12032 };
12033
12034 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12035     /**
12036      * Get the Element this UpdateManager is bound to
12037      * @return {Roo.Element} The element
12038      */
12039     getEl : function(){
12040         return this.el;
12041     },
12042     /**
12043      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12044      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
12045 <pre><code>
12046 um.update({<br/>
12047     url: "your-url.php",<br/>
12048     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12049     callback: yourFunction,<br/>
12050     scope: yourObject, //(optional scope)  <br/>
12051     discardUrl: false, <br/>
12052     nocache: false,<br/>
12053     text: "Loading...",<br/>
12054     timeout: 30,<br/>
12055     scripts: false<br/>
12056 });
12057 </code></pre>
12058      * The only required property is url. The optional properties nocache, text and scripts
12059      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12060      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
12061      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12062      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used url. If true, it will not store the url.
12063      */
12064     update : function(url, params, callback, discardUrl){
12065         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12066             var method = this.method,
12067                 cfg;
12068             if(typeof url == "object"){ // must be config object
12069                 cfg = url;
12070                 url = cfg.url;
12071                 params = params || cfg.params;
12072                 callback = callback || cfg.callback;
12073                 discardUrl = discardUrl || cfg.discardUrl;
12074                 if(callback && cfg.scope){
12075                     callback = callback.createDelegate(cfg.scope);
12076                 }
12077                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12078                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12079                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12080                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12081                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12082             }
12083             this.showLoading();
12084             if(!discardUrl){
12085                 this.defaultUrl = url;
12086             }
12087             if(typeof url == "function"){
12088                 url = url.call(this);
12089             }
12090
12091             method = method || (params ? "POST" : "GET");
12092             if(method == "GET"){
12093                 url = this.prepareUrl(url);
12094             }
12095
12096             var o = Roo.apply(cfg ||{}, {
12097                 url : url,
12098                 params: params,
12099                 success: this.successDelegate,
12100                 failure: this.failureDelegate,
12101                 callback: undefined,
12102                 timeout: (this.timeout*1000),
12103                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12104             });
12105             Roo.log("updated manager called with timeout of " + o.timeout);
12106             this.transaction = Roo.Ajax.request(o);
12107         }
12108     },
12109
12110     /**
12111      * Performs an async form post, updating this element with the response. If the form has the attribute enctype="multipart/form-data", it assumes it's a file upload.
12112      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12113      * @param {String/HTMLElement} form The form Id or form element
12114      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12115      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12116      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12117      */
12118     formUpdate : function(form, url, reset, callback){
12119         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12120             if(typeof url == "function"){
12121                 url = url.call(this);
12122             }
12123             form = Roo.getDom(form);
12124             this.transaction = Roo.Ajax.request({
12125                 form: form,
12126                 url:url,
12127                 success: this.successDelegate,
12128                 failure: this.failureDelegate,
12129                 timeout: (this.timeout*1000),
12130                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12131             });
12132             this.showLoading.defer(1, this);
12133         }
12134     },
12135
12136     /**
12137      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12138      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12139      */
12140     refresh : function(callback){
12141         if(this.defaultUrl == null){
12142             return;
12143         }
12144         this.update(this.defaultUrl, null, callback, true);
12145     },
12146
12147     /**
12148      * Set this element to auto refresh.
12149      * @param {Number} interval How often to update (in seconds).
12150      * @param {String/Function} url (optional) The url for this request or a function to call to get the url (Defaults to the last used url)
12151      * @param {String/Object} params (optional) The parameters to pass as either a url encoded string "&param1=1&param2=2" or as an object {param1: 1, param2: 2}
12152      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12153      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12154      */
12155     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12156         if(refreshNow){
12157             this.update(url || this.defaultUrl, params, callback, true);
12158         }
12159         if(this.autoRefreshProcId){
12160             clearInterval(this.autoRefreshProcId);
12161         }
12162         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12163     },
12164
12165     /**
12166      * Stop auto refresh on this element.
12167      */
12168      stopAutoRefresh : function(){
12169         if(this.autoRefreshProcId){
12170             clearInterval(this.autoRefreshProcId);
12171             delete this.autoRefreshProcId;
12172         }
12173     },
12174
12175     isAutoRefreshing : function(){
12176        return this.autoRefreshProcId ? true : false;
12177     },
12178     /**
12179      * Called to update the element to "Loading" state. Override to perform custom action.
12180      */
12181     showLoading : function(){
12182         if(this.showLoadIndicator){
12183             this.el.update(this.indicatorText);
12184         }
12185     },
12186
12187     /**
12188      * Adds unique parameter to query string if disableCaching = true
12189      * @private
12190      */
12191     prepareUrl : function(url){
12192         if(this.disableCaching){
12193             var append = "_dc=" + (new Date().getTime());
12194             if(url.indexOf("?") !== -1){
12195                 url += "&" + append;
12196             }else{
12197                 url += "?" + append;
12198             }
12199         }
12200         return url;
12201     },
12202
12203     /**
12204      * @private
12205      */
12206     processSuccess : function(response){
12207         this.transaction = null;
12208         if(response.argument.form && response.argument.reset){
12209             try{ // put in try/catch since some older FF releases had problems with this
12210                 response.argument.form.reset();
12211             }catch(e){}
12212         }
12213         if(this.loadScripts){
12214             this.renderer.render(this.el, response, this,
12215                 this.updateComplete.createDelegate(this, [response]));
12216         }else{
12217             this.renderer.render(this.el, response, this);
12218             this.updateComplete(response);
12219         }
12220     },
12221
12222     updateComplete : function(response){
12223         this.fireEvent("update", this.el, response);
12224         if(typeof response.argument.callback == "function"){
12225             response.argument.callback(this.el, true, response);
12226         }
12227     },
12228
12229     /**
12230      * @private
12231      */
12232     processFailure : function(response){
12233         this.transaction = null;
12234         this.fireEvent("failure", this.el, response);
12235         if(typeof response.argument.callback == "function"){
12236             response.argument.callback(this.el, false, response);
12237         }
12238     },
12239
12240     /**
12241      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12242      * @param {Object} renderer The object implementing the render() method
12243      */
12244     setRenderer : function(renderer){
12245         this.renderer = renderer;
12246     },
12247
12248     getRenderer : function(){
12249        return this.renderer;
12250     },
12251
12252     /**
12253      * Set the defaultUrl used for updates
12254      * @param {String/Function} defaultUrl The url or a function to call to get the url
12255      */
12256     setDefaultUrl : function(defaultUrl){
12257         this.defaultUrl = defaultUrl;
12258     },
12259
12260     /**
12261      * Aborts the executing transaction
12262      */
12263     abort : function(){
12264         if(this.transaction){
12265             Roo.Ajax.abort(this.transaction);
12266         }
12267     },
12268
12269     /**
12270      * Returns true if an update is in progress
12271      * @return {Boolean}
12272      */
12273     isUpdating : function(){
12274         if(this.transaction){
12275             return Roo.Ajax.isLoading(this.transaction);
12276         }
12277         return false;
12278     }
12279 });
12280
12281 /**
12282  * @class Roo.UpdateManager.defaults
12283  * @static (not really - but it helps the doc tool)
12284  * The defaults collection enables customizing the default properties of UpdateManager
12285  */
12286    Roo.UpdateManager.defaults = {
12287        /**
12288          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12289          * @type Number
12290          */
12291          timeout : 30,
12292
12293          /**
12294          * True to process scripts by default (Defaults to false).
12295          * @type Boolean
12296          */
12297         loadScripts : false,
12298
12299         /**
12300         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12301         * @type String
12302         */
12303         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12304         /**
12305          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12306          * @type Boolean
12307          */
12308         disableCaching : false,
12309         /**
12310          * Whether to show indicatorText when loading (Defaults to true).
12311          * @type Boolean
12312          */
12313         showLoadIndicator : true,
12314         /**
12315          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12316          * @type String
12317          */
12318         indicatorText : '<div class="loading-indicator">Loading...</div>'
12319    };
12320
12321 /**
12322  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12323  *Usage:
12324  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12325  * @param {String/HTMLElement/Roo.Element} el The element to update
12326  * @param {String} url The url
12327  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12328  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12329  * @static
12330  * @deprecated
12331  * @member Roo.UpdateManager
12332  */
12333 Roo.UpdateManager.updateElement = function(el, url, params, options){
12334     var um = Roo.get(el, true).getUpdateManager();
12335     Roo.apply(um, options);
12336     um.update(url, params, options ? options.callback : null);
12337 };
12338 // alias for backwards compat
12339 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12340 /**
12341  * @class Roo.UpdateManager.BasicRenderer
12342  * Default Content renderer. Updates the elements innerHTML with the responseText.
12343  */
12344 Roo.UpdateManager.BasicRenderer = function(){};
12345
12346 Roo.UpdateManager.BasicRenderer.prototype = {
12347     /**
12348      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12349      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12350      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12351      * @param {Roo.Element} el The element being rendered
12352      * @param {Object} response The YUI Connect response object
12353      * @param {UpdateManager} updateManager The calling update manager
12354      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12355      */
12356      render : function(el, response, updateManager, callback){
12357         el.update(response.responseText, updateManager.loadScripts, callback);
12358     }
12359 };
12360 /*
12361  * Based on:
12362  * Roo JS
12363  * (c)) Alan Knowles
12364  * Licence : LGPL
12365  */
12366
12367
12368 /**
12369  * @class Roo.DomTemplate
12370  * @extends Roo.Template
12371  * An effort at a dom based template engine..
12372  *
12373  * Similar to XTemplate, except it uses dom parsing to create the template..
12374  *
12375  * Supported features:
12376  *
12377  *  Tags:
12378
12379 <pre><code>
12380       {a_variable} - output encoded.
12381       {a_variable.format:("Y-m-d")} - call a method on the variable
12382       {a_variable:raw} - unencoded output
12383       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12384       {a_variable:this.method_on_template(...)} - call a method on the template object.
12385  
12386 </code></pre>
12387  *  The tpl tag:
12388 <pre><code>
12389         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12390         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12391         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12392         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12393   
12394 </code></pre>
12395  *      
12396  */
12397 Roo.DomTemplate = function()
12398 {
12399      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12400      if (this.html) {
12401         this.compile();
12402      }
12403 };
12404
12405
12406 Roo.extend(Roo.DomTemplate, Roo.Template, {
12407     /**
12408      * id counter for sub templates.
12409      */
12410     id : 0,
12411     /**
12412      * flag to indicate if dom parser is inside a pre,
12413      * it will strip whitespace if not.
12414      */
12415     inPre : false,
12416     
12417     /**
12418      * The various sub templates
12419      */
12420     tpls : false,
12421     
12422     
12423     
12424     /**
12425      *
12426      * basic tag replacing syntax
12427      * WORD:WORD()
12428      *
12429      * // you can fake an object call by doing this
12430      *  x.t:(test,tesT) 
12431      * 
12432      */
12433     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12434     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12435     
12436     iterChild : function (node, method) {
12437         
12438         var oldPre = this.inPre;
12439         if (node.tagName == 'PRE') {
12440             this.inPre = true;
12441         }
12442         for( var i = 0; i < node.childNodes.length; i++) {
12443             method.call(this, node.childNodes[i]);
12444         }
12445         this.inPre = oldPre;
12446     },
12447     
12448     
12449     
12450     /**
12451      * compile the template
12452      *
12453      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12454      *
12455      */
12456     compile: function()
12457     {
12458         var s = this.html;
12459         
12460         // covert the html into DOM...
12461         var doc = false;
12462         var div =false;
12463         try {
12464             doc = document.implementation.createHTMLDocument("");
12465             doc.documentElement.innerHTML =   this.html  ;
12466             div = doc.documentElement;
12467         } catch (e) {
12468             // old IE... - nasty -- it causes all sorts of issues.. with
12469             // images getting pulled from server..
12470             div = document.createElement('div');
12471             div.innerHTML = this.html;
12472         }
12473         //doc.documentElement.innerHTML = htmlBody
12474          
12475         
12476         
12477         this.tpls = [];
12478         var _t = this;
12479         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12480         
12481         var tpls = this.tpls;
12482         
12483         // create a top level template from the snippet..
12484         
12485         //Roo.log(div.innerHTML);
12486         
12487         var tpl = {
12488             uid : 'master',
12489             id : this.id++,
12490             attr : false,
12491             value : false,
12492             body : div.innerHTML,
12493             
12494             forCall : false,
12495             execCall : false,
12496             dom : div,
12497             isTop : true
12498             
12499         };
12500         tpls.unshift(tpl);
12501         
12502         
12503         // compile them...
12504         this.tpls = [];
12505         Roo.each(tpls, function(tp){
12506             this.compileTpl(tp);
12507             this.tpls[tp.id] = tp;
12508         }, this);
12509         
12510         this.master = tpls[0];
12511         return this;
12512         
12513         
12514     },
12515     
12516     compileNode : function(node, istop) {
12517         // test for
12518         //Roo.log(node);
12519         
12520         
12521         // skip anything not a tag..
12522         if (node.nodeType != 1) {
12523             if (node.nodeType == 3 && !this.inPre) {
12524                 // reduce white space..
12525                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12526                 
12527             }
12528             return;
12529         }
12530         
12531         var tpl = {
12532             uid : false,
12533             id : false,
12534             attr : false,
12535             value : false,
12536             body : '',
12537             
12538             forCall : false,
12539             execCall : false,
12540             dom : false,
12541             isTop : istop
12542             
12543             
12544         };
12545         
12546         
12547         switch(true) {
12548             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12549             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12550             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12551             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12552             // no default..
12553         }
12554         
12555         
12556         if (!tpl.attr) {
12557             // just itterate children..
12558             this.iterChild(node,this.compileNode);
12559             return;
12560         }
12561         tpl.uid = this.id++;
12562         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12563         node.removeAttribute('roo-'+ tpl.attr);
12564         if (tpl.attr != 'name') {
12565             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12566             node.parentNode.replaceChild(placeholder,  node);
12567         } else {
12568             
12569             var placeholder =  document.createElement('span');
12570             placeholder.className = 'roo-tpl-' + tpl.value;
12571             node.parentNode.replaceChild(placeholder,  node);
12572         }
12573         
12574         // parent now sees '{domtplXXXX}
12575         this.iterChild(node,this.compileNode);
12576         
12577         // we should now have node body...
12578         var div = document.createElement('div');
12579         div.appendChild(node);
12580         tpl.dom = node;
12581         // this has the unfortunate side effect of converting tagged attributes
12582         // eg. href="{...}" into %7C...%7D
12583         // this has been fixed by searching for those combo's although it's a bit hacky..
12584         
12585         
12586         tpl.body = div.innerHTML;
12587         
12588         
12589          
12590         tpl.id = tpl.uid;
12591         switch(tpl.attr) {
12592             case 'for' :
12593                 switch (tpl.value) {
12594                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12595                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12596                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12597                 }
12598                 break;
12599             
12600             case 'exec':
12601                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12602                 break;
12603             
12604             case 'if':     
12605                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12606                 break;
12607             
12608             case 'name':
12609                 tpl.id  = tpl.value; // replace non characters???
12610                 break;
12611             
12612         }
12613         
12614         
12615         this.tpls.push(tpl);
12616         
12617         
12618         
12619     },
12620     
12621     
12622     
12623     
12624     /**
12625      * Compile a segment of the template into a 'sub-template'
12626      *
12627      * 
12628      * 
12629      *
12630      */
12631     compileTpl : function(tpl)
12632     {
12633         var fm = Roo.util.Format;
12634         var useF = this.disableFormats !== true;
12635         
12636         var sep = Roo.isGecko ? "+\n" : ",\n";
12637         
12638         var undef = function(str) {
12639             Roo.debug && Roo.log("Property not found :"  + str);
12640             return '';
12641         };
12642           
12643         //Roo.log(tpl.body);
12644         
12645         
12646         
12647         var fn = function(m, lbrace, name, format, args)
12648         {
12649             //Roo.log("ARGS");
12650             //Roo.log(arguments);
12651             args = args ? args.replace(/\\'/g,"'") : args;
12652             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12653             if (typeof(format) == 'undefined') {
12654                 format =  'htmlEncode'; 
12655             }
12656             if (format == 'raw' ) {
12657                 format = false;
12658             }
12659             
12660             if(name.substr(0, 6) == 'domtpl'){
12661                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12662             }
12663             
12664             // build an array of options to determine if value is undefined..
12665             
12666             // basically get 'xxxx.yyyy' then do
12667             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12668             //    (function () { Roo.log("Property not found"); return ''; })() :
12669             //    ......
12670             
12671             var udef_ar = [];
12672             var lookfor = '';
12673             Roo.each(name.split('.'), function(st) {
12674                 lookfor += (lookfor.length ? '.': '') + st;
12675                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12676             });
12677             
12678             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12679             
12680             
12681             if(format && useF){
12682                 
12683                 args = args ? ',' + args : "";
12684                  
12685                 if(format.substr(0, 5) != "this."){
12686                     format = "fm." + format + '(';
12687                 }else{
12688                     format = 'this.call("'+ format.substr(5) + '", ';
12689                     args = ", values";
12690                 }
12691                 
12692                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12693             }
12694              
12695             if (args && args.length) {
12696                 // called with xxyx.yuu:(test,test)
12697                 // change to ()
12698                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12699             }
12700             // raw.. - :raw modifier..
12701             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12702             
12703         };
12704         var body;
12705         // branched to use + in gecko and [].join() in others
12706         if(Roo.isGecko){
12707             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12708                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12709                     "';};};";
12710         }else{
12711             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12712             body.push(tpl.body.replace(/(\r\n|\n)/g,
12713                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12714             body.push("'].join('');};};");
12715             body = body.join('');
12716         }
12717         
12718         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12719        
12720         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12721         eval(body);
12722         
12723         return this;
12724     },
12725      
12726     /**
12727      * same as applyTemplate, except it's done to one of the subTemplates
12728      * when using named templates, you can do:
12729      *
12730      * var str = pl.applySubTemplate('your-name', values);
12731      *
12732      * 
12733      * @param {Number} id of the template
12734      * @param {Object} values to apply to template
12735      * @param {Object} parent (normaly the instance of this object)
12736      */
12737     applySubTemplate : function(id, values, parent)
12738     {
12739         
12740         
12741         var t = this.tpls[id];
12742         
12743         
12744         try { 
12745             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12746                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12747                 return '';
12748             }
12749         } catch(e) {
12750             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12751             Roo.log(values);
12752           
12753             return '';
12754         }
12755         try { 
12756             
12757             if(t.execCall && t.execCall.call(this, values, parent)){
12758                 return '';
12759             }
12760         } catch(e) {
12761             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12762             Roo.log(values);
12763             return '';
12764         }
12765         
12766         try {
12767             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12768             parent = t.target ? values : parent;
12769             if(t.forCall && vs instanceof Array){
12770                 var buf = [];
12771                 for(var i = 0, len = vs.length; i < len; i++){
12772                     try {
12773                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12774                     } catch (e) {
12775                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12776                         Roo.log(e.body);
12777                         //Roo.log(t.compiled);
12778                         Roo.log(vs[i]);
12779                     }   
12780                 }
12781                 return buf.join('');
12782             }
12783         } catch (e) {
12784             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12785             Roo.log(values);
12786             return '';
12787         }
12788         try {
12789             return t.compiled.call(this, vs, parent);
12790         } catch (e) {
12791             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12792             Roo.log(e.body);
12793             //Roo.log(t.compiled);
12794             Roo.log(values);
12795             return '';
12796         }
12797     },
12798
12799    
12800
12801     applyTemplate : function(values){
12802         return this.master.compiled.call(this, values, {});
12803         //var s = this.subs;
12804     },
12805
12806     apply : function(){
12807         return this.applyTemplate.apply(this, arguments);
12808     }
12809
12810  });
12811
12812 Roo.DomTemplate.from = function(el){
12813     el = Roo.getDom(el);
12814     return new Roo.Domtemplate(el.value || el.innerHTML);
12815 };/*
12816  * Based on:
12817  * Ext JS Library 1.1.1
12818  * Copyright(c) 2006-2007, Ext JS, LLC.
12819  *
12820  * Originally Released Under LGPL - original licence link has changed is not relivant.
12821  *
12822  * Fork - LGPL
12823  * <script type="text/javascript">
12824  */
12825
12826 /**
12827  * @class Roo.util.DelayedTask
12828  * Provides a convenient method of performing setTimeout where a new
12829  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12830  * You can use this class to buffer
12831  * the keypress events for a certain number of milliseconds, and perform only if they stop
12832  * for that amount of time.
12833  * @constructor The parameters to this constructor serve as defaults and are not required.
12834  * @param {Function} fn (optional) The default function to timeout
12835  * @param {Object} scope (optional) The default scope of that timeout
12836  * @param {Array} args (optional) The default Array of arguments
12837  */
12838 Roo.util.DelayedTask = function(fn, scope, args){
12839     var id = null, d, t;
12840
12841     var call = function(){
12842         var now = new Date().getTime();
12843         if(now - t >= d){
12844             clearInterval(id);
12845             id = null;
12846             fn.apply(scope, args || []);
12847         }
12848     };
12849     /**
12850      * Cancels any pending timeout and queues a new one
12851      * @param {Number} delay The milliseconds to delay
12852      * @param {Function} newFn (optional) Overrides function passed to constructor
12853      * @param {Object} newScope (optional) Overrides scope passed to constructor
12854      * @param {Array} newArgs (optional) Overrides args passed to constructor
12855      */
12856     this.delay = function(delay, newFn, newScope, newArgs){
12857         if(id && delay != d){
12858             this.cancel();
12859         }
12860         d = delay;
12861         t = new Date().getTime();
12862         fn = newFn || fn;
12863         scope = newScope || scope;
12864         args = newArgs || args;
12865         if(!id){
12866             id = setInterval(call, d);
12867         }
12868     };
12869
12870     /**
12871      * Cancel the last queued timeout
12872      */
12873     this.cancel = function(){
12874         if(id){
12875             clearInterval(id);
12876             id = null;
12877         }
12878     };
12879 };/*
12880  * Based on:
12881  * Ext JS Library 1.1.1
12882  * Copyright(c) 2006-2007, Ext JS, LLC.
12883  *
12884  * Originally Released Under LGPL - original licence link has changed is not relivant.
12885  *
12886  * Fork - LGPL
12887  * <script type="text/javascript">
12888  */
12889  
12890  
12891 Roo.util.TaskRunner = function(interval){
12892     interval = interval || 10;
12893     var tasks = [], removeQueue = [];
12894     var id = 0;
12895     var running = false;
12896
12897     var stopThread = function(){
12898         running = false;
12899         clearInterval(id);
12900         id = 0;
12901     };
12902
12903     var startThread = function(){
12904         if(!running){
12905             running = true;
12906             id = setInterval(runTasks, interval);
12907         }
12908     };
12909
12910     var removeTask = function(task){
12911         removeQueue.push(task);
12912         if(task.onStop){
12913             task.onStop();
12914         }
12915     };
12916
12917     var runTasks = function(){
12918         if(removeQueue.length > 0){
12919             for(var i = 0, len = removeQueue.length; i < len; i++){
12920                 tasks.remove(removeQueue[i]);
12921             }
12922             removeQueue = [];
12923             if(tasks.length < 1){
12924                 stopThread();
12925                 return;
12926             }
12927         }
12928         var now = new Date().getTime();
12929         for(var i = 0, len = tasks.length; i < len; ++i){
12930             var t = tasks[i];
12931             var itime = now - t.taskRunTime;
12932             if(t.interval <= itime){
12933                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12934                 t.taskRunTime = now;
12935                 if(rt === false || t.taskRunCount === t.repeat){
12936                     removeTask(t);
12937                     return;
12938                 }
12939             }
12940             if(t.duration && t.duration <= (now - t.taskStartTime)){
12941                 removeTask(t);
12942             }
12943         }
12944     };
12945
12946     /**
12947      * Queues a new task.
12948      * @param {Object} task
12949      */
12950     this.start = function(task){
12951         tasks.push(task);
12952         task.taskStartTime = new Date().getTime();
12953         task.taskRunTime = 0;
12954         task.taskRunCount = 0;
12955         startThread();
12956         return task;
12957     };
12958
12959     this.stop = function(task){
12960         removeTask(task);
12961         return task;
12962     };
12963
12964     this.stopAll = function(){
12965         stopThread();
12966         for(var i = 0, len = tasks.length; i < len; i++){
12967             if(tasks[i].onStop){
12968                 tasks[i].onStop();
12969             }
12970         }
12971         tasks = [];
12972         removeQueue = [];
12973     };
12974 };
12975
12976 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12977  * Based on:
12978  * Ext JS Library 1.1.1
12979  * Copyright(c) 2006-2007, Ext JS, LLC.
12980  *
12981  * Originally Released Under LGPL - original licence link has changed is not relivant.
12982  *
12983  * Fork - LGPL
12984  * <script type="text/javascript">
12985  */
12986
12987  
12988 /**
12989  * @class Roo.util.MixedCollection
12990  * @extends Roo.util.Observable
12991  * A Collection class that maintains both numeric indexes and keys and exposes events.
12992  * @constructor
12993  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
12994  * collection (defaults to false)
12995  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
12996  * and return the key value for that item.  This is used when available to look up the key on items that
12997  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
12998  * equivalent to providing an implementation for the {@link #getKey} method.
12999  */
13000 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13001     this.items = [];
13002     this.map = {};
13003     this.keys = [];
13004     this.length = 0;
13005     this.addEvents({
13006         /**
13007          * @event clear
13008          * Fires when the collection is cleared.
13009          */
13010         "clear" : true,
13011         /**
13012          * @event add
13013          * Fires when an item is added to the collection.
13014          * @param {Number} index The index at which the item was added.
13015          * @param {Object} o The item added.
13016          * @param {String} key The key associated with the added item.
13017          */
13018         "add" : true,
13019         /**
13020          * @event replace
13021          * Fires when an item is replaced in the collection.
13022          * @param {String} key he key associated with the new added.
13023          * @param {Object} old The item being replaced.
13024          * @param {Object} new The new item.
13025          */
13026         "replace" : true,
13027         /**
13028          * @event remove
13029          * Fires when an item is removed from the collection.
13030          * @param {Object} o The item being removed.
13031          * @param {String} key (optional) The key associated with the removed item.
13032          */
13033         "remove" : true,
13034         "sort" : true
13035     });
13036     this.allowFunctions = allowFunctions === true;
13037     if(keyFn){
13038         this.getKey = keyFn;
13039     }
13040     Roo.util.MixedCollection.superclass.constructor.call(this);
13041 };
13042
13043 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13044     allowFunctions : false,
13045     
13046 /**
13047  * Adds an item to the collection.
13048  * @param {String} key The key to associate with the item
13049  * @param {Object} o The item to add.
13050  * @return {Object} The item added.
13051  */
13052     add : function(key, o){
13053         if(arguments.length == 1){
13054             o = arguments[0];
13055             key = this.getKey(o);
13056         }
13057         if(typeof key == "undefined" || key === null){
13058             this.length++;
13059             this.items.push(o);
13060             this.keys.push(null);
13061         }else{
13062             var old = this.map[key];
13063             if(old){
13064                 return this.replace(key, o);
13065             }
13066             this.length++;
13067             this.items.push(o);
13068             this.map[key] = o;
13069             this.keys.push(key);
13070         }
13071         this.fireEvent("add", this.length-1, o, key);
13072         return o;
13073     },
13074        
13075 /**
13076   * MixedCollection has a generic way to fetch keys if you implement getKey.
13077 <pre><code>
13078 // normal way
13079 var mc = new Roo.util.MixedCollection();
13080 mc.add(someEl.dom.id, someEl);
13081 mc.add(otherEl.dom.id, otherEl);
13082 //and so on
13083
13084 // using getKey
13085 var mc = new Roo.util.MixedCollection();
13086 mc.getKey = function(el){
13087    return el.dom.id;
13088 };
13089 mc.add(someEl);
13090 mc.add(otherEl);
13091
13092 // or via the constructor
13093 var mc = new Roo.util.MixedCollection(false, function(el){
13094    return el.dom.id;
13095 });
13096 mc.add(someEl);
13097 mc.add(otherEl);
13098 </code></pre>
13099  * @param o {Object} The item for which to find the key.
13100  * @return {Object} The key for the passed item.
13101  */
13102     getKey : function(o){
13103          return o.id; 
13104     },
13105    
13106 /**
13107  * Replaces an item in the collection.
13108  * @param {String} key The key associated with the item to replace, or the item to replace.
13109  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13110  * @return {Object}  The new item.
13111  */
13112     replace : function(key, o){
13113         if(arguments.length == 1){
13114             o = arguments[0];
13115             key = this.getKey(o);
13116         }
13117         var old = this.item(key);
13118         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13119              return this.add(key, o);
13120         }
13121         var index = this.indexOfKey(key);
13122         this.items[index] = o;
13123         this.map[key] = o;
13124         this.fireEvent("replace", key, old, o);
13125         return o;
13126     },
13127    
13128 /**
13129  * Adds all elements of an Array or an Object to the collection.
13130  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13131  * an Array of values, each of which are added to the collection.
13132  */
13133     addAll : function(objs){
13134         if(arguments.length > 1 || objs instanceof Array){
13135             var args = arguments.length > 1 ? arguments : objs;
13136             for(var i = 0, len = args.length; i < len; i++){
13137                 this.add(args[i]);
13138             }
13139         }else{
13140             for(var key in objs){
13141                 if(this.allowFunctions || typeof objs[key] != "function"){
13142                     this.add(key, objs[key]);
13143                 }
13144             }
13145         }
13146     },
13147    
13148 /**
13149  * Executes the specified function once for every item in the collection, passing each
13150  * item as the first and only parameter. returning false from the function will stop the iteration.
13151  * @param {Function} fn The function to execute for each item.
13152  * @param {Object} scope (optional) The scope in which to execute the function.
13153  */
13154     each : function(fn, scope){
13155         var items = [].concat(this.items); // each safe for removal
13156         for(var i = 0, len = items.length; i < len; i++){
13157             if(fn.call(scope || items[i], items[i], i, len) === false){
13158                 break;
13159             }
13160         }
13161     },
13162    
13163 /**
13164  * Executes the specified function once for every key in the collection, passing each
13165  * key, and its associated item as the first two parameters.
13166  * @param {Function} fn The function to execute for each item.
13167  * @param {Object} scope (optional) The scope in which to execute the function.
13168  */
13169     eachKey : function(fn, scope){
13170         for(var i = 0, len = this.keys.length; i < len; i++){
13171             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13172         }
13173     },
13174    
13175 /**
13176  * Returns the first item in the collection which elicits a true return value from the
13177  * passed selection function.
13178  * @param {Function} fn The selection function to execute for each item.
13179  * @param {Object} scope (optional) The scope in which to execute the function.
13180  * @return {Object} The first item in the collection which returned true from the selection function.
13181  */
13182     find : function(fn, scope){
13183         for(var i = 0, len = this.items.length; i < len; i++){
13184             if(fn.call(scope || window, this.items[i], this.keys[i])){
13185                 return this.items[i];
13186             }
13187         }
13188         return null;
13189     },
13190    
13191 /**
13192  * Inserts an item at the specified index in the collection.
13193  * @param {Number} index The index to insert the item at.
13194  * @param {String} key The key to associate with the new item, or the item itself.
13195  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13196  * @return {Object} The item inserted.
13197  */
13198     insert : function(index, key, o){
13199         if(arguments.length == 2){
13200             o = arguments[1];
13201             key = this.getKey(o);
13202         }
13203         if(index >= this.length){
13204             return this.add(key, o);
13205         }
13206         this.length++;
13207         this.items.splice(index, 0, o);
13208         if(typeof key != "undefined" && key != null){
13209             this.map[key] = o;
13210         }
13211         this.keys.splice(index, 0, key);
13212         this.fireEvent("add", index, o, key);
13213         return o;
13214     },
13215    
13216 /**
13217  * Removed an item from the collection.
13218  * @param {Object} o The item to remove.
13219  * @return {Object} The item removed.
13220  */
13221     remove : function(o){
13222         return this.removeAt(this.indexOf(o));
13223     },
13224    
13225 /**
13226  * Remove an item from a specified index in the collection.
13227  * @param {Number} index The index within the collection of the item to remove.
13228  */
13229     removeAt : function(index){
13230         if(index < this.length && index >= 0){
13231             this.length--;
13232             var o = this.items[index];
13233             this.items.splice(index, 1);
13234             var key = this.keys[index];
13235             if(typeof key != "undefined"){
13236                 delete this.map[key];
13237             }
13238             this.keys.splice(index, 1);
13239             this.fireEvent("remove", o, key);
13240         }
13241     },
13242    
13243 /**
13244  * Removed an item associated with the passed key fom the collection.
13245  * @param {String} key The key of the item to remove.
13246  */
13247     removeKey : function(key){
13248         return this.removeAt(this.indexOfKey(key));
13249     },
13250    
13251 /**
13252  * Returns the number of items in the collection.
13253  * @return {Number} the number of items in the collection.
13254  */
13255     getCount : function(){
13256         return this.length; 
13257     },
13258    
13259 /**
13260  * Returns index within the collection of the passed Object.
13261  * @param {Object} o The item to find the index of.
13262  * @return {Number} index of the item.
13263  */
13264     indexOf : function(o){
13265         if(!this.items.indexOf){
13266             for(var i = 0, len = this.items.length; i < len; i++){
13267                 if(this.items[i] == o) {
13268                     return i;
13269                 }
13270             }
13271             return -1;
13272         }else{
13273             return this.items.indexOf(o);
13274         }
13275     },
13276    
13277 /**
13278  * Returns index within the collection of the passed key.
13279  * @param {String} key The key to find the index of.
13280  * @return {Number} index of the key.
13281  */
13282     indexOfKey : function(key){
13283         if(!this.keys.indexOf){
13284             for(var i = 0, len = this.keys.length; i < len; i++){
13285                 if(this.keys[i] == key) {
13286                     return i;
13287                 }
13288             }
13289             return -1;
13290         }else{
13291             return this.keys.indexOf(key);
13292         }
13293     },
13294    
13295 /**
13296  * Returns the item associated with the passed key OR index. Key has priority over index.
13297  * @param {String/Number} key The key or index of the item.
13298  * @return {Object} The item associated with the passed key.
13299  */
13300     item : function(key){
13301         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13302         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13303     },
13304     
13305 /**
13306  * Returns the item at the specified index.
13307  * @param {Number} index The index of the item.
13308  * @return {Object}
13309  */
13310     itemAt : function(index){
13311         return this.items[index];
13312     },
13313     
13314 /**
13315  * Returns the item associated with the passed key.
13316  * @param {String/Number} key The key of the item.
13317  * @return {Object} The item associated with the passed key.
13318  */
13319     key : function(key){
13320         return this.map[key];
13321     },
13322    
13323 /**
13324  * Returns true if the collection contains the passed Object as an item.
13325  * @param {Object} o  The Object to look for in the collection.
13326  * @return {Boolean} True if the collection contains the Object as an item.
13327  */
13328     contains : function(o){
13329         return this.indexOf(o) != -1;
13330     },
13331    
13332 /**
13333  * Returns true if the collection contains the passed Object as a key.
13334  * @param {String} key The key to look for in the collection.
13335  * @return {Boolean} True if the collection contains the Object as a key.
13336  */
13337     containsKey : function(key){
13338         return typeof this.map[key] != "undefined";
13339     },
13340    
13341 /**
13342  * Removes all items from the collection.
13343  */
13344     clear : function(){
13345         this.length = 0;
13346         this.items = [];
13347         this.keys = [];
13348         this.map = {};
13349         this.fireEvent("clear");
13350     },
13351    
13352 /**
13353  * Returns the first item in the collection.
13354  * @return {Object} the first item in the collection..
13355  */
13356     first : function(){
13357         return this.items[0]; 
13358     },
13359    
13360 /**
13361  * Returns the last item in the collection.
13362  * @return {Object} the last item in the collection..
13363  */
13364     last : function(){
13365         return this.items[this.length-1];   
13366     },
13367     
13368     _sort : function(property, dir, fn){
13369         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13370         fn = fn || function(a, b){
13371             return a-b;
13372         };
13373         var c = [], k = this.keys, items = this.items;
13374         for(var i = 0, len = items.length; i < len; i++){
13375             c[c.length] = {key: k[i], value: items[i], index: i};
13376         }
13377         c.sort(function(a, b){
13378             var v = fn(a[property], b[property]) * dsc;
13379             if(v == 0){
13380                 v = (a.index < b.index ? -1 : 1);
13381             }
13382             return v;
13383         });
13384         for(var i = 0, len = c.length; i < len; i++){
13385             items[i] = c[i].value;
13386             k[i] = c[i].key;
13387         }
13388         this.fireEvent("sort", this);
13389     },
13390     
13391     /**
13392      * Sorts this collection with the passed comparison function
13393      * @param {String} direction (optional) "ASC" or "DESC"
13394      * @param {Function} fn (optional) comparison function
13395      */
13396     sort : function(dir, fn){
13397         this._sort("value", dir, fn);
13398     },
13399     
13400     /**
13401      * Sorts this collection by keys
13402      * @param {String} direction (optional) "ASC" or "DESC"
13403      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13404      */
13405     keySort : function(dir, fn){
13406         this._sort("key", dir, fn || function(a, b){
13407             return String(a).toUpperCase()-String(b).toUpperCase();
13408         });
13409     },
13410     
13411     /**
13412      * Returns a range of items in this collection
13413      * @param {Number} startIndex (optional) defaults to 0
13414      * @param {Number} endIndex (optional) default to the last item
13415      * @return {Array} An array of items
13416      */
13417     getRange : function(start, end){
13418         var items = this.items;
13419         if(items.length < 1){
13420             return [];
13421         }
13422         start = start || 0;
13423         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13424         var r = [];
13425         if(start <= end){
13426             for(var i = start; i <= end; i++) {
13427                     r[r.length] = items[i];
13428             }
13429         }else{
13430             for(var i = start; i >= end; i--) {
13431                     r[r.length] = items[i];
13432             }
13433         }
13434         return r;
13435     },
13436         
13437     /**
13438      * Filter the <i>objects</i> in this collection by a specific property. 
13439      * Returns a new collection that has been filtered.
13440      * @param {String} property A property on your objects
13441      * @param {String/RegExp} value Either string that the property values 
13442      * should start with or a RegExp to test against the property
13443      * @return {MixedCollection} The new filtered collection
13444      */
13445     filter : function(property, value){
13446         if(!value.exec){ // not a regex
13447             value = String(value);
13448             if(value.length == 0){
13449                 return this.clone();
13450             }
13451             value = new RegExp("^" + Roo.escapeRe(value), "i");
13452         }
13453         return this.filterBy(function(o){
13454             return o && value.test(o[property]);
13455         });
13456         },
13457     
13458     /**
13459      * Filter by a function. * Returns a new collection that has been filtered.
13460      * The passed function will be called with each 
13461      * object in the collection. If the function returns true, the value is included 
13462      * otherwise it is filtered.
13463      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13464      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13465      * @return {MixedCollection} The new filtered collection
13466      */
13467     filterBy : function(fn, scope){
13468         var r = new Roo.util.MixedCollection();
13469         r.getKey = this.getKey;
13470         var k = this.keys, it = this.items;
13471         for(var i = 0, len = it.length; i < len; i++){
13472             if(fn.call(scope||this, it[i], k[i])){
13473                                 r.add(k[i], it[i]);
13474                         }
13475         }
13476         return r;
13477     },
13478     
13479     /**
13480      * Creates a duplicate of this collection
13481      * @return {MixedCollection}
13482      */
13483     clone : function(){
13484         var r = new Roo.util.MixedCollection();
13485         var k = this.keys, it = this.items;
13486         for(var i = 0, len = it.length; i < len; i++){
13487             r.add(k[i], it[i]);
13488         }
13489         r.getKey = this.getKey;
13490         return r;
13491     }
13492 });
13493 /**
13494  * Returns the item associated with the passed key or index.
13495  * @method
13496  * @param {String/Number} key The key or index of the item.
13497  * @return {Object} The item associated with the passed key.
13498  */
13499 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13500  * Based on:
13501  * Ext JS Library 1.1.1
13502  * Copyright(c) 2006-2007, Ext JS, LLC.
13503  *
13504  * Originally Released Under LGPL - original licence link has changed is not relivant.
13505  *
13506  * Fork - LGPL
13507  * <script type="text/javascript">
13508  */
13509 /**
13510  * @class Roo.util.JSON
13511  * Modified version of Douglas Crockford"s json.js that doesn"t
13512  * mess with the Object prototype 
13513  * http://www.json.org/js.html
13514  * @singleton
13515  */
13516 Roo.util.JSON = new (function(){
13517     var useHasOwn = {}.hasOwnProperty ? true : false;
13518     
13519     // crashes Safari in some instances
13520     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13521     
13522     var pad = function(n) {
13523         return n < 10 ? "0" + n : n;
13524     };
13525     
13526     var m = {
13527         "\b": '\\b',
13528         "\t": '\\t',
13529         "\n": '\\n',
13530         "\f": '\\f',
13531         "\r": '\\r',
13532         '"' : '\\"',
13533         "\\": '\\\\'
13534     };
13535
13536     var encodeString = function(s){
13537         if (/["\\\x00-\x1f]/.test(s)) {
13538             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13539                 var c = m[b];
13540                 if(c){
13541                     return c;
13542                 }
13543                 c = b.charCodeAt();
13544                 return "\\u00" +
13545                     Math.floor(c / 16).toString(16) +
13546                     (c % 16).toString(16);
13547             }) + '"';
13548         }
13549         return '"' + s + '"';
13550     };
13551     
13552     var encodeArray = function(o){
13553         var a = ["["], b, i, l = o.length, v;
13554             for (i = 0; i < l; i += 1) {
13555                 v = o[i];
13556                 switch (typeof v) {
13557                     case "undefined":
13558                     case "function":
13559                     case "unknown":
13560                         break;
13561                     default:
13562                         if (b) {
13563                             a.push(',');
13564                         }
13565                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13566                         b = true;
13567                 }
13568             }
13569             a.push("]");
13570             return a.join("");
13571     };
13572     
13573     var encodeDate = function(o){
13574         return '"' + o.getFullYear() + "-" +
13575                 pad(o.getMonth() + 1) + "-" +
13576                 pad(o.getDate()) + "T" +
13577                 pad(o.getHours()) + ":" +
13578                 pad(o.getMinutes()) + ":" +
13579                 pad(o.getSeconds()) + '"';
13580     };
13581     
13582     /**
13583      * Encodes an Object, Array or other value
13584      * @param {Mixed} o The variable to encode
13585      * @return {String} The JSON string
13586      */
13587     this.encode = function(o)
13588     {
13589         // should this be extended to fully wrap stringify..
13590         
13591         if(typeof o == "undefined" || o === null){
13592             return "null";
13593         }else if(o instanceof Array){
13594             return encodeArray(o);
13595         }else if(o instanceof Date){
13596             return encodeDate(o);
13597         }else if(typeof o == "string"){
13598             return encodeString(o);
13599         }else if(typeof o == "number"){
13600             return isFinite(o) ? String(o) : "null";
13601         }else if(typeof o == "boolean"){
13602             return String(o);
13603         }else {
13604             var a = ["{"], b, i, v;
13605             for (i in o) {
13606                 if(!useHasOwn || o.hasOwnProperty(i)) {
13607                     v = o[i];
13608                     switch (typeof v) {
13609                     case "undefined":
13610                     case "function":
13611                     case "unknown":
13612                         break;
13613                     default:
13614                         if(b){
13615                             a.push(',');
13616                         }
13617                         a.push(this.encode(i), ":",
13618                                 v === null ? "null" : this.encode(v));
13619                         b = true;
13620                     }
13621                 }
13622             }
13623             a.push("}");
13624             return a.join("");
13625         }
13626     };
13627     
13628     /**
13629      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13630      * @param {String} json The JSON string
13631      * @return {Object} The resulting object
13632      */
13633     this.decode = function(json){
13634         
13635         return  /** eval:var:json */ eval("(" + json + ')');
13636     };
13637 })();
13638 /** 
13639  * Shorthand for {@link Roo.util.JSON#encode}
13640  * @member Roo encode 
13641  * @method */
13642 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13643 /** 
13644  * Shorthand for {@link Roo.util.JSON#decode}
13645  * @member Roo decode 
13646  * @method */
13647 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13648 /*
13649  * Based on:
13650  * Ext JS Library 1.1.1
13651  * Copyright(c) 2006-2007, Ext JS, LLC.
13652  *
13653  * Originally Released Under LGPL - original licence link has changed is not relivant.
13654  *
13655  * Fork - LGPL
13656  * <script type="text/javascript">
13657  */
13658  
13659 /**
13660  * @class Roo.util.Format
13661  * Reusable data formatting functions
13662  * @singleton
13663  */
13664 Roo.util.Format = function(){
13665     var trimRe = /^\s+|\s+$/g;
13666     return {
13667         /**
13668          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13669          * @param {String} value The string to truncate
13670          * @param {Number} length The maximum length to allow before truncating
13671          * @return {String} The converted text
13672          */
13673         ellipsis : function(value, len){
13674             if(value && value.length > len){
13675                 return value.substr(0, len-3)+"...";
13676             }
13677             return value;
13678         },
13679
13680         /**
13681          * Checks a reference and converts it to empty string if it is undefined
13682          * @param {Mixed} value Reference to check
13683          * @return {Mixed} Empty string if converted, otherwise the original value
13684          */
13685         undef : function(value){
13686             return typeof value != "undefined" ? value : "";
13687         },
13688
13689         /**
13690          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13691          * @param {String} value The string to encode
13692          * @return {String} The encoded text
13693          */
13694         htmlEncode : function(value){
13695             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13696         },
13697
13698         /**
13699          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13700          * @param {String} value The string to decode
13701          * @return {String} The decoded text
13702          */
13703         htmlDecode : function(value){
13704             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13705         },
13706
13707         /**
13708          * Trims any whitespace from either side of a string
13709          * @param {String} value The text to trim
13710          * @return {String} The trimmed text
13711          */
13712         trim : function(value){
13713             return String(value).replace(trimRe, "");
13714         },
13715
13716         /**
13717          * Returns a substring from within an original string
13718          * @param {String} value The original text
13719          * @param {Number} start The start index of the substring
13720          * @param {Number} length The length of the substring
13721          * @return {String} The substring
13722          */
13723         substr : function(value, start, length){
13724             return String(value).substr(start, length);
13725         },
13726
13727         /**
13728          * Converts a string to all lower case letters
13729          * @param {String} value The text to convert
13730          * @return {String} The converted text
13731          */
13732         lowercase : function(value){
13733             return String(value).toLowerCase();
13734         },
13735
13736         /**
13737          * Converts a string to all upper case letters
13738          * @param {String} value The text to convert
13739          * @return {String} The converted text
13740          */
13741         uppercase : function(value){
13742             return String(value).toUpperCase();
13743         },
13744
13745         /**
13746          * Converts the first character only of a string to upper case
13747          * @param {String} value The text to convert
13748          * @return {String} The converted text
13749          */
13750         capitalize : function(value){
13751             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13752         },
13753
13754         // private
13755         call : function(value, fn){
13756             if(arguments.length > 2){
13757                 var args = Array.prototype.slice.call(arguments, 2);
13758                 args.unshift(value);
13759                  
13760                 return /** eval:var:value */  eval(fn).apply(window, args);
13761             }else{
13762                 /** eval:var:value */
13763                 return /** eval:var:value */ eval(fn).call(window, value);
13764             }
13765         },
13766
13767        
13768         /**
13769          * safer version of Math.toFixed..??/
13770          * @param {Number/String} value The numeric value to format
13771          * @param {Number/String} value Decimal places 
13772          * @return {String} The formatted currency string
13773          */
13774         toFixed : function(v, n)
13775         {
13776             // why not use to fixed - precision is buggered???
13777             if (!n) {
13778                 return Math.round(v-0);
13779             }
13780             var fact = Math.pow(10,n+1);
13781             v = (Math.round((v-0)*fact))/fact;
13782             var z = (''+fact).substring(2);
13783             if (v == Math.floor(v)) {
13784                 return Math.floor(v) + '.' + z;
13785             }
13786             
13787             // now just padd decimals..
13788             var ps = String(v).split('.');
13789             var fd = (ps[1] + z);
13790             var r = fd.substring(0,n); 
13791             var rm = fd.substring(n); 
13792             if (rm < 5) {
13793                 return ps[0] + '.' + r;
13794             }
13795             r*=1; // turn it into a number;
13796             r++;
13797             if (String(r).length != n) {
13798                 ps[0]*=1;
13799                 ps[0]++;
13800                 r = String(r).substring(1); // chop the end off.
13801             }
13802             
13803             return ps[0] + '.' + r;
13804              
13805         },
13806         
13807         /**
13808          * Format a number as US currency
13809          * @param {Number/String} value The numeric value to format
13810          * @return {String} The formatted currency string
13811          */
13812         usMoney : function(v){
13813             return '$' + Roo.util.Format.number(v);
13814         },
13815         
13816         /**
13817          * Format a number
13818          * eventually this should probably emulate php's number_format
13819          * @param {Number/String} value The numeric value to format
13820          * @param {Number} decimals number of decimal places
13821          * @param {String} delimiter for thousands (default comma)
13822          * @return {String} The formatted currency string
13823          */
13824         number : function(v, decimals, thousandsDelimiter)
13825         {
13826             // multiply and round.
13827             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13828             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13829             
13830             var mul = Math.pow(10, decimals);
13831             var zero = String(mul).substring(1);
13832             v = (Math.round((v-0)*mul))/mul;
13833             
13834             // if it's '0' number.. then
13835             
13836             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13837             v = String(v);
13838             var ps = v.split('.');
13839             var whole = ps[0];
13840             
13841             var r = /(\d+)(\d{3})/;
13842             // add comma's
13843             
13844             if(thousandsDelimiter.length != 0) {
13845                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13846             } 
13847             
13848             var sub = ps[1] ?
13849                     // has decimals..
13850                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13851                     // does not have decimals
13852                     (decimals ? ('.' + zero) : '');
13853             
13854             
13855             return whole + sub ;
13856         },
13857         
13858         /**
13859          * Parse a value into a formatted date using the specified format pattern.
13860          * @param {Mixed} value The value to format
13861          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13862          * @return {String} The formatted date string
13863          */
13864         date : function(v, format){
13865             if(!v){
13866                 return "";
13867             }
13868             if(!(v instanceof Date)){
13869                 v = new Date(Date.parse(v));
13870             }
13871             return v.dateFormat(format || Roo.util.Format.defaults.date);
13872         },
13873
13874         /**
13875          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13876          * @param {String} format Any valid date format string
13877          * @return {Function} The date formatting function
13878          */
13879         dateRenderer : function(format){
13880             return function(v){
13881                 return Roo.util.Format.date(v, format);  
13882             };
13883         },
13884
13885         // private
13886         stripTagsRE : /<\/?[^>]+>/gi,
13887         
13888         /**
13889          * Strips all HTML tags
13890          * @param {Mixed} value The text from which to strip tags
13891          * @return {String} The stripped text
13892          */
13893         stripTags : function(v){
13894             return !v ? v : String(v).replace(this.stripTagsRE, "");
13895         }
13896     };
13897 }();
13898 Roo.util.Format.defaults = {
13899     date : 'd/M/Y'
13900 };/*
13901  * Based on:
13902  * Ext JS Library 1.1.1
13903  * Copyright(c) 2006-2007, Ext JS, LLC.
13904  *
13905  * Originally Released Under LGPL - original licence link has changed is not relivant.
13906  *
13907  * Fork - LGPL
13908  * <script type="text/javascript">
13909  */
13910
13911
13912  
13913
13914 /**
13915  * @class Roo.MasterTemplate
13916  * @extends Roo.Template
13917  * Provides a template that can have child templates. The syntax is:
13918 <pre><code>
13919 var t = new Roo.MasterTemplate(
13920         '&lt;select name="{name}"&gt;',
13921                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13922         '&lt;/select&gt;'
13923 );
13924 t.add('options', {value: 'foo', text: 'bar'});
13925 // or you can add multiple child elements in one shot
13926 t.addAll('options', [
13927     {value: 'foo', text: 'bar'},
13928     {value: 'foo2', text: 'bar2'},
13929     {value: 'foo3', text: 'bar3'}
13930 ]);
13931 // then append, applying the master template values
13932 t.append('my-form', {name: 'my-select'});
13933 </code></pre>
13934 * A name attribute for the child template is not required if you have only one child
13935 * template or you want to refer to them by index.
13936  */
13937 Roo.MasterTemplate = function(){
13938     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13939     this.originalHtml = this.html;
13940     var st = {};
13941     var m, re = this.subTemplateRe;
13942     re.lastIndex = 0;
13943     var subIndex = 0;
13944     while(m = re.exec(this.html)){
13945         var name = m[1], content = m[2];
13946         st[subIndex] = {
13947             name: name,
13948             index: subIndex,
13949             buffer: [],
13950             tpl : new Roo.Template(content)
13951         };
13952         if(name){
13953             st[name] = st[subIndex];
13954         }
13955         st[subIndex].tpl.compile();
13956         st[subIndex].tpl.call = this.call.createDelegate(this);
13957         subIndex++;
13958     }
13959     this.subCount = subIndex;
13960     this.subs = st;
13961 };
13962 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13963     /**
13964     * The regular expression used to match sub templates
13965     * @type RegExp
13966     * @property
13967     */
13968     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13969
13970     /**
13971      * Applies the passed values to a child template.
13972      * @param {String/Number} name (optional) The name or index of the child template
13973      * @param {Array/Object} values The values to be applied to the template
13974      * @return {MasterTemplate} this
13975      */
13976      add : function(name, values){
13977         if(arguments.length == 1){
13978             values = arguments[0];
13979             name = 0;
13980         }
13981         var s = this.subs[name];
13982         s.buffer[s.buffer.length] = s.tpl.apply(values);
13983         return this;
13984     },
13985
13986     /**
13987      * Applies all the passed values to a child template.
13988      * @param {String/Number} name (optional) The name or index of the child template
13989      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13990      * @param {Boolean} reset (optional) True to reset the template first
13991      * @return {MasterTemplate} this
13992      */
13993     fill : function(name, values, reset){
13994         var a = arguments;
13995         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
13996             values = a[0];
13997             name = 0;
13998             reset = a[1];
13999         }
14000         if(reset){
14001             this.reset();
14002         }
14003         for(var i = 0, len = values.length; i < len; i++){
14004             this.add(name, values[i]);
14005         }
14006         return this;
14007     },
14008
14009     /**
14010      * Resets the template for reuse
14011      * @return {MasterTemplate} this
14012      */
14013      reset : function(){
14014         var s = this.subs;
14015         for(var i = 0; i < this.subCount; i++){
14016             s[i].buffer = [];
14017         }
14018         return this;
14019     },
14020
14021     applyTemplate : function(values){
14022         var s = this.subs;
14023         var replaceIndex = -1;
14024         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14025             return s[++replaceIndex].buffer.join("");
14026         });
14027         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14028     },
14029
14030     apply : function(){
14031         return this.applyTemplate.apply(this, arguments);
14032     },
14033
14034     compile : function(){return this;}
14035 });
14036
14037 /**
14038  * Alias for fill().
14039  * @method
14040  */
14041 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14042  /**
14043  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14044  * var tpl = Roo.MasterTemplate.from('element-id');
14045  * @param {String/HTMLElement} el
14046  * @param {Object} config
14047  * @static
14048  */
14049 Roo.MasterTemplate.from = function(el, config){
14050     el = Roo.getDom(el);
14051     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14052 };/*
14053  * Based on:
14054  * Ext JS Library 1.1.1
14055  * Copyright(c) 2006-2007, Ext JS, LLC.
14056  *
14057  * Originally Released Under LGPL - original licence link has changed is not relivant.
14058  *
14059  * Fork - LGPL
14060  * <script type="text/javascript">
14061  */
14062
14063  
14064 /**
14065  * @class Roo.util.CSS
14066  * Utility class for manipulating CSS rules
14067  * @singleton
14068  */
14069 Roo.util.CSS = function(){
14070         var rules = null;
14071         var doc = document;
14072
14073     var camelRe = /(-[a-z])/gi;
14074     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14075
14076    return {
14077    /**
14078     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14079     * tag and appended to the HEAD of the document.
14080     * @param {String|Object} cssText The text containing the css rules
14081     * @param {String} id An id to add to the stylesheet for later removal
14082     * @return {StyleSheet}
14083     */
14084     createStyleSheet : function(cssText, id){
14085         var ss;
14086         var head = doc.getElementsByTagName("head")[0];
14087         var nrules = doc.createElement("style");
14088         nrules.setAttribute("type", "text/css");
14089         if(id){
14090             nrules.setAttribute("id", id);
14091         }
14092         if (typeof(cssText) != 'string') {
14093             // support object maps..
14094             // not sure if this a good idea.. 
14095             // perhaps it should be merged with the general css handling
14096             // and handle js style props.
14097             var cssTextNew = [];
14098             for(var n in cssText) {
14099                 var citems = [];
14100                 for(var k in cssText[n]) {
14101                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14102                 }
14103                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14104                 
14105             }
14106             cssText = cssTextNew.join("\n");
14107             
14108         }
14109        
14110        
14111        if(Roo.isIE){
14112            head.appendChild(nrules);
14113            ss = nrules.styleSheet;
14114            ss.cssText = cssText;
14115        }else{
14116            try{
14117                 nrules.appendChild(doc.createTextNode(cssText));
14118            }catch(e){
14119                nrules.cssText = cssText; 
14120            }
14121            head.appendChild(nrules);
14122            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14123        }
14124        this.cacheStyleSheet(ss);
14125        return ss;
14126    },
14127
14128    /**
14129     * Removes a style or link tag by id
14130     * @param {String} id The id of the tag
14131     */
14132    removeStyleSheet : function(id){
14133        var existing = doc.getElementById(id);
14134        if(existing){
14135            existing.parentNode.removeChild(existing);
14136        }
14137    },
14138
14139    /**
14140     * Dynamically swaps an existing stylesheet reference for a new one
14141     * @param {String} id The id of an existing link tag to remove
14142     * @param {String} url The href of the new stylesheet to include
14143     */
14144    swapStyleSheet : function(id, url){
14145        this.removeStyleSheet(id);
14146        var ss = doc.createElement("link");
14147        ss.setAttribute("rel", "stylesheet");
14148        ss.setAttribute("type", "text/css");
14149        ss.setAttribute("id", id);
14150        ss.setAttribute("href", url);
14151        doc.getElementsByTagName("head")[0].appendChild(ss);
14152    },
14153    
14154    /**
14155     * Refresh the rule cache if you have dynamically added stylesheets
14156     * @return {Object} An object (hash) of rules indexed by selector
14157     */
14158    refreshCache : function(){
14159        return this.getRules(true);
14160    },
14161
14162    // private
14163    cacheStyleSheet : function(stylesheet){
14164        if(!rules){
14165            rules = {};
14166        }
14167        try{// try catch for cross domain access issue
14168            var ssRules = stylesheet.cssRules || stylesheet.rules;
14169            for(var j = ssRules.length-1; j >= 0; --j){
14170                rules[ssRules[j].selectorText] = ssRules[j];
14171            }
14172        }catch(e){}
14173    },
14174    
14175    /**
14176     * Gets all css rules for the document
14177     * @param {Boolean} refreshCache true to refresh the internal cache
14178     * @return {Object} An object (hash) of rules indexed by selector
14179     */
14180    getRules : function(refreshCache){
14181                 if(rules == null || refreshCache){
14182                         rules = {};
14183                         var ds = doc.styleSheets;
14184                         for(var i =0, len = ds.length; i < len; i++){
14185                             try{
14186                         this.cacheStyleSheet(ds[i]);
14187                     }catch(e){} 
14188                 }
14189                 }
14190                 return rules;
14191         },
14192         
14193         /**
14194     * Gets an an individual CSS rule by selector(s)
14195     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14196     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14197     * @return {CSSRule} The CSS rule or null if one is not found
14198     */
14199    getRule : function(selector, refreshCache){
14200                 var rs = this.getRules(refreshCache);
14201                 if(!(selector instanceof Array)){
14202                     return rs[selector];
14203                 }
14204                 for(var i = 0; i < selector.length; i++){
14205                         if(rs[selector[i]]){
14206                                 return rs[selector[i]];
14207                         }
14208                 }
14209                 return null;
14210         },
14211         
14212         
14213         /**
14214     * Updates a rule property
14215     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14216     * @param {String} property The css property
14217     * @param {String} value The new value for the property
14218     * @return {Boolean} true If a rule was found and updated
14219     */
14220    updateRule : function(selector, property, value){
14221                 if(!(selector instanceof Array)){
14222                         var rule = this.getRule(selector);
14223                         if(rule){
14224                                 rule.style[property.replace(camelRe, camelFn)] = value;
14225                                 return true;
14226                         }
14227                 }else{
14228                         for(var i = 0; i < selector.length; i++){
14229                                 if(this.updateRule(selector[i], property, value)){
14230                                         return true;
14231                                 }
14232                         }
14233                 }
14234                 return false;
14235         }
14236    };   
14237 }();/*
14238  * Based on:
14239  * Ext JS Library 1.1.1
14240  * Copyright(c) 2006-2007, Ext JS, LLC.
14241  *
14242  * Originally Released Under LGPL - original licence link has changed is not relivant.
14243  *
14244  * Fork - LGPL
14245  * <script type="text/javascript">
14246  */
14247
14248  
14249
14250 /**
14251  * @class Roo.util.ClickRepeater
14252  * @extends Roo.util.Observable
14253  * 
14254  * A wrapper class which can be applied to any element. Fires a "click" event while the
14255  * mouse is pressed. The interval between firings may be specified in the config but
14256  * defaults to 10 milliseconds.
14257  * 
14258  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14259  * 
14260  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14261  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14262  * Similar to an autorepeat key delay.
14263  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14264  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14265  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14266  *           "interval" and "delay" are ignored. "immediate" is honored.
14267  * @cfg {Boolean} preventDefault True to prevent the default click event
14268  * @cfg {Boolean} stopDefault True to stop the default click event
14269  * 
14270  * @history
14271  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14272  *     2007-02-02 jvs Renamed to ClickRepeater
14273  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14274  *
14275  *  @constructor
14276  * @param {String/HTMLElement/Element} el The element to listen on
14277  * @param {Object} config
14278  **/
14279 Roo.util.ClickRepeater = function(el, config)
14280 {
14281     this.el = Roo.get(el);
14282     this.el.unselectable();
14283
14284     Roo.apply(this, config);
14285
14286     this.addEvents({
14287     /**
14288      * @event mousedown
14289      * Fires when the mouse button is depressed.
14290      * @param {Roo.util.ClickRepeater} this
14291      */
14292         "mousedown" : true,
14293     /**
14294      * @event click
14295      * Fires on a specified interval during the time the element is pressed.
14296      * @param {Roo.util.ClickRepeater} this
14297      */
14298         "click" : true,
14299     /**
14300      * @event mouseup
14301      * Fires when the mouse key is released.
14302      * @param {Roo.util.ClickRepeater} this
14303      */
14304         "mouseup" : true
14305     });
14306
14307     this.el.on("mousedown", this.handleMouseDown, this);
14308     if(this.preventDefault || this.stopDefault){
14309         this.el.on("click", function(e){
14310             if(this.preventDefault){
14311                 e.preventDefault();
14312             }
14313             if(this.stopDefault){
14314                 e.stopEvent();
14315             }
14316         }, this);
14317     }
14318
14319     // allow inline handler
14320     if(this.handler){
14321         this.on("click", this.handler,  this.scope || this);
14322     }
14323
14324     Roo.util.ClickRepeater.superclass.constructor.call(this);
14325 };
14326
14327 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14328     interval : 20,
14329     delay: 250,
14330     preventDefault : true,
14331     stopDefault : false,
14332     timer : 0,
14333
14334     // private
14335     handleMouseDown : function(){
14336         clearTimeout(this.timer);
14337         this.el.blur();
14338         if(this.pressClass){
14339             this.el.addClass(this.pressClass);
14340         }
14341         this.mousedownTime = new Date();
14342
14343         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14344         this.el.on("mouseout", this.handleMouseOut, this);
14345
14346         this.fireEvent("mousedown", this);
14347         this.fireEvent("click", this);
14348         
14349         this.timer = this.click.defer(this.delay || this.interval, this);
14350     },
14351
14352     // private
14353     click : function(){
14354         this.fireEvent("click", this);
14355         this.timer = this.click.defer(this.getInterval(), this);
14356     },
14357
14358     // private
14359     getInterval: function(){
14360         if(!this.accelerate){
14361             return this.interval;
14362         }
14363         var pressTime = this.mousedownTime.getElapsed();
14364         if(pressTime < 500){
14365             return 400;
14366         }else if(pressTime < 1700){
14367             return 320;
14368         }else if(pressTime < 2600){
14369             return 250;
14370         }else if(pressTime < 3500){
14371             return 180;
14372         }else if(pressTime < 4400){
14373             return 140;
14374         }else if(pressTime < 5300){
14375             return 80;
14376         }else if(pressTime < 6200){
14377             return 50;
14378         }else{
14379             return 10;
14380         }
14381     },
14382
14383     // private
14384     handleMouseOut : function(){
14385         clearTimeout(this.timer);
14386         if(this.pressClass){
14387             this.el.removeClass(this.pressClass);
14388         }
14389         this.el.on("mouseover", this.handleMouseReturn, this);
14390     },
14391
14392     // private
14393     handleMouseReturn : function(){
14394         this.el.un("mouseover", this.handleMouseReturn);
14395         if(this.pressClass){
14396             this.el.addClass(this.pressClass);
14397         }
14398         this.click();
14399     },
14400
14401     // private
14402     handleMouseUp : function(){
14403         clearTimeout(this.timer);
14404         this.el.un("mouseover", this.handleMouseReturn);
14405         this.el.un("mouseout", this.handleMouseOut);
14406         Roo.get(document).un("mouseup", this.handleMouseUp);
14407         this.el.removeClass(this.pressClass);
14408         this.fireEvent("mouseup", this);
14409     }
14410 });/*
14411  * Based on:
14412  * Ext JS Library 1.1.1
14413  * Copyright(c) 2006-2007, Ext JS, LLC.
14414  *
14415  * Originally Released Under LGPL - original licence link has changed is not relivant.
14416  *
14417  * Fork - LGPL
14418  * <script type="text/javascript">
14419  */
14420
14421  
14422 /**
14423  * @class Roo.KeyNav
14424  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14425  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14426  * way to implement custom navigation schemes for any UI component.</p>
14427  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14428  * pageUp, pageDown, del, home, end.  Usage:</p>
14429  <pre><code>
14430 var nav = new Roo.KeyNav("my-element", {
14431     "left" : function(e){
14432         this.moveLeft(e.ctrlKey);
14433     },
14434     "right" : function(e){
14435         this.moveRight(e.ctrlKey);
14436     },
14437     "enter" : function(e){
14438         this.save();
14439     },
14440     scope : this
14441 });
14442 </code></pre>
14443  * @constructor
14444  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14445  * @param {Object} config The config
14446  */
14447 Roo.KeyNav = function(el, config){
14448     this.el = Roo.get(el);
14449     Roo.apply(this, config);
14450     if(!this.disabled){
14451         this.disabled = true;
14452         this.enable();
14453     }
14454 };
14455
14456 Roo.KeyNav.prototype = {
14457     /**
14458      * @cfg {Boolean} disabled
14459      * True to disable this KeyNav instance (defaults to false)
14460      */
14461     disabled : false,
14462     /**
14463      * @cfg {String} defaultEventAction
14464      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14465      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14466      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14467      */
14468     defaultEventAction: "stopEvent",
14469     /**
14470      * @cfg {Boolean} forceKeyDown
14471      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14472      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14473      * handle keydown instead of keypress.
14474      */
14475     forceKeyDown : false,
14476
14477     // private
14478     prepareEvent : function(e){
14479         var k = e.getKey();
14480         var h = this.keyToHandler[k];
14481         //if(h && this[h]){
14482         //    e.stopPropagation();
14483         //}
14484         if(Roo.isSafari && h && k >= 37 && k <= 40){
14485             e.stopEvent();
14486         }
14487     },
14488
14489     // private
14490     relay : function(e){
14491         var k = e.getKey();
14492         var h = this.keyToHandler[k];
14493         if(h && this[h]){
14494             if(this.doRelay(e, this[h], h) !== true){
14495                 e[this.defaultEventAction]();
14496             }
14497         }
14498     },
14499
14500     // private
14501     doRelay : function(e, h, hname){
14502         return h.call(this.scope || this, e);
14503     },
14504
14505     // possible handlers
14506     enter : false,
14507     left : false,
14508     right : false,
14509     up : false,
14510     down : false,
14511     tab : false,
14512     esc : false,
14513     pageUp : false,
14514     pageDown : false,
14515     del : false,
14516     home : false,
14517     end : false,
14518
14519     // quick lookup hash
14520     keyToHandler : {
14521         37 : "left",
14522         39 : "right",
14523         38 : "up",
14524         40 : "down",
14525         33 : "pageUp",
14526         34 : "pageDown",
14527         46 : "del",
14528         36 : "home",
14529         35 : "end",
14530         13 : "enter",
14531         27 : "esc",
14532         9  : "tab"
14533     },
14534
14535         /**
14536          * Enable this KeyNav
14537          */
14538         enable: function(){
14539                 if(this.disabled){
14540             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14541             // the EventObject will normalize Safari automatically
14542             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14543                 this.el.on("keydown", this.relay,  this);
14544             }else{
14545                 this.el.on("keydown", this.prepareEvent,  this);
14546                 this.el.on("keypress", this.relay,  this);
14547             }
14548                     this.disabled = false;
14549                 }
14550         },
14551
14552         /**
14553          * Disable this KeyNav
14554          */
14555         disable: function(){
14556                 if(!this.disabled){
14557                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14558                 this.el.un("keydown", this.relay);
14559             }else{
14560                 this.el.un("keydown", this.prepareEvent);
14561                 this.el.un("keypress", this.relay);
14562             }
14563                     this.disabled = true;
14564                 }
14565         }
14566 };/*
14567  * Based on:
14568  * Ext JS Library 1.1.1
14569  * Copyright(c) 2006-2007, Ext JS, LLC.
14570  *
14571  * Originally Released Under LGPL - original licence link has changed is not relivant.
14572  *
14573  * Fork - LGPL
14574  * <script type="text/javascript">
14575  */
14576
14577  
14578 /**
14579  * @class Roo.KeyMap
14580  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14581  * The constructor accepts the same config object as defined by {@link #addBinding}.
14582  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14583  * combination it will call the function with this signature (if the match is a multi-key
14584  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14585  * A KeyMap can also handle a string representation of keys.<br />
14586  * Usage:
14587  <pre><code>
14588 // map one key by key code
14589 var map = new Roo.KeyMap("my-element", {
14590     key: 13, // or Roo.EventObject.ENTER
14591     fn: myHandler,
14592     scope: myObject
14593 });
14594
14595 // map multiple keys to one action by string
14596 var map = new Roo.KeyMap("my-element", {
14597     key: "a\r\n\t",
14598     fn: myHandler,
14599     scope: myObject
14600 });
14601
14602 // map multiple keys to multiple actions by strings and array of codes
14603 var map = new Roo.KeyMap("my-element", [
14604     {
14605         key: [10,13],
14606         fn: function(){ alert("Return was pressed"); }
14607     }, {
14608         key: "abc",
14609         fn: function(){ alert('a, b or c was pressed'); }
14610     }, {
14611         key: "\t",
14612         ctrl:true,
14613         shift:true,
14614         fn: function(){ alert('Control + shift + tab was pressed.'); }
14615     }
14616 ]);
14617 </code></pre>
14618  * <b>Note: A KeyMap starts enabled</b>
14619  * @constructor
14620  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14621  * @param {Object} config The config (see {@link #addBinding})
14622  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14623  */
14624 Roo.KeyMap = function(el, config, eventName){
14625     this.el  = Roo.get(el);
14626     this.eventName = eventName || "keydown";
14627     this.bindings = [];
14628     if(config){
14629         this.addBinding(config);
14630     }
14631     this.enable();
14632 };
14633
14634 Roo.KeyMap.prototype = {
14635     /**
14636      * True to stop the event from bubbling and prevent the default browser action if the
14637      * key was handled by the KeyMap (defaults to false)
14638      * @type Boolean
14639      */
14640     stopEvent : false,
14641
14642     /**
14643      * Add a new binding to this KeyMap. The following config object properties are supported:
14644      * <pre>
14645 Property    Type             Description
14646 ----------  ---------------  ----------------------------------------------------------------------
14647 key         String/Array     A single keycode or an array of keycodes to handle
14648 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14649 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14650 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14651 fn          Function         The function to call when KeyMap finds the expected key combination
14652 scope       Object           The scope of the callback function
14653 </pre>
14654      *
14655      * Usage:
14656      * <pre><code>
14657 // Create a KeyMap
14658 var map = new Roo.KeyMap(document, {
14659     key: Roo.EventObject.ENTER,
14660     fn: handleKey,
14661     scope: this
14662 });
14663
14664 //Add a new binding to the existing KeyMap later
14665 map.addBinding({
14666     key: 'abc',
14667     shift: true,
14668     fn: handleKey,
14669     scope: this
14670 });
14671 </code></pre>
14672      * @param {Object/Array} config A single KeyMap config or an array of configs
14673      */
14674         addBinding : function(config){
14675         if(config instanceof Array){
14676             for(var i = 0, len = config.length; i < len; i++){
14677                 this.addBinding(config[i]);
14678             }
14679             return;
14680         }
14681         var keyCode = config.key,
14682             shift = config.shift, 
14683             ctrl = config.ctrl, 
14684             alt = config.alt,
14685             fn = config.fn,
14686             scope = config.scope;
14687         if(typeof keyCode == "string"){
14688             var ks = [];
14689             var keyString = keyCode.toUpperCase();
14690             for(var j = 0, len = keyString.length; j < len; j++){
14691                 ks.push(keyString.charCodeAt(j));
14692             }
14693             keyCode = ks;
14694         }
14695         var keyArray = keyCode instanceof Array;
14696         var handler = function(e){
14697             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14698                 var k = e.getKey();
14699                 if(keyArray){
14700                     for(var i = 0, len = keyCode.length; i < len; i++){
14701                         if(keyCode[i] == k){
14702                           if(this.stopEvent){
14703                               e.stopEvent();
14704                           }
14705                           fn.call(scope || window, k, e);
14706                           return;
14707                         }
14708                     }
14709                 }else{
14710                     if(k == keyCode){
14711                         if(this.stopEvent){
14712                            e.stopEvent();
14713                         }
14714                         fn.call(scope || window, k, e);
14715                     }
14716                 }
14717             }
14718         };
14719         this.bindings.push(handler);  
14720         },
14721
14722     /**
14723      * Shorthand for adding a single key listener
14724      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14725      * following options:
14726      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14727      * @param {Function} fn The function to call
14728      * @param {Object} scope (optional) The scope of the function
14729      */
14730     on : function(key, fn, scope){
14731         var keyCode, shift, ctrl, alt;
14732         if(typeof key == "object" && !(key instanceof Array)){
14733             keyCode = key.key;
14734             shift = key.shift;
14735             ctrl = key.ctrl;
14736             alt = key.alt;
14737         }else{
14738             keyCode = key;
14739         }
14740         this.addBinding({
14741             key: keyCode,
14742             shift: shift,
14743             ctrl: ctrl,
14744             alt: alt,
14745             fn: fn,
14746             scope: scope
14747         })
14748     },
14749
14750     // private
14751     handleKeyDown : function(e){
14752             if(this.enabled){ //just in case
14753             var b = this.bindings;
14754             for(var i = 0, len = b.length; i < len; i++){
14755                 b[i].call(this, e);
14756             }
14757             }
14758         },
14759         
14760         /**
14761          * Returns true if this KeyMap is enabled
14762          * @return {Boolean} 
14763          */
14764         isEnabled : function(){
14765             return this.enabled;  
14766         },
14767         
14768         /**
14769          * Enables this KeyMap
14770          */
14771         enable: function(){
14772                 if(!this.enabled){
14773                     this.el.on(this.eventName, this.handleKeyDown, this);
14774                     this.enabled = true;
14775                 }
14776         },
14777
14778         /**
14779          * Disable this KeyMap
14780          */
14781         disable: function(){
14782                 if(this.enabled){
14783                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14784                     this.enabled = false;
14785                 }
14786         }
14787 };/*
14788  * Based on:
14789  * Ext JS Library 1.1.1
14790  * Copyright(c) 2006-2007, Ext JS, LLC.
14791  *
14792  * Originally Released Under LGPL - original licence link has changed is not relivant.
14793  *
14794  * Fork - LGPL
14795  * <script type="text/javascript">
14796  */
14797
14798  
14799 /**
14800  * @class Roo.util.TextMetrics
14801  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14802  * wide, in pixels, a given block of text will be.
14803  * @singleton
14804  */
14805 Roo.util.TextMetrics = function(){
14806     var shared;
14807     return {
14808         /**
14809          * Measures the size of the specified text
14810          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14811          * that can affect the size of the rendered text
14812          * @param {String} text The text to measure
14813          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14814          * in order to accurately measure the text height
14815          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14816          */
14817         measure : function(el, text, fixedWidth){
14818             if(!shared){
14819                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14820             }
14821             shared.bind(el);
14822             shared.setFixedWidth(fixedWidth || 'auto');
14823             return shared.getSize(text);
14824         },
14825
14826         /**
14827          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14828          * the overhead of multiple calls to initialize the style properties on each measurement.
14829          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14830          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14831          * in order to accurately measure the text height
14832          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14833          */
14834         createInstance : function(el, fixedWidth){
14835             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14836         }
14837     };
14838 }();
14839
14840  
14841
14842 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14843     var ml = new Roo.Element(document.createElement('div'));
14844     document.body.appendChild(ml.dom);
14845     ml.position('absolute');
14846     ml.setLeftTop(-1000, -1000);
14847     ml.hide();
14848
14849     if(fixedWidth){
14850         ml.setWidth(fixedWidth);
14851     }
14852      
14853     var instance = {
14854         /**
14855          * Returns the size of the specified text based on the internal element's style and width properties
14856          * @memberOf Roo.util.TextMetrics.Instance#
14857          * @param {String} text The text to measure
14858          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14859          */
14860         getSize : function(text){
14861             ml.update(text);
14862             var s = ml.getSize();
14863             ml.update('');
14864             return s;
14865         },
14866
14867         /**
14868          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14869          * that can affect the size of the rendered text
14870          * @memberOf Roo.util.TextMetrics.Instance#
14871          * @param {String/HTMLElement} el The element, dom node or id
14872          */
14873         bind : function(el){
14874             ml.setStyle(
14875                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14876             );
14877         },
14878
14879         /**
14880          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14881          * to set a fixed width in order to accurately measure the text height.
14882          * @memberOf Roo.util.TextMetrics.Instance#
14883          * @param {Number} width The width to set on the element
14884          */
14885         setFixedWidth : function(width){
14886             ml.setWidth(width);
14887         },
14888
14889         /**
14890          * Returns the measured width of the specified text
14891          * @memberOf Roo.util.TextMetrics.Instance#
14892          * @param {String} text The text to measure
14893          * @return {Number} width The width in pixels
14894          */
14895         getWidth : function(text){
14896             ml.dom.style.width = 'auto';
14897             return this.getSize(text).width;
14898         },
14899
14900         /**
14901          * Returns the measured height of the specified text.  For multiline text, be sure to call
14902          * {@link #setFixedWidth} if necessary.
14903          * @memberOf Roo.util.TextMetrics.Instance#
14904          * @param {String} text The text to measure
14905          * @return {Number} height The height in pixels
14906          */
14907         getHeight : function(text){
14908             return this.getSize(text).height;
14909         }
14910     };
14911
14912     instance.bind(bindTo);
14913
14914     return instance;
14915 };
14916
14917 // backwards compat
14918 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14919  * Based on:
14920  * Ext JS Library 1.1.1
14921  * Copyright(c) 2006-2007, Ext JS, LLC.
14922  *
14923  * Originally Released Under LGPL - original licence link has changed is not relivant.
14924  *
14925  * Fork - LGPL
14926  * <script type="text/javascript">
14927  */
14928
14929 /**
14930  * @class Roo.state.Provider
14931  * Abstract base class for state provider implementations. This class provides methods
14932  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14933  * Provider interface.
14934  */
14935 Roo.state.Provider = function(){
14936     /**
14937      * @event statechange
14938      * Fires when a state change occurs.
14939      * @param {Provider} this This state provider
14940      * @param {String} key The state key which was changed
14941      * @param {String} value The encoded value for the state
14942      */
14943     this.addEvents({
14944         "statechange": true
14945     });
14946     this.state = {};
14947     Roo.state.Provider.superclass.constructor.call(this);
14948 };
14949 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14950     /**
14951      * Returns the current value for a key
14952      * @param {String} name The key name
14953      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14954      * @return {Mixed} The state data
14955      */
14956     get : function(name, defaultValue){
14957         return typeof this.state[name] == "undefined" ?
14958             defaultValue : this.state[name];
14959     },
14960     
14961     /**
14962      * Clears a value from the state
14963      * @param {String} name The key name
14964      */
14965     clear : function(name){
14966         delete this.state[name];
14967         this.fireEvent("statechange", this, name, null);
14968     },
14969     
14970     /**
14971      * Sets the value for a key
14972      * @param {String} name The key name
14973      * @param {Mixed} value The value to set
14974      */
14975     set : function(name, value){
14976         this.state[name] = value;
14977         this.fireEvent("statechange", this, name, value);
14978     },
14979     
14980     /**
14981      * Decodes a string previously encoded with {@link #encodeValue}.
14982      * @param {String} value The value to decode
14983      * @return {Mixed} The decoded value
14984      */
14985     decodeValue : function(cookie){
14986         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14987         var matches = re.exec(unescape(cookie));
14988         if(!matches || !matches[1]) {
14989             return; // non state cookie
14990         }
14991         var type = matches[1];
14992         var v = matches[2];
14993         switch(type){
14994             case "n":
14995                 return parseFloat(v);
14996             case "d":
14997                 return new Date(Date.parse(v));
14998             case "b":
14999                 return (v == "1");
15000             case "a":
15001                 var all = [];
15002                 var values = v.split("^");
15003                 for(var i = 0, len = values.length; i < len; i++){
15004                     all.push(this.decodeValue(values[i]));
15005                 }
15006                 return all;
15007            case "o":
15008                 var all = {};
15009                 var values = v.split("^");
15010                 for(var i = 0, len = values.length; i < len; i++){
15011                     var kv = values[i].split("=");
15012                     all[kv[0]] = this.decodeValue(kv[1]);
15013                 }
15014                 return all;
15015            default:
15016                 return v;
15017         }
15018     },
15019     
15020     /**
15021      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15022      * @param {Mixed} value The value to encode
15023      * @return {String} The encoded value
15024      */
15025     encodeValue : function(v){
15026         var enc;
15027         if(typeof v == "number"){
15028             enc = "n:" + v;
15029         }else if(typeof v == "boolean"){
15030             enc = "b:" + (v ? "1" : "0");
15031         }else if(v instanceof Date){
15032             enc = "d:" + v.toGMTString();
15033         }else if(v instanceof Array){
15034             var flat = "";
15035             for(var i = 0, len = v.length; i < len; i++){
15036                 flat += this.encodeValue(v[i]);
15037                 if(i != len-1) {
15038                     flat += "^";
15039                 }
15040             }
15041             enc = "a:" + flat;
15042         }else if(typeof v == "object"){
15043             var flat = "";
15044             for(var key in v){
15045                 if(typeof v[key] != "function"){
15046                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15047                 }
15048             }
15049             enc = "o:" + flat.substring(0, flat.length-1);
15050         }else{
15051             enc = "s:" + v;
15052         }
15053         return escape(enc);        
15054     }
15055 });
15056
15057 /*
15058  * Based on:
15059  * Ext JS Library 1.1.1
15060  * Copyright(c) 2006-2007, Ext JS, LLC.
15061  *
15062  * Originally Released Under LGPL - original licence link has changed is not relivant.
15063  *
15064  * Fork - LGPL
15065  * <script type="text/javascript">
15066  */
15067 /**
15068  * @class Roo.state.Manager
15069  * This is the global state manager. By default all components that are "state aware" check this class
15070  * for state information if you don't pass them a custom state provider. In order for this class
15071  * to be useful, it must be initialized with a provider when your application initializes.
15072  <pre><code>
15073 // in your initialization function
15074 init : function(){
15075    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15076    ...
15077    // supposed you have a {@link Roo.BorderLayout}
15078    var layout = new Roo.BorderLayout(...);
15079    layout.restoreState();
15080    // or a {Roo.BasicDialog}
15081    var dialog = new Roo.BasicDialog(...);
15082    dialog.restoreState();
15083  </code></pre>
15084  * @singleton
15085  */
15086 Roo.state.Manager = function(){
15087     var provider = new Roo.state.Provider();
15088     
15089     return {
15090         /**
15091          * Configures the default state provider for your application
15092          * @param {Provider} stateProvider The state provider to set
15093          */
15094         setProvider : function(stateProvider){
15095             provider = stateProvider;
15096         },
15097         
15098         /**
15099          * Returns the current value for a key
15100          * @param {String} name The key name
15101          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15102          * @return {Mixed} The state data
15103          */
15104         get : function(key, defaultValue){
15105             return provider.get(key, defaultValue);
15106         },
15107         
15108         /**
15109          * Sets the value for a key
15110          * @param {String} name The key name
15111          * @param {Mixed} value The state data
15112          */
15113          set : function(key, value){
15114             provider.set(key, value);
15115         },
15116         
15117         /**
15118          * Clears a value from the state
15119          * @param {String} name The key name
15120          */
15121         clear : function(key){
15122             provider.clear(key);
15123         },
15124         
15125         /**
15126          * Gets the currently configured state provider
15127          * @return {Provider} The state provider
15128          */
15129         getProvider : function(){
15130             return provider;
15131         }
15132     };
15133 }();
15134 /*
15135  * Based on:
15136  * Ext JS Library 1.1.1
15137  * Copyright(c) 2006-2007, Ext JS, LLC.
15138  *
15139  * Originally Released Under LGPL - original licence link has changed is not relivant.
15140  *
15141  * Fork - LGPL
15142  * <script type="text/javascript">
15143  */
15144 /**
15145  * @class Roo.state.CookieProvider
15146  * @extends Roo.state.Provider
15147  * The default Provider implementation which saves state via cookies.
15148  * <br />Usage:
15149  <pre><code>
15150    var cp = new Roo.state.CookieProvider({
15151        path: "/cgi-bin/",
15152        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15153        domain: "roojs.com"
15154    })
15155    Roo.state.Manager.setProvider(cp);
15156  </code></pre>
15157  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15158  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15159  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15160  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15161  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15162  * domain the page is running on including the 'www' like 'www.roojs.com')
15163  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15164  * @constructor
15165  * Create a new CookieProvider
15166  * @param {Object} config The configuration object
15167  */
15168 Roo.state.CookieProvider = function(config){
15169     Roo.state.CookieProvider.superclass.constructor.call(this);
15170     this.path = "/";
15171     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15172     this.domain = null;
15173     this.secure = false;
15174     Roo.apply(this, config);
15175     this.state = this.readCookies();
15176 };
15177
15178 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15179     // private
15180     set : function(name, value){
15181         if(typeof value == "undefined" || value === null){
15182             this.clear(name);
15183             return;
15184         }
15185         this.setCookie(name, value);
15186         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15187     },
15188
15189     // private
15190     clear : function(name){
15191         this.clearCookie(name);
15192         Roo.state.CookieProvider.superclass.clear.call(this, name);
15193     },
15194
15195     // private
15196     readCookies : function(){
15197         var cookies = {};
15198         var c = document.cookie + ";";
15199         var re = /\s?(.*?)=(.*?);/g;
15200         var matches;
15201         while((matches = re.exec(c)) != null){
15202             var name = matches[1];
15203             var value = matches[2];
15204             if(name && name.substring(0,3) == "ys-"){
15205                 cookies[name.substr(3)] = this.decodeValue(value);
15206             }
15207         }
15208         return cookies;
15209     },
15210
15211     // private
15212     setCookie : function(name, value){
15213         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15214            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15215            ((this.path == null) ? "" : ("; path=" + this.path)) +
15216            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15217            ((this.secure == true) ? "; secure" : "");
15218     },
15219
15220     // private
15221     clearCookie : function(name){
15222         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15223            ((this.path == null) ? "" : ("; path=" + this.path)) +
15224            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15225            ((this.secure == true) ? "; secure" : "");
15226     }
15227 });/*
15228  * Based on:
15229  * Ext JS Library 1.1.1
15230  * Copyright(c) 2006-2007, Ext JS, LLC.
15231  *
15232  * Originally Released Under LGPL - original licence link has changed is not relivant.
15233  *
15234  * Fork - LGPL
15235  * <script type="text/javascript">
15236  */
15237  
15238
15239 /**
15240  * @class Roo.ComponentMgr
15241  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15242  * @singleton
15243  */
15244 Roo.ComponentMgr = function(){
15245     var all = new Roo.util.MixedCollection();
15246
15247     return {
15248         /**
15249          * Registers a component.
15250          * @param {Roo.Component} c The component
15251          */
15252         register : function(c){
15253             all.add(c);
15254         },
15255
15256         /**
15257          * Unregisters a component.
15258          * @param {Roo.Component} c The component
15259          */
15260         unregister : function(c){
15261             all.remove(c);
15262         },
15263
15264         /**
15265          * Returns a component by id
15266          * @param {String} id The component id
15267          */
15268         get : function(id){
15269             return all.get(id);
15270         },
15271
15272         /**
15273          * Registers a function that will be called when a specified component is added to ComponentMgr
15274          * @param {String} id The component id
15275          * @param {Funtction} fn The callback function
15276          * @param {Object} scope The scope of the callback
15277          */
15278         onAvailable : function(id, fn, scope){
15279             all.on("add", function(index, o){
15280                 if(o.id == id){
15281                     fn.call(scope || o, o);
15282                     all.un("add", fn, scope);
15283                 }
15284             });
15285         }
15286     };
15287 }();/*
15288  * Based on:
15289  * Ext JS Library 1.1.1
15290  * Copyright(c) 2006-2007, Ext JS, LLC.
15291  *
15292  * Originally Released Under LGPL - original licence link has changed is not relivant.
15293  *
15294  * Fork - LGPL
15295  * <script type="text/javascript">
15296  */
15297  
15298 /**
15299  * @class Roo.Component
15300  * @extends Roo.util.Observable
15301  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15302  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15303  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15304  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15305  * All visual components (widgets) that require rendering into a layout should subclass Component.
15306  * @constructor
15307  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15308  * element and its id used as the component id.  If a string is passed, it is assumed to be the id of an existing element
15309  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15310  */
15311 Roo.Component = function(config){
15312     config = config || {};
15313     if(config.tagName || config.dom || typeof config == "string"){ // element object
15314         config = {el: config, id: config.id || config};
15315     }
15316     this.initialConfig = config;
15317
15318     Roo.apply(this, config);
15319     this.addEvents({
15320         /**
15321          * @event disable
15322          * Fires after the component is disabled.
15323              * @param {Roo.Component} this
15324              */
15325         disable : true,
15326         /**
15327          * @event enable
15328          * Fires after the component is enabled.
15329              * @param {Roo.Component} this
15330              */
15331         enable : true,
15332         /**
15333          * @event beforeshow
15334          * Fires before the component is shown.  Return false to stop the show.
15335              * @param {Roo.Component} this
15336              */
15337         beforeshow : true,
15338         /**
15339          * @event show
15340          * Fires after the component is shown.
15341              * @param {Roo.Component} this
15342              */
15343         show : true,
15344         /**
15345          * @event beforehide
15346          * Fires before the component is hidden. Return false to stop the hide.
15347              * @param {Roo.Component} this
15348              */
15349         beforehide : true,
15350         /**
15351          * @event hide
15352          * Fires after the component is hidden.
15353              * @param {Roo.Component} this
15354              */
15355         hide : true,
15356         /**
15357          * @event beforerender
15358          * Fires before the component is rendered. Return false to stop the render.
15359              * @param {Roo.Component} this
15360              */
15361         beforerender : true,
15362         /**
15363          * @event render
15364          * Fires after the component is rendered.
15365              * @param {Roo.Component} this
15366              */
15367         render : true,
15368         /**
15369          * @event beforedestroy
15370          * Fires before the component is destroyed. Return false to stop the destroy.
15371              * @param {Roo.Component} this
15372              */
15373         beforedestroy : true,
15374         /**
15375          * @event destroy
15376          * Fires after the component is destroyed.
15377              * @param {Roo.Component} this
15378              */
15379         destroy : true
15380     });
15381     if(!this.id){
15382         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15383     }
15384     Roo.ComponentMgr.register(this);
15385     Roo.Component.superclass.constructor.call(this);
15386     this.initComponent();
15387     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15388         this.render(this.renderTo);
15389         delete this.renderTo;
15390     }
15391 };
15392
15393 /** @private */
15394 Roo.Component.AUTO_ID = 1000;
15395
15396 Roo.extend(Roo.Component, Roo.util.Observable, {
15397     /**
15398      * @scope Roo.Component.prototype
15399      * @type {Boolean}
15400      * true if this component is hidden. Read-only.
15401      */
15402     hidden : false,
15403     /**
15404      * @type {Boolean}
15405      * true if this component is disabled. Read-only.
15406      */
15407     disabled : false,
15408     /**
15409      * @type {Boolean}
15410      * true if this component has been rendered. Read-only.
15411      */
15412     rendered : false,
15413     
15414     /** @cfg {String} disableClass
15415      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15416      */
15417     disabledClass : "x-item-disabled",
15418         /** @cfg {Boolean} allowDomMove
15419          * Whether the component can move the Dom node when rendering (defaults to true).
15420          */
15421     allowDomMove : true,
15422     /** @cfg {String} hideMode (display|visibility)
15423      * How this component should hidden. Supported values are
15424      * "visibility" (css visibility), "offsets" (negative offset position) and
15425      * "display" (css display) - defaults to "display".
15426      */
15427     hideMode: 'display',
15428
15429     /** @private */
15430     ctype : "Roo.Component",
15431
15432     /**
15433      * @cfg {String} actionMode 
15434      * which property holds the element that used for  hide() / show() / disable() / enable()
15435      * default is 'el' 
15436      */
15437     actionMode : "el",
15438
15439     /** @private */
15440     getActionEl : function(){
15441         return this[this.actionMode];
15442     },
15443
15444     initComponent : Roo.emptyFn,
15445     /**
15446      * If this is a lazy rendering component, render it to its container element.
15447      * @param {String/HTMLElement/Element} container (optional) The element this component should be rendered into. If it is being applied to existing markup, this should be left off.
15448      */
15449     render : function(container, position){
15450         
15451         if(this.rendered){
15452             return this;
15453         }
15454         
15455         if(this.fireEvent("beforerender", this) === false){
15456             return false;
15457         }
15458         
15459         if(!container && this.el){
15460             this.el = Roo.get(this.el);
15461             container = this.el.dom.parentNode;
15462             this.allowDomMove = false;
15463         }
15464         this.container = Roo.get(container);
15465         this.rendered = true;
15466         if(position !== undefined){
15467             if(typeof position == 'number'){
15468                 position = this.container.dom.childNodes[position];
15469             }else{
15470                 position = Roo.getDom(position);
15471             }
15472         }
15473         this.onRender(this.container, position || null);
15474         if(this.cls){
15475             this.el.addClass(this.cls);
15476             delete this.cls;
15477         }
15478         if(this.style){
15479             this.el.applyStyles(this.style);
15480             delete this.style;
15481         }
15482         this.fireEvent("render", this);
15483         this.afterRender(this.container);
15484         if(this.hidden){
15485             this.hide();
15486         }
15487         if(this.disabled){
15488             this.disable();
15489         }
15490
15491         return this;
15492         
15493     },
15494
15495     /** @private */
15496     // default function is not really useful
15497     onRender : function(ct, position){
15498         if(this.el){
15499             this.el = Roo.get(this.el);
15500             if(this.allowDomMove !== false){
15501                 ct.dom.insertBefore(this.el.dom, position);
15502             }
15503         }
15504     },
15505
15506     /** @private */
15507     getAutoCreate : function(){
15508         var cfg = typeof this.autoCreate == "object" ?
15509                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15510         if(this.id && !cfg.id){
15511             cfg.id = this.id;
15512         }
15513         return cfg;
15514     },
15515
15516     /** @private */
15517     afterRender : Roo.emptyFn,
15518
15519     /**
15520      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15521      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15522      */
15523     destroy : function(){
15524         if(this.fireEvent("beforedestroy", this) !== false){
15525             this.purgeListeners();
15526             this.beforeDestroy();
15527             if(this.rendered){
15528                 this.el.removeAllListeners();
15529                 this.el.remove();
15530                 if(this.actionMode == "container"){
15531                     this.container.remove();
15532                 }
15533             }
15534             this.onDestroy();
15535             Roo.ComponentMgr.unregister(this);
15536             this.fireEvent("destroy", this);
15537         }
15538     },
15539
15540         /** @private */
15541     beforeDestroy : function(){
15542
15543     },
15544
15545         /** @private */
15546         onDestroy : function(){
15547
15548     },
15549
15550     /**
15551      * Returns the underlying {@link Roo.Element}.
15552      * @return {Roo.Element} The element
15553      */
15554     getEl : function(){
15555         return this.el;
15556     },
15557
15558     /**
15559      * Returns the id of this component.
15560      * @return {String}
15561      */
15562     getId : function(){
15563         return this.id;
15564     },
15565
15566     /**
15567      * Try to focus this component.
15568      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15569      * @return {Roo.Component} this
15570      */
15571     focus : function(selectText){
15572         if(this.rendered){
15573             this.el.focus();
15574             if(selectText === true){
15575                 this.el.dom.select();
15576             }
15577         }
15578         return this;
15579     },
15580
15581     /** @private */
15582     blur : function(){
15583         if(this.rendered){
15584             this.el.blur();
15585         }
15586         return this;
15587     },
15588
15589     /**
15590      * Disable this component.
15591      * @return {Roo.Component} this
15592      */
15593     disable : function(){
15594         if(this.rendered){
15595             this.onDisable();
15596         }
15597         this.disabled = true;
15598         this.fireEvent("disable", this);
15599         return this;
15600     },
15601
15602         // private
15603     onDisable : function(){
15604         this.getActionEl().addClass(this.disabledClass);
15605         this.el.dom.disabled = true;
15606     },
15607
15608     /**
15609      * Enable this component.
15610      * @return {Roo.Component} this
15611      */
15612     enable : function(){
15613         if(this.rendered){
15614             this.onEnable();
15615         }
15616         this.disabled = false;
15617         this.fireEvent("enable", this);
15618         return this;
15619     },
15620
15621         // private
15622     onEnable : function(){
15623         this.getActionEl().removeClass(this.disabledClass);
15624         this.el.dom.disabled = false;
15625     },
15626
15627     /**
15628      * Convenience function for setting disabled/enabled by boolean.
15629      * @param {Boolean} disabled
15630      */
15631     setDisabled : function(disabled){
15632         this[disabled ? "disable" : "enable"]();
15633     },
15634
15635     /**
15636      * Show this component.
15637      * @return {Roo.Component} this
15638      */
15639     show: function(){
15640         if(this.fireEvent("beforeshow", this) !== false){
15641             this.hidden = false;
15642             if(this.rendered){
15643                 this.onShow();
15644             }
15645             this.fireEvent("show", this);
15646         }
15647         return this;
15648     },
15649
15650     // private
15651     onShow : function(){
15652         var ae = this.getActionEl();
15653         if(this.hideMode == 'visibility'){
15654             ae.dom.style.visibility = "visible";
15655         }else if(this.hideMode == 'offsets'){
15656             ae.removeClass('x-hidden');
15657         }else{
15658             ae.dom.style.display = "";
15659         }
15660     },
15661
15662     /**
15663      * Hide this component.
15664      * @return {Roo.Component} this
15665      */
15666     hide: function(){
15667         if(this.fireEvent("beforehide", this) !== false){
15668             this.hidden = true;
15669             if(this.rendered){
15670                 this.onHide();
15671             }
15672             this.fireEvent("hide", this);
15673         }
15674         return this;
15675     },
15676
15677     // private
15678     onHide : function(){
15679         var ae = this.getActionEl();
15680         if(this.hideMode == 'visibility'){
15681             ae.dom.style.visibility = "hidden";
15682         }else if(this.hideMode == 'offsets'){
15683             ae.addClass('x-hidden');
15684         }else{
15685             ae.dom.style.display = "none";
15686         }
15687     },
15688
15689     /**
15690      * Convenience function to hide or show this component by boolean.
15691      * @param {Boolean} visible True to show, false to hide
15692      * @return {Roo.Component} this
15693      */
15694     setVisible: function(visible){
15695         if(visible) {
15696             this.show();
15697         }else{
15698             this.hide();
15699         }
15700         return this;
15701     },
15702
15703     /**
15704      * Returns true if this component is visible.
15705      */
15706     isVisible : function(){
15707         return this.getActionEl().isVisible();
15708     },
15709
15710     cloneConfig : function(overrides){
15711         overrides = overrides || {};
15712         var id = overrides.id || Roo.id();
15713         var cfg = Roo.applyIf(overrides, this.initialConfig);
15714         cfg.id = id; // prevent dup id
15715         return new this.constructor(cfg);
15716     }
15717 });/*
15718  * Based on:
15719  * Ext JS Library 1.1.1
15720  * Copyright(c) 2006-2007, Ext JS, LLC.
15721  *
15722  * Originally Released Under LGPL - original licence link has changed is not relivant.
15723  *
15724  * Fork - LGPL
15725  * <script type="text/javascript">
15726  */
15727
15728 /**
15729  * @class Roo.BoxComponent
15730  * @extends Roo.Component
15731  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15732  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15733  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15734  * layout containers.
15735  * @constructor
15736  * @param {Roo.Element/String/Object} config The configuration options.
15737  */
15738 Roo.BoxComponent = function(config){
15739     Roo.Component.call(this, config);
15740     this.addEvents({
15741         /**
15742          * @event resize
15743          * Fires after the component is resized.
15744              * @param {Roo.Component} this
15745              * @param {Number} adjWidth The box-adjusted width that was set
15746              * @param {Number} adjHeight The box-adjusted height that was set
15747              * @param {Number} rawWidth The width that was originally specified
15748              * @param {Number} rawHeight The height that was originally specified
15749              */
15750         resize : true,
15751         /**
15752          * @event move
15753          * Fires after the component is moved.
15754              * @param {Roo.Component} this
15755              * @param {Number} x The new x position
15756              * @param {Number} y The new y position
15757              */
15758         move : true
15759     });
15760 };
15761
15762 Roo.extend(Roo.BoxComponent, Roo.Component, {
15763     // private, set in afterRender to signify that the component has been rendered
15764     boxReady : false,
15765     // private, used to defer height settings to subclasses
15766     deferHeight: false,
15767     /** @cfg {Number} width
15768      * width (optional) size of component
15769      */
15770      /** @cfg {Number} height
15771      * height (optional) size of component
15772      */
15773      
15774     /**
15775      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15776      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15777      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15778      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15779      * @return {Roo.BoxComponent} this
15780      */
15781     setSize : function(w, h){
15782         // support for standard size objects
15783         if(typeof w == 'object'){
15784             h = w.height;
15785             w = w.width;
15786         }
15787         // not rendered
15788         if(!this.boxReady){
15789             this.width = w;
15790             this.height = h;
15791             return this;
15792         }
15793
15794         // prevent recalcs when not needed
15795         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15796             return this;
15797         }
15798         this.lastSize = {width: w, height: h};
15799
15800         var adj = this.adjustSize(w, h);
15801         var aw = adj.width, ah = adj.height;
15802         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15803             var rz = this.getResizeEl();
15804             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15805                 rz.setSize(aw, ah);
15806             }else if(!this.deferHeight && ah !== undefined){
15807                 rz.setHeight(ah);
15808             }else if(aw !== undefined){
15809                 rz.setWidth(aw);
15810             }
15811             this.onResize(aw, ah, w, h);
15812             this.fireEvent('resize', this, aw, ah, w, h);
15813         }
15814         return this;
15815     },
15816
15817     /**
15818      * Gets the current size of the component's underlying element.
15819      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15820      */
15821     getSize : function(){
15822         return this.el.getSize();
15823     },
15824
15825     /**
15826      * Gets the current XY position of the component's underlying element.
15827      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15828      * @return {Array} The XY position of the element (e.g., [100, 200])
15829      */
15830     getPosition : function(local){
15831         if(local === true){
15832             return [this.el.getLeft(true), this.el.getTop(true)];
15833         }
15834         return this.xy || this.el.getXY();
15835     },
15836
15837     /**
15838      * Gets the current box measurements of the component's underlying element.
15839      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15840      * @returns {Object} box An object in the format {x, y, width, height}
15841      */
15842     getBox : function(local){
15843         var s = this.el.getSize();
15844         if(local){
15845             s.x = this.el.getLeft(true);
15846             s.y = this.el.getTop(true);
15847         }else{
15848             var xy = this.xy || this.el.getXY();
15849             s.x = xy[0];
15850             s.y = xy[1];
15851         }
15852         return s;
15853     },
15854
15855     /**
15856      * Sets the current box measurements of the component's underlying element.
15857      * @param {Object} box An object in the format {x, y, width, height}
15858      * @returns {Roo.BoxComponent} this
15859      */
15860     updateBox : function(box){
15861         this.setSize(box.width, box.height);
15862         this.setPagePosition(box.x, box.y);
15863         return this;
15864     },
15865
15866     // protected
15867     getResizeEl : function(){
15868         return this.resizeEl || this.el;
15869     },
15870
15871     // protected
15872     getPositionEl : function(){
15873         return this.positionEl || this.el;
15874     },
15875
15876     /**
15877      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15878      * This method fires the move event.
15879      * @param {Number} left The new left
15880      * @param {Number} top The new top
15881      * @returns {Roo.BoxComponent} this
15882      */
15883     setPosition : function(x, y){
15884         this.x = x;
15885         this.y = y;
15886         if(!this.boxReady){
15887             return this;
15888         }
15889         var adj = this.adjustPosition(x, y);
15890         var ax = adj.x, ay = adj.y;
15891
15892         var el = this.getPositionEl();
15893         if(ax !== undefined || ay !== undefined){
15894             if(ax !== undefined && ay !== undefined){
15895                 el.setLeftTop(ax, ay);
15896             }else if(ax !== undefined){
15897                 el.setLeft(ax);
15898             }else if(ay !== undefined){
15899                 el.setTop(ay);
15900             }
15901             this.onPosition(ax, ay);
15902             this.fireEvent('move', this, ax, ay);
15903         }
15904         return this;
15905     },
15906
15907     /**
15908      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15909      * This method fires the move event.
15910      * @param {Number} x The new x position
15911      * @param {Number} y The new y position
15912      * @returns {Roo.BoxComponent} this
15913      */
15914     setPagePosition : function(x, y){
15915         this.pageX = x;
15916         this.pageY = y;
15917         if(!this.boxReady){
15918             return;
15919         }
15920         if(x === undefined || y === undefined){ // cannot translate undefined points
15921             return;
15922         }
15923         var p = this.el.translatePoints(x, y);
15924         this.setPosition(p.left, p.top);
15925         return this;
15926     },
15927
15928     // private
15929     onRender : function(ct, position){
15930         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15931         if(this.resizeEl){
15932             this.resizeEl = Roo.get(this.resizeEl);
15933         }
15934         if(this.positionEl){
15935             this.positionEl = Roo.get(this.positionEl);
15936         }
15937     },
15938
15939     // private
15940     afterRender : function(){
15941         Roo.BoxComponent.superclass.afterRender.call(this);
15942         this.boxReady = true;
15943         this.setSize(this.width, this.height);
15944         if(this.x || this.y){
15945             this.setPosition(this.x, this.y);
15946         }
15947         if(this.pageX || this.pageY){
15948             this.setPagePosition(this.pageX, this.pageY);
15949         }
15950     },
15951
15952     /**
15953      * Force the component's size to recalculate based on the underlying element's current height and width.
15954      * @returns {Roo.BoxComponent} this
15955      */
15956     syncSize : function(){
15957         delete this.lastSize;
15958         this.setSize(this.el.getWidth(), this.el.getHeight());
15959         return this;
15960     },
15961
15962     /**
15963      * Called after the component is resized, this method is empty by default but can be implemented by any
15964      * subclass that needs to perform custom logic after a resize occurs.
15965      * @param {Number} adjWidth The box-adjusted width that was set
15966      * @param {Number} adjHeight The box-adjusted height that was set
15967      * @param {Number} rawWidth The width that was originally specified
15968      * @param {Number} rawHeight The height that was originally specified
15969      */
15970     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15971
15972     },
15973
15974     /**
15975      * Called after the component is moved, this method is empty by default but can be implemented by any
15976      * subclass that needs to perform custom logic after a move occurs.
15977      * @param {Number} x The new x position
15978      * @param {Number} y The new y position
15979      */
15980     onPosition : function(x, y){
15981
15982     },
15983
15984     // private
15985     adjustSize : function(w, h){
15986         if(this.autoWidth){
15987             w = 'auto';
15988         }
15989         if(this.autoHeight){
15990             h = 'auto';
15991         }
15992         return {width : w, height: h};
15993     },
15994
15995     // private
15996     adjustPosition : function(x, y){
15997         return {x : x, y: y};
15998     }
15999 });/*
16000  * Original code for Roojs - LGPL
16001  * <script type="text/javascript">
16002  */
16003  
16004 /**
16005  * @class Roo.XComponent
16006  * A delayed Element creator...
16007  * Or a way to group chunks of interface together.
16008  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16009  *  used in conjunction with XComponent.build() it will create an instance of each element,
16010  *  then call addxtype() to build the User interface.
16011  * 
16012  * Mypart.xyx = new Roo.XComponent({
16013
16014     parent : 'Mypart.xyz', // empty == document.element.!!
16015     order : '001',
16016     name : 'xxxx'
16017     region : 'xxxx'
16018     disabled : function() {} 
16019      
16020     tree : function() { // return an tree of xtype declared components
16021         var MODULE = this;
16022         return 
16023         {
16024             xtype : 'NestedLayoutPanel',
16025             // technicall
16026         }
16027      ]
16028  *})
16029  *
16030  *
16031  * It can be used to build a big heiracy, with parent etc.
16032  * or you can just use this to render a single compoent to a dom element
16033  * MYPART.render(Roo.Element | String(id) | dom_element )
16034  *
16035  *
16036  * Usage patterns.
16037  *
16038  * Classic Roo
16039  *
16040  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16041  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16042  *
16043  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16044  *
16045  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16046  * - if mulitple topModules exist, the last one is defined as the top module.
16047  *
16048  * Embeded Roo
16049  * 
16050  * When the top level or multiple modules are to embedded into a existing HTML page,
16051  * the parent element can container '#id' of the element where the module will be drawn.
16052  *
16053  * Bootstrap Roo
16054  *
16055  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16056  * it relies more on a include mechanism, where sub modules are included into an outer page.
16057  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16058  * 
16059  * Bootstrap Roo Included elements
16060  *
16061  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16062  * hence confusing the component builder as it thinks there are multiple top level elements. 
16063  *
16064  * String Over-ride & Translations
16065  *
16066  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16067  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16068  * are needed. @see Roo.XComponent.overlayString  
16069  * 
16070  * 
16071  * 
16072  * @extends Roo.util.Observable
16073  * @constructor
16074  * @param cfg {Object} configuration of component
16075  * 
16076  */
16077 Roo.XComponent = function(cfg) {
16078     Roo.apply(this, cfg);
16079     this.addEvents({ 
16080         /**
16081              * @event built
16082              * Fires when this the componnt is built
16083              * @param {Roo.XComponent} c the component
16084              */
16085         'built' : true
16086         
16087     });
16088     this.region = this.region || 'center'; // default..
16089     Roo.XComponent.register(this);
16090     this.modules = false;
16091     this.el = false; // where the layout goes..
16092     
16093     
16094 }
16095 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16096     /**
16097      * @property el
16098      * The created element (with Roo.factory())
16099      * @type {Roo.Layout}
16100      */
16101     el  : false,
16102     
16103     /**
16104      * @property el
16105      * for BC  - use el in new code
16106      * @type {Roo.Layout}
16107      */
16108     panel : false,
16109     
16110     /**
16111      * @property layout
16112      * for BC  - use el in new code
16113      * @type {Roo.Layout}
16114      */
16115     layout : false,
16116     
16117      /**
16118      * @cfg {Function|boolean} disabled
16119      * If this module is disabled by some rule, return true from the funtion
16120      */
16121     disabled : false,
16122     
16123     /**
16124      * @cfg {String} parent 
16125      * Name of parent element which it get xtype added to..
16126      */
16127     parent: false,
16128     
16129     /**
16130      * @cfg {String} order
16131      * Used to set the order in which elements are created (usefull for multiple tabs)
16132      */
16133     
16134     order : false,
16135     /**
16136      * @cfg {String} name
16137      * String to display while loading.
16138      */
16139     name : false,
16140     /**
16141      * @cfg {String} region
16142      * Region to render component to (defaults to center)
16143      */
16144     region : 'center',
16145     
16146     /**
16147      * @cfg {Array} items
16148      * A single item array - the first element is the root of the tree..
16149      * It's done this way to stay compatible with the Xtype system...
16150      */
16151     items : false,
16152     
16153     /**
16154      * @property _tree
16155      * The method that retuns the tree of parts that make up this compoennt 
16156      * @type {function}
16157      */
16158     _tree  : false,
16159     
16160      /**
16161      * render
16162      * render element to dom or tree
16163      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16164      */
16165     
16166     render : function(el)
16167     {
16168         
16169         el = el || false;
16170         var hp = this.parent ? 1 : 0;
16171         Roo.debug &&  Roo.log(this);
16172         
16173         var tree = this._tree ? this._tree() : this.tree();
16174
16175         
16176         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16177             // if parent is a '#.....' string, then let's use that..
16178             var ename = this.parent.substr(1);
16179             this.parent = false;
16180             Roo.debug && Roo.log(ename);
16181             switch (ename) {
16182                 case 'bootstrap-body':
16183                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16184                         // this is the BorderLayout standard?
16185                        this.parent = { el : true };
16186                        break;
16187                     }
16188                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16189                         // need to insert stuff...
16190                         this.parent =  {
16191                              el : new Roo.bootstrap.layout.Border({
16192                                  el : document.body, 
16193                      
16194                                  center: {
16195                                     titlebar: false,
16196                                     autoScroll:false,
16197                                     closeOnTab: true,
16198                                     tabPosition: 'top',
16199                                       //resizeTabs: true,
16200                                     alwaysShowTabs: true,
16201                                     hideTabs: false
16202                                      //minTabWidth: 140
16203                                  }
16204                              })
16205                         
16206                          };
16207                          break;
16208                     }
16209                          
16210                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16211                         this.parent = { el :  new  Roo.bootstrap.Body() };
16212                         Roo.debug && Roo.log("setting el to doc body");
16213                          
16214                     } else {
16215                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16216                     }
16217                     break;
16218                 case 'bootstrap':
16219                     this.parent = { el : true};
16220                     // fall through
16221                 default:
16222                     el = Roo.get(ename);
16223                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16224                         this.parent = { el : true};
16225                     }
16226                     
16227                     break;
16228             }
16229                 
16230             
16231             if (!el && !this.parent) {
16232                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16233                 return;
16234             }
16235         }
16236         
16237         Roo.debug && Roo.log("EL:");
16238         Roo.debug && Roo.log(el);
16239         Roo.debug && Roo.log("this.parent.el:");
16240         Roo.debug && Roo.log(this.parent.el);
16241         
16242
16243         // altertive root elements ??? - we need a better way to indicate these.
16244         var is_alt = Roo.XComponent.is_alt ||
16245                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16246                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16247                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16248         
16249         
16250         
16251         if (!this.parent && is_alt) {
16252             //el = Roo.get(document.body);
16253             this.parent = { el : true };
16254         }
16255             
16256             
16257         
16258         if (!this.parent) {
16259             
16260             Roo.debug && Roo.log("no parent - creating one");
16261             
16262             el = el ? Roo.get(el) : false;      
16263             
16264             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16265                 
16266                 this.parent =  {
16267                     el : new Roo.bootstrap.layout.Border({
16268                         el: el || document.body,
16269                     
16270                         center: {
16271                             titlebar: false,
16272                             autoScroll:false,
16273                             closeOnTab: true,
16274                             tabPosition: 'top',
16275                              //resizeTabs: true,
16276                             alwaysShowTabs: false,
16277                             hideTabs: true,
16278                             minTabWidth: 140,
16279                             overflow: 'visible'
16280                          }
16281                      })
16282                 };
16283             } else {
16284             
16285                 // it's a top level one..
16286                 this.parent =  {
16287                     el : new Roo.BorderLayout(el || document.body, {
16288                         center: {
16289                             titlebar: false,
16290                             autoScroll:false,
16291                             closeOnTab: true,
16292                             tabPosition: 'top',
16293                              //resizeTabs: true,
16294                             alwaysShowTabs: el && hp? false :  true,
16295                             hideTabs: el || !hp ? true :  false,
16296                             minTabWidth: 140
16297                          }
16298                     })
16299                 };
16300             }
16301         }
16302         
16303         if (!this.parent.el) {
16304                 // probably an old style ctor, which has been disabled.
16305                 return;
16306
16307         }
16308                 // The 'tree' method is  '_tree now' 
16309             
16310         tree.region = tree.region || this.region;
16311         var is_body = false;
16312         if (this.parent.el === true) {
16313             // bootstrap... - body..
16314             if (el) {
16315                 tree.el = el;
16316             }
16317             this.parent.el = Roo.factory(tree);
16318             is_body = true;
16319         }
16320         
16321         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16322         this.fireEvent('built', this);
16323         
16324         this.panel = this.el;
16325         this.layout = this.panel.layout;
16326         this.parentLayout = this.parent.layout  || false;  
16327          
16328     }
16329     
16330 });
16331
16332 Roo.apply(Roo.XComponent, {
16333     /**
16334      * @property  hideProgress
16335      * true to disable the building progress bar.. usefull on single page renders.
16336      * @type Boolean
16337      */
16338     hideProgress : false,
16339     /**
16340      * @property  buildCompleted
16341      * True when the builder has completed building the interface.
16342      * @type Boolean
16343      */
16344     buildCompleted : false,
16345      
16346     /**
16347      * @property  topModule
16348      * the upper most module - uses document.element as it's constructor.
16349      * @type Object
16350      */
16351      
16352     topModule  : false,
16353       
16354     /**
16355      * @property  modules
16356      * array of modules to be created by registration system.
16357      * @type {Array} of Roo.XComponent
16358      */
16359     
16360     modules : [],
16361     /**
16362      * @property  elmodules
16363      * array of modules to be created by which use #ID 
16364      * @type {Array} of Roo.XComponent
16365      */
16366      
16367     elmodules : [],
16368
16369      /**
16370      * @property  is_alt
16371      * Is an alternative Root - normally used by bootstrap or other systems,
16372      *    where the top element in the tree can wrap 'body' 
16373      * @type {boolean}  (default false)
16374      */
16375      
16376     is_alt : false,
16377     /**
16378      * @property  build_from_html
16379      * Build elements from html - used by bootstrap HTML stuff 
16380      *    - this is cleared after build is completed
16381      * @type {boolean}    (default false)
16382      */
16383      
16384     build_from_html : false,
16385     /**
16386      * Register components to be built later.
16387      *
16388      * This solves the following issues
16389      * - Building is not done on page load, but after an authentication process has occured.
16390      * - Interface elements are registered on page load
16391      * - Parent Interface elements may not be loaded before child, so this handles that..
16392      * 
16393      *
16394      * example:
16395      * 
16396      * MyApp.register({
16397           order : '000001',
16398           module : 'Pman.Tab.projectMgr',
16399           region : 'center',
16400           parent : 'Pman.layout',
16401           disabled : false,  // or use a function..
16402         })
16403      
16404      * * @param {Object} details about module
16405      */
16406     register : function(obj) {
16407                 
16408         Roo.XComponent.event.fireEvent('register', obj);
16409         switch(typeof(obj.disabled) ) {
16410                 
16411             case 'undefined':
16412                 break;
16413             
16414             case 'function':
16415                 if ( obj.disabled() ) {
16416                         return;
16417                 }
16418                 break;
16419             
16420             default:
16421                 if (obj.disabled || obj.region == '#disabled') {
16422                         return;
16423                 }
16424                 break;
16425         }
16426                 
16427         this.modules.push(obj);
16428          
16429     },
16430     /**
16431      * convert a string to an object..
16432      * eg. 'AAA.BBB' -> finds AAA.BBB
16433
16434      */
16435     
16436     toObject : function(str)
16437     {
16438         if (!str || typeof(str) == 'object') {
16439             return str;
16440         }
16441         if (str.substring(0,1) == '#') {
16442             return str;
16443         }
16444
16445         var ar = str.split('.');
16446         var rt, o;
16447         rt = ar.shift();
16448             /** eval:var:o */
16449         try {
16450             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16451         } catch (e) {
16452             throw "Module not found : " + str;
16453         }
16454         
16455         if (o === false) {
16456             throw "Module not found : " + str;
16457         }
16458         Roo.each(ar, function(e) {
16459             if (typeof(o[e]) == 'undefined') {
16460                 throw "Module not found : " + str;
16461             }
16462             o = o[e];
16463         });
16464         
16465         return o;
16466         
16467     },
16468     
16469     
16470     /**
16471      * move modules into their correct place in the tree..
16472      * 
16473      */
16474     preBuild : function ()
16475     {
16476         var _t = this;
16477         Roo.each(this.modules , function (obj)
16478         {
16479             Roo.XComponent.event.fireEvent('beforebuild', obj);
16480             
16481             var opar = obj.parent;
16482             try { 
16483                 obj.parent = this.toObject(opar);
16484             } catch(e) {
16485                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16486                 return;
16487             }
16488             
16489             if (!obj.parent) {
16490                 Roo.debug && Roo.log("GOT top level module");
16491                 Roo.debug && Roo.log(obj);
16492                 obj.modules = new Roo.util.MixedCollection(false, 
16493                     function(o) { return o.order + '' }
16494                 );
16495                 this.topModule = obj;
16496                 return;
16497             }
16498                         // parent is a string (usually a dom element name..)
16499             if (typeof(obj.parent) == 'string') {
16500                 this.elmodules.push(obj);
16501                 return;
16502             }
16503             if (obj.parent.constructor != Roo.XComponent) {
16504                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16505             }
16506             if (!obj.parent.modules) {
16507                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16508                     function(o) { return o.order + '' }
16509                 );
16510             }
16511             if (obj.parent.disabled) {
16512                 obj.disabled = true;
16513             }
16514             obj.parent.modules.add(obj);
16515         }, this);
16516     },
16517     
16518      /**
16519      * make a list of modules to build.
16520      * @return {Array} list of modules. 
16521      */ 
16522     
16523     buildOrder : function()
16524     {
16525         var _this = this;
16526         var cmp = function(a,b) {   
16527             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16528         };
16529         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16530             throw "No top level modules to build";
16531         }
16532         
16533         // make a flat list in order of modules to build.
16534         var mods = this.topModule ? [ this.topModule ] : [];
16535                 
16536         
16537         // elmodules (is a list of DOM based modules )
16538         Roo.each(this.elmodules, function(e) {
16539             mods.push(e);
16540             if (!this.topModule &&
16541                 typeof(e.parent) == 'string' &&
16542                 e.parent.substring(0,1) == '#' &&
16543                 Roo.get(e.parent.substr(1))
16544                ) {
16545                 
16546                 _this.topModule = e;
16547             }
16548             
16549         });
16550
16551         
16552         // add modules to their parents..
16553         var addMod = function(m) {
16554             Roo.debug && Roo.log("build Order: add: " + m.name);
16555                 
16556             mods.push(m);
16557             if (m.modules && !m.disabled) {
16558                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16559                 m.modules.keySort('ASC',  cmp );
16560                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16561     
16562                 m.modules.each(addMod);
16563             } else {
16564                 Roo.debug && Roo.log("build Order: no child modules");
16565             }
16566             // not sure if this is used any more..
16567             if (m.finalize) {
16568                 m.finalize.name = m.name + " (clean up) ";
16569                 mods.push(m.finalize);
16570             }
16571             
16572         }
16573         if (this.topModule && this.topModule.modules) { 
16574             this.topModule.modules.keySort('ASC',  cmp );
16575             this.topModule.modules.each(addMod);
16576         } 
16577         return mods;
16578     },
16579     
16580      /**
16581      * Build the registered modules.
16582      * @param {Object} parent element.
16583      * @param {Function} optional method to call after module has been added.
16584      * 
16585      */ 
16586    
16587     build : function(opts) 
16588     {
16589         
16590         if (typeof(opts) != 'undefined') {
16591             Roo.apply(this,opts);
16592         }
16593         
16594         this.preBuild();
16595         var mods = this.buildOrder();
16596       
16597         //this.allmods = mods;
16598         //Roo.debug && Roo.log(mods);
16599         //return;
16600         if (!mods.length) { // should not happen
16601             throw "NO modules!!!";
16602         }
16603         
16604         
16605         var msg = "Building Interface...";
16606         // flash it up as modal - so we store the mask!?
16607         if (!this.hideProgress && Roo.MessageBox) {
16608             Roo.MessageBox.show({ title: 'loading' });
16609             Roo.MessageBox.show({
16610                title: "Please wait...",
16611                msg: msg,
16612                width:450,
16613                progress:true,
16614                buttons : false,
16615                closable:false,
16616                modal: false
16617               
16618             });
16619         }
16620         var total = mods.length;
16621         
16622         var _this = this;
16623         var progressRun = function() {
16624             if (!mods.length) {
16625                 Roo.debug && Roo.log('hide?');
16626                 if (!this.hideProgress && Roo.MessageBox) {
16627                     Roo.MessageBox.hide();
16628                 }
16629                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16630                 
16631                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16632                 
16633                 // THE END...
16634                 return false;   
16635             }
16636             
16637             var m = mods.shift();
16638             
16639             
16640             Roo.debug && Roo.log(m);
16641             // not sure if this is supported any more.. - modules that are are just function
16642             if (typeof(m) == 'function') { 
16643                 m.call(this);
16644                 return progressRun.defer(10, _this);
16645             } 
16646             
16647             
16648             msg = "Building Interface " + (total  - mods.length) + 
16649                     " of " + total + 
16650                     (m.name ? (' - ' + m.name) : '');
16651                         Roo.debug && Roo.log(msg);
16652             if (!_this.hideProgress &&  Roo.MessageBox) { 
16653                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16654             }
16655             
16656          
16657             // is the module disabled?
16658             var disabled = (typeof(m.disabled) == 'function') ?
16659                 m.disabled.call(m.module.disabled) : m.disabled;    
16660             
16661             
16662             if (disabled) {
16663                 return progressRun(); // we do not update the display!
16664             }
16665             
16666             // now build 
16667             
16668                         
16669                         
16670             m.render();
16671             // it's 10 on top level, and 1 on others??? why...
16672             return progressRun.defer(10, _this);
16673              
16674         }
16675         progressRun.defer(1, _this);
16676      
16677         
16678         
16679     },
16680     /**
16681      * Overlay a set of modified strings onto a component
16682      * This is dependant on our builder exporting the strings and 'named strings' elements.
16683      * 
16684      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16685      * @param {Object} associative array of 'named' string and it's new value.
16686      * 
16687      */
16688         overlayStrings : function( component, strings )
16689     {
16690         if (typeof(component['_named_strings']) == 'undefined') {
16691             throw "ERROR: component does not have _named_strings";
16692         }
16693         for ( var k in strings ) {
16694             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16695             if (md !== false) {
16696                 component['_strings'][md] = strings[k];
16697             } else {
16698                 Roo.log('could not find named string: ' + k + ' in');
16699                 Roo.log(component);
16700             }
16701             
16702         }
16703         
16704     },
16705     
16706         
16707         /**
16708          * Event Object.
16709          *
16710          *
16711          */
16712         event: false, 
16713     /**
16714          * wrapper for event.on - aliased later..  
16715          * Typically use to register a event handler for register:
16716          *
16717          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16718          *
16719          */
16720     on : false
16721    
16722     
16723     
16724 });
16725
16726 Roo.XComponent.event = new Roo.util.Observable({
16727                 events : { 
16728                         /**
16729                          * @event register
16730                          * Fires when an Component is registered,
16731                          * set the disable property on the Component to stop registration.
16732                          * @param {Roo.XComponent} c the component being registerd.
16733                          * 
16734                          */
16735                         'register' : true,
16736             /**
16737                          * @event beforebuild
16738                          * Fires before each Component is built
16739                          * can be used to apply permissions.
16740                          * @param {Roo.XComponent} c the component being registerd.
16741                          * 
16742                          */
16743                         'beforebuild' : true,
16744                         /**
16745                          * @event buildcomplete
16746                          * Fires on the top level element when all elements have been built
16747                          * @param {Roo.XComponent} the top level component.
16748                          */
16749                         'buildcomplete' : true
16750                         
16751                 }
16752 });
16753
16754 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16755  //
16756  /**
16757  * marked - a markdown parser
16758  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16759  * https://github.com/chjj/marked
16760  */
16761
16762
16763 /**
16764  *
16765  * Roo.Markdown - is a very crude wrapper around marked..
16766  *
16767  * usage:
16768  * 
16769  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16770  * 
16771  * Note: move the sample code to the bottom of this
16772  * file before uncommenting it.
16773  *
16774  */
16775
16776 Roo.Markdown = {};
16777 Roo.Markdown.toHtml = function(text) {
16778     
16779     var c = new Roo.Markdown.marked.setOptions({
16780             renderer: new Roo.Markdown.marked.Renderer(),
16781             gfm: true,
16782             tables: true,
16783             breaks: false,
16784             pedantic: false,
16785             sanitize: false,
16786             smartLists: true,
16787             smartypants: false
16788           });
16789     // A FEW HACKS!!?
16790     
16791     text = text.replace(/\\\n/g,' ');
16792     return Roo.Markdown.marked(text);
16793 };
16794 //
16795 // converter
16796 //
16797 // Wraps all "globals" so that the only thing
16798 // exposed is makeHtml().
16799 //
16800 (function() {
16801     
16802     /**
16803      * Block-Level Grammar
16804      */
16805     
16806     var block = {
16807       newline: /^\n+/,
16808       code: /^( {4}[^\n]+\n*)+/,
16809       fences: noop,
16810       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16811       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16812       nptable: noop,
16813       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16814       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16815       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16816       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16817       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16818       table: noop,
16819       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16820       text: /^[^\n]+/
16821     };
16822     
16823     block.bullet = /(?:[*+-]|\d+\.)/;
16824     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16825     block.item = replace(block.item, 'gm')
16826       (/bull/g, block.bullet)
16827       ();
16828     
16829     block.list = replace(block.list)
16830       (/bull/g, block.bullet)
16831       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16832       ('def', '\\n+(?=' + block.def.source + ')')
16833       ();
16834     
16835     block.blockquote = replace(block.blockquote)
16836       ('def', block.def)
16837       ();
16838     
16839     block._tag = '(?!(?:'
16840       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16841       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16842       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16843     
16844     block.html = replace(block.html)
16845       ('comment', /<!--[\s\S]*?-->/)
16846       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16847       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16848       (/tag/g, block._tag)
16849       ();
16850     
16851     block.paragraph = replace(block.paragraph)
16852       ('hr', block.hr)
16853       ('heading', block.heading)
16854       ('lheading', block.lheading)
16855       ('blockquote', block.blockquote)
16856       ('tag', '<' + block._tag)
16857       ('def', block.def)
16858       ();
16859     
16860     /**
16861      * Normal Block Grammar
16862      */
16863     
16864     block.normal = merge({}, block);
16865     
16866     /**
16867      * GFM Block Grammar
16868      */
16869     
16870     block.gfm = merge({}, block.normal, {
16871       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16872       paragraph: /^/,
16873       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16874     });
16875     
16876     block.gfm.paragraph = replace(block.paragraph)
16877       ('(?!', '(?!'
16878         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16879         + block.list.source.replace('\\1', '\\3') + '|')
16880       ();
16881     
16882     /**
16883      * GFM + Tables Block Grammar
16884      */
16885     
16886     block.tables = merge({}, block.gfm, {
16887       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16888       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16889     });
16890     
16891     /**
16892      * Block Lexer
16893      */
16894     
16895     function Lexer(options) {
16896       this.tokens = [];
16897       this.tokens.links = {};
16898       this.options = options || marked.defaults;
16899       this.rules = block.normal;
16900     
16901       if (this.options.gfm) {
16902         if (this.options.tables) {
16903           this.rules = block.tables;
16904         } else {
16905           this.rules = block.gfm;
16906         }
16907       }
16908     }
16909     
16910     /**
16911      * Expose Block Rules
16912      */
16913     
16914     Lexer.rules = block;
16915     
16916     /**
16917      * Static Lex Method
16918      */
16919     
16920     Lexer.lex = function(src, options) {
16921       var lexer = new Lexer(options);
16922       return lexer.lex(src);
16923     };
16924     
16925     /**
16926      * Preprocessing
16927      */
16928     
16929     Lexer.prototype.lex = function(src) {
16930       src = src
16931         .replace(/\r\n|\r/g, '\n')
16932         .replace(/\t/g, '    ')
16933         .replace(/\u00a0/g, ' ')
16934         .replace(/\u2424/g, '\n');
16935     
16936       return this.token(src, true);
16937     };
16938     
16939     /**
16940      * Lexing
16941      */
16942     
16943     Lexer.prototype.token = function(src, top, bq) {
16944       var src = src.replace(/^ +$/gm, '')
16945         , next
16946         , loose
16947         , cap
16948         , bull
16949         , b
16950         , item
16951         , space
16952         , i
16953         , l;
16954     
16955       while (src) {
16956         // newline
16957         if (cap = this.rules.newline.exec(src)) {
16958           src = src.substring(cap[0].length);
16959           if (cap[0].length > 1) {
16960             this.tokens.push({
16961               type: 'space'
16962             });
16963           }
16964         }
16965     
16966         // code
16967         if (cap = this.rules.code.exec(src)) {
16968           src = src.substring(cap[0].length);
16969           cap = cap[0].replace(/^ {4}/gm, '');
16970           this.tokens.push({
16971             type: 'code',
16972             text: !this.options.pedantic
16973               ? cap.replace(/\n+$/, '')
16974               : cap
16975           });
16976           continue;
16977         }
16978     
16979         // fences (gfm)
16980         if (cap = this.rules.fences.exec(src)) {
16981           src = src.substring(cap[0].length);
16982           this.tokens.push({
16983             type: 'code',
16984             lang: cap[2],
16985             text: cap[3] || ''
16986           });
16987           continue;
16988         }
16989     
16990         // heading
16991         if (cap = this.rules.heading.exec(src)) {
16992           src = src.substring(cap[0].length);
16993           this.tokens.push({
16994             type: 'heading',
16995             depth: cap[1].length,
16996             text: cap[2]
16997           });
16998           continue;
16999         }
17000     
17001         // table no leading pipe (gfm)
17002         if (top && (cap = this.rules.nptable.exec(src))) {
17003           src = src.substring(cap[0].length);
17004     
17005           item = {
17006             type: 'table',
17007             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17008             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17009             cells: cap[3].replace(/\n$/, '').split('\n')
17010           };
17011     
17012           for (i = 0; i < item.align.length; i++) {
17013             if (/^ *-+: *$/.test(item.align[i])) {
17014               item.align[i] = 'right';
17015             } else if (/^ *:-+: *$/.test(item.align[i])) {
17016               item.align[i] = 'center';
17017             } else if (/^ *:-+ *$/.test(item.align[i])) {
17018               item.align[i] = 'left';
17019             } else {
17020               item.align[i] = null;
17021             }
17022           }
17023     
17024           for (i = 0; i < item.cells.length; i++) {
17025             item.cells[i] = item.cells[i].split(/ *\| */);
17026           }
17027     
17028           this.tokens.push(item);
17029     
17030           continue;
17031         }
17032     
17033         // lheading
17034         if (cap = this.rules.lheading.exec(src)) {
17035           src = src.substring(cap[0].length);
17036           this.tokens.push({
17037             type: 'heading',
17038             depth: cap[2] === '=' ? 1 : 2,
17039             text: cap[1]
17040           });
17041           continue;
17042         }
17043     
17044         // hr
17045         if (cap = this.rules.hr.exec(src)) {
17046           src = src.substring(cap[0].length);
17047           this.tokens.push({
17048             type: 'hr'
17049           });
17050           continue;
17051         }
17052     
17053         // blockquote
17054         if (cap = this.rules.blockquote.exec(src)) {
17055           src = src.substring(cap[0].length);
17056     
17057           this.tokens.push({
17058             type: 'blockquote_start'
17059           });
17060     
17061           cap = cap[0].replace(/^ *> ?/gm, '');
17062     
17063           // Pass `top` to keep the current
17064           // "toplevel" state. This is exactly
17065           // how markdown.pl works.
17066           this.token(cap, top, true);
17067     
17068           this.tokens.push({
17069             type: 'blockquote_end'
17070           });
17071     
17072           continue;
17073         }
17074     
17075         // list
17076         if (cap = this.rules.list.exec(src)) {
17077           src = src.substring(cap[0].length);
17078           bull = cap[2];
17079     
17080           this.tokens.push({
17081             type: 'list_start',
17082             ordered: bull.length > 1
17083           });
17084     
17085           // Get each top-level item.
17086           cap = cap[0].match(this.rules.item);
17087     
17088           next = false;
17089           l = cap.length;
17090           i = 0;
17091     
17092           for (; i < l; i++) {
17093             item = cap[i];
17094     
17095             // Remove the list item's bullet
17096             // so it is seen as the next token.
17097             space = item.length;
17098             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17099     
17100             // Outdent whatever the
17101             // list item contains. Hacky.
17102             if (~item.indexOf('\n ')) {
17103               space -= item.length;
17104               item = !this.options.pedantic
17105                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17106                 : item.replace(/^ {1,4}/gm, '');
17107             }
17108     
17109             // Determine whether the next list item belongs here.
17110             // Backpedal if it does not belong in this list.
17111             if (this.options.smartLists && i !== l - 1) {
17112               b = block.bullet.exec(cap[i + 1])[0];
17113               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17114                 src = cap.slice(i + 1).join('\n') + src;
17115                 i = l - 1;
17116               }
17117             }
17118     
17119             // Determine whether item is loose or not.
17120             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17121             // for discount behavior.
17122             loose = next || /\n\n(?!\s*$)/.test(item);
17123             if (i !== l - 1) {
17124               next = item.charAt(item.length - 1) === '\n';
17125               if (!loose) { loose = next; }
17126             }
17127     
17128             this.tokens.push({
17129               type: loose
17130                 ? 'loose_item_start'
17131                 : 'list_item_start'
17132             });
17133     
17134             // Recurse.
17135             this.token(item, false, bq);
17136     
17137             this.tokens.push({
17138               type: 'list_item_end'
17139             });
17140           }
17141     
17142           this.tokens.push({
17143             type: 'list_end'
17144           });
17145     
17146           continue;
17147         }
17148     
17149         // html
17150         if (cap = this.rules.html.exec(src)) {
17151           src = src.substring(cap[0].length);
17152           this.tokens.push({
17153             type: this.options.sanitize
17154               ? 'paragraph'
17155               : 'html',
17156             pre: !this.options.sanitizer
17157               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17158             text: cap[0]
17159           });
17160           continue;
17161         }
17162     
17163         // def
17164         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17165           src = src.substring(cap[0].length);
17166           this.tokens.links[cap[1].toLowerCase()] = {
17167             href: cap[2],
17168             title: cap[3]
17169           };
17170           continue;
17171         }
17172     
17173         // table (gfm)
17174         if (top && (cap = this.rules.table.exec(src))) {
17175           src = src.substring(cap[0].length);
17176     
17177           item = {
17178             type: 'table',
17179             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17180             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17181             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17182           };
17183     
17184           for (i = 0; i < item.align.length; i++) {
17185             if (/^ *-+: *$/.test(item.align[i])) {
17186               item.align[i] = 'right';
17187             } else if (/^ *:-+: *$/.test(item.align[i])) {
17188               item.align[i] = 'center';
17189             } else if (/^ *:-+ *$/.test(item.align[i])) {
17190               item.align[i] = 'left';
17191             } else {
17192               item.align[i] = null;
17193             }
17194           }
17195     
17196           for (i = 0; i < item.cells.length; i++) {
17197             item.cells[i] = item.cells[i]
17198               .replace(/^ *\| *| *\| *$/g, '')
17199               .split(/ *\| */);
17200           }
17201     
17202           this.tokens.push(item);
17203     
17204           continue;
17205         }
17206     
17207         // top-level paragraph
17208         if (top && (cap = this.rules.paragraph.exec(src))) {
17209           src = src.substring(cap[0].length);
17210           this.tokens.push({
17211             type: 'paragraph',
17212             text: cap[1].charAt(cap[1].length - 1) === '\n'
17213               ? cap[1].slice(0, -1)
17214               : cap[1]
17215           });
17216           continue;
17217         }
17218     
17219         // text
17220         if (cap = this.rules.text.exec(src)) {
17221           // Top-level should never reach here.
17222           src = src.substring(cap[0].length);
17223           this.tokens.push({
17224             type: 'text',
17225             text: cap[0]
17226           });
17227           continue;
17228         }
17229     
17230         if (src) {
17231           throw new
17232             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17233         }
17234       }
17235     
17236       return this.tokens;
17237     };
17238     
17239     /**
17240      * Inline-Level Grammar
17241      */
17242     
17243     var inline = {
17244       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17245       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17246       url: noop,
17247       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17248       link: /^!?\[(inside)\]\(href\)/,
17249       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17250       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17251       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17252       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17253       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17254       br: /^ {2,}\n(?!\s*$)/,
17255       del: noop,
17256       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17257     };
17258     
17259     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17260     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17261     
17262     inline.link = replace(inline.link)
17263       ('inside', inline._inside)
17264       ('href', inline._href)
17265       ();
17266     
17267     inline.reflink = replace(inline.reflink)
17268       ('inside', inline._inside)
17269       ();
17270     
17271     /**
17272      * Normal Inline Grammar
17273      */
17274     
17275     inline.normal = merge({}, inline);
17276     
17277     /**
17278      * Pedantic Inline Grammar
17279      */
17280     
17281     inline.pedantic = merge({}, inline.normal, {
17282       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17283       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17284     });
17285     
17286     /**
17287      * GFM Inline Grammar
17288      */
17289     
17290     inline.gfm = merge({}, inline.normal, {
17291       escape: replace(inline.escape)('])', '~|])')(),
17292       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17293       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17294       text: replace(inline.text)
17295         (']|', '~]|')
17296         ('|', '|https?://|')
17297         ()
17298     });
17299     
17300     /**
17301      * GFM + Line Breaks Inline Grammar
17302      */
17303     
17304     inline.breaks = merge({}, inline.gfm, {
17305       br: replace(inline.br)('{2,}', '*')(),
17306       text: replace(inline.gfm.text)('{2,}', '*')()
17307     });
17308     
17309     /**
17310      * Inline Lexer & Compiler
17311      */
17312     
17313     function InlineLexer(links, options) {
17314       this.options = options || marked.defaults;
17315       this.links = links;
17316       this.rules = inline.normal;
17317       this.renderer = this.options.renderer || new Renderer;
17318       this.renderer.options = this.options;
17319     
17320       if (!this.links) {
17321         throw new
17322           Error('Tokens array requires a `links` property.');
17323       }
17324     
17325       if (this.options.gfm) {
17326         if (this.options.breaks) {
17327           this.rules = inline.breaks;
17328         } else {
17329           this.rules = inline.gfm;
17330         }
17331       } else if (this.options.pedantic) {
17332         this.rules = inline.pedantic;
17333       }
17334     }
17335     
17336     /**
17337      * Expose Inline Rules
17338      */
17339     
17340     InlineLexer.rules = inline;
17341     
17342     /**
17343      * Static Lexing/Compiling Method
17344      */
17345     
17346     InlineLexer.output = function(src, links, options) {
17347       var inline = new InlineLexer(links, options);
17348       return inline.output(src);
17349     };
17350     
17351     /**
17352      * Lexing/Compiling
17353      */
17354     
17355     InlineLexer.prototype.output = function(src) {
17356       var out = ''
17357         , link
17358         , text
17359         , href
17360         , cap;
17361     
17362       while (src) {
17363         // escape
17364         if (cap = this.rules.escape.exec(src)) {
17365           src = src.substring(cap[0].length);
17366           out += cap[1];
17367           continue;
17368         }
17369     
17370         // autolink
17371         if (cap = this.rules.autolink.exec(src)) {
17372           src = src.substring(cap[0].length);
17373           if (cap[2] === '@') {
17374             text = cap[1].charAt(6) === ':'
17375               ? this.mangle(cap[1].substring(7))
17376               : this.mangle(cap[1]);
17377             href = this.mangle('mailto:') + text;
17378           } else {
17379             text = escape(cap[1]);
17380             href = text;
17381           }
17382           out += this.renderer.link(href, null, text);
17383           continue;
17384         }
17385     
17386         // url (gfm)
17387         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17388           src = src.substring(cap[0].length);
17389           text = escape(cap[1]);
17390           href = text;
17391           out += this.renderer.link(href, null, text);
17392           continue;
17393         }
17394     
17395         // tag
17396         if (cap = this.rules.tag.exec(src)) {
17397           if (!this.inLink && /^<a /i.test(cap[0])) {
17398             this.inLink = true;
17399           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17400             this.inLink = false;
17401           }
17402           src = src.substring(cap[0].length);
17403           out += this.options.sanitize
17404             ? this.options.sanitizer
17405               ? this.options.sanitizer(cap[0])
17406               : escape(cap[0])
17407             : cap[0];
17408           continue;
17409         }
17410     
17411         // link
17412         if (cap = this.rules.link.exec(src)) {
17413           src = src.substring(cap[0].length);
17414           this.inLink = true;
17415           out += this.outputLink(cap, {
17416             href: cap[2],
17417             title: cap[3]
17418           });
17419           this.inLink = false;
17420           continue;
17421         }
17422     
17423         // reflink, nolink
17424         if ((cap = this.rules.reflink.exec(src))
17425             || (cap = this.rules.nolink.exec(src))) {
17426           src = src.substring(cap[0].length);
17427           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17428           link = this.links[link.toLowerCase()];
17429           if (!link || !link.href) {
17430             out += cap[0].charAt(0);
17431             src = cap[0].substring(1) + src;
17432             continue;
17433           }
17434           this.inLink = true;
17435           out += this.outputLink(cap, link);
17436           this.inLink = false;
17437           continue;
17438         }
17439     
17440         // strong
17441         if (cap = this.rules.strong.exec(src)) {
17442           src = src.substring(cap[0].length);
17443           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17444           continue;
17445         }
17446     
17447         // em
17448         if (cap = this.rules.em.exec(src)) {
17449           src = src.substring(cap[0].length);
17450           out += this.renderer.em(this.output(cap[2] || cap[1]));
17451           continue;
17452         }
17453     
17454         // code
17455         if (cap = this.rules.code.exec(src)) {
17456           src = src.substring(cap[0].length);
17457           out += this.renderer.codespan(escape(cap[2], true));
17458           continue;
17459         }
17460     
17461         // br
17462         if (cap = this.rules.br.exec(src)) {
17463           src = src.substring(cap[0].length);
17464           out += this.renderer.br();
17465           continue;
17466         }
17467     
17468         // del (gfm)
17469         if (cap = this.rules.del.exec(src)) {
17470           src = src.substring(cap[0].length);
17471           out += this.renderer.del(this.output(cap[1]));
17472           continue;
17473         }
17474     
17475         // text
17476         if (cap = this.rules.text.exec(src)) {
17477           src = src.substring(cap[0].length);
17478           out += this.renderer.text(escape(this.smartypants(cap[0])));
17479           continue;
17480         }
17481     
17482         if (src) {
17483           throw new
17484             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17485         }
17486       }
17487     
17488       return out;
17489     };
17490     
17491     /**
17492      * Compile Link
17493      */
17494     
17495     InlineLexer.prototype.outputLink = function(cap, link) {
17496       var href = escape(link.href)
17497         , title = link.title ? escape(link.title) : null;
17498     
17499       return cap[0].charAt(0) !== '!'
17500         ? this.renderer.link(href, title, this.output(cap[1]))
17501         : this.renderer.image(href, title, escape(cap[1]));
17502     };
17503     
17504     /**
17505      * Smartypants Transformations
17506      */
17507     
17508     InlineLexer.prototype.smartypants = function(text) {
17509       if (!this.options.smartypants)  { return text; }
17510       return text
17511         // em-dashes
17512         .replace(/---/g, '\u2014')
17513         // en-dashes
17514         .replace(/--/g, '\u2013')
17515         // opening singles
17516         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17517         // closing singles & apostrophes
17518         .replace(/'/g, '\u2019')
17519         // opening doubles
17520         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17521         // closing doubles
17522         .replace(/"/g, '\u201d')
17523         // ellipses
17524         .replace(/\.{3}/g, '\u2026');
17525     };
17526     
17527     /**
17528      * Mangle Links
17529      */
17530     
17531     InlineLexer.prototype.mangle = function(text) {
17532       if (!this.options.mangle) { return text; }
17533       var out = ''
17534         , l = text.length
17535         , i = 0
17536         , ch;
17537     
17538       for (; i < l; i++) {
17539         ch = text.charCodeAt(i);
17540         if (Math.random() > 0.5) {
17541           ch = 'x' + ch.toString(16);
17542         }
17543         out += '&#' + ch + ';';
17544       }
17545     
17546       return out;
17547     };
17548     
17549     /**
17550      * Renderer
17551      */
17552     
17553     function Renderer(options) {
17554       this.options = options || {};
17555     }
17556     
17557     Renderer.prototype.code = function(code, lang, escaped) {
17558       if (this.options.highlight) {
17559         var out = this.options.highlight(code, lang);
17560         if (out != null && out !== code) {
17561           escaped = true;
17562           code = out;
17563         }
17564       } else {
17565             // hack!!! - it's already escapeD?
17566             escaped = true;
17567       }
17568     
17569       if (!lang) {
17570         return '<pre><code>'
17571           + (escaped ? code : escape(code, true))
17572           + '\n</code></pre>';
17573       }
17574     
17575       return '<pre><code class="'
17576         + this.options.langPrefix
17577         + escape(lang, true)
17578         + '">'
17579         + (escaped ? code : escape(code, true))
17580         + '\n</code></pre>\n';
17581     };
17582     
17583     Renderer.prototype.blockquote = function(quote) {
17584       return '<blockquote>\n' + quote + '</blockquote>\n';
17585     };
17586     
17587     Renderer.prototype.html = function(html) {
17588       return html;
17589     };
17590     
17591     Renderer.prototype.heading = function(text, level, raw) {
17592       return '<h'
17593         + level
17594         + ' id="'
17595         + this.options.headerPrefix
17596         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17597         + '">'
17598         + text
17599         + '</h'
17600         + level
17601         + '>\n';
17602     };
17603     
17604     Renderer.prototype.hr = function() {
17605       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17606     };
17607     
17608     Renderer.prototype.list = function(body, ordered) {
17609       var type = ordered ? 'ol' : 'ul';
17610       return '<' + type + '>\n' + body + '</' + type + '>\n';
17611     };
17612     
17613     Renderer.prototype.listitem = function(text) {
17614       return '<li>' + text + '</li>\n';
17615     };
17616     
17617     Renderer.prototype.paragraph = function(text) {
17618       return '<p>' + text + '</p>\n';
17619     };
17620     
17621     Renderer.prototype.table = function(header, body) {
17622       return '<table class="table table-striped">\n'
17623         + '<thead>\n'
17624         + header
17625         + '</thead>\n'
17626         + '<tbody>\n'
17627         + body
17628         + '</tbody>\n'
17629         + '</table>\n';
17630     };
17631     
17632     Renderer.prototype.tablerow = function(content) {
17633       return '<tr>\n' + content + '</tr>\n';
17634     };
17635     
17636     Renderer.prototype.tablecell = function(content, flags) {
17637       var type = flags.header ? 'th' : 'td';
17638       var tag = flags.align
17639         ? '<' + type + ' style="text-align:' + flags.align + '">'
17640         : '<' + type + '>';
17641       return tag + content + '</' + type + '>\n';
17642     };
17643     
17644     // span level renderer
17645     Renderer.prototype.strong = function(text) {
17646       return '<strong>' + text + '</strong>';
17647     };
17648     
17649     Renderer.prototype.em = function(text) {
17650       return '<em>' + text + '</em>';
17651     };
17652     
17653     Renderer.prototype.codespan = function(text) {
17654       return '<code>' + text + '</code>';
17655     };
17656     
17657     Renderer.prototype.br = function() {
17658       return this.options.xhtml ? '<br/>' : '<br>';
17659     };
17660     
17661     Renderer.prototype.del = function(text) {
17662       return '<del>' + text + '</del>';
17663     };
17664     
17665     Renderer.prototype.link = function(href, title, text) {
17666       if (this.options.sanitize) {
17667         try {
17668           var prot = decodeURIComponent(unescape(href))
17669             .replace(/[^\w:]/g, '')
17670             .toLowerCase();
17671         } catch (e) {
17672           return '';
17673         }
17674         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17675           return '';
17676         }
17677       }
17678       var out = '<a href="' + href + '"';
17679       if (title) {
17680         out += ' title="' + title + '"';
17681       }
17682       out += '>' + text + '</a>';
17683       return out;
17684     };
17685     
17686     Renderer.prototype.image = function(href, title, text) {
17687       var out = '<img src="' + href + '" alt="' + text + '"';
17688       if (title) {
17689         out += ' title="' + title + '"';
17690       }
17691       out += this.options.xhtml ? '/>' : '>';
17692       return out;
17693     };
17694     
17695     Renderer.prototype.text = function(text) {
17696       return text;
17697     };
17698     
17699     /**
17700      * Parsing & Compiling
17701      */
17702     
17703     function Parser(options) {
17704       this.tokens = [];
17705       this.token = null;
17706       this.options = options || marked.defaults;
17707       this.options.renderer = this.options.renderer || new Renderer;
17708       this.renderer = this.options.renderer;
17709       this.renderer.options = this.options;
17710     }
17711     
17712     /**
17713      * Static Parse Method
17714      */
17715     
17716     Parser.parse = function(src, options, renderer) {
17717       var parser = new Parser(options, renderer);
17718       return parser.parse(src);
17719     };
17720     
17721     /**
17722      * Parse Loop
17723      */
17724     
17725     Parser.prototype.parse = function(src) {
17726       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17727       this.tokens = src.reverse();
17728     
17729       var out = '';
17730       while (this.next()) {
17731         out += this.tok();
17732       }
17733     
17734       return out;
17735     };
17736     
17737     /**
17738      * Next Token
17739      */
17740     
17741     Parser.prototype.next = function() {
17742       return this.token = this.tokens.pop();
17743     };
17744     
17745     /**
17746      * Preview Next Token
17747      */
17748     
17749     Parser.prototype.peek = function() {
17750       return this.tokens[this.tokens.length - 1] || 0;
17751     };
17752     
17753     /**
17754      * Parse Text Tokens
17755      */
17756     
17757     Parser.prototype.parseText = function() {
17758       var body = this.token.text;
17759     
17760       while (this.peek().type === 'text') {
17761         body += '\n' + this.next().text;
17762       }
17763     
17764       return this.inline.output(body);
17765     };
17766     
17767     /**
17768      * Parse Current Token
17769      */
17770     
17771     Parser.prototype.tok = function() {
17772       switch (this.token.type) {
17773         case 'space': {
17774           return '';
17775         }
17776         case 'hr': {
17777           return this.renderer.hr();
17778         }
17779         case 'heading': {
17780           return this.renderer.heading(
17781             this.inline.output(this.token.text),
17782             this.token.depth,
17783             this.token.text);
17784         }
17785         case 'code': {
17786           return this.renderer.code(this.token.text,
17787             this.token.lang,
17788             this.token.escaped);
17789         }
17790         case 'table': {
17791           var header = ''
17792             , body = ''
17793             , i
17794             , row
17795             , cell
17796             , flags
17797             , j;
17798     
17799           // header
17800           cell = '';
17801           for (i = 0; i < this.token.header.length; i++) {
17802             flags = { header: true, align: this.token.align[i] };
17803             cell += this.renderer.tablecell(
17804               this.inline.output(this.token.header[i]),
17805               { header: true, align: this.token.align[i] }
17806             );
17807           }
17808           header += this.renderer.tablerow(cell);
17809     
17810           for (i = 0; i < this.token.cells.length; i++) {
17811             row = this.token.cells[i];
17812     
17813             cell = '';
17814             for (j = 0; j < row.length; j++) {
17815               cell += this.renderer.tablecell(
17816                 this.inline.output(row[j]),
17817                 { header: false, align: this.token.align[j] }
17818               );
17819             }
17820     
17821             body += this.renderer.tablerow(cell);
17822           }
17823           return this.renderer.table(header, body);
17824         }
17825         case 'blockquote_start': {
17826           var body = '';
17827     
17828           while (this.next().type !== 'blockquote_end') {
17829             body += this.tok();
17830           }
17831     
17832           return this.renderer.blockquote(body);
17833         }
17834         case 'list_start': {
17835           var body = ''
17836             , ordered = this.token.ordered;
17837     
17838           while (this.next().type !== 'list_end') {
17839             body += this.tok();
17840           }
17841     
17842           return this.renderer.list(body, ordered);
17843         }
17844         case 'list_item_start': {
17845           var body = '';
17846     
17847           while (this.next().type !== 'list_item_end') {
17848             body += this.token.type === 'text'
17849               ? this.parseText()
17850               : this.tok();
17851           }
17852     
17853           return this.renderer.listitem(body);
17854         }
17855         case 'loose_item_start': {
17856           var body = '';
17857     
17858           while (this.next().type !== 'list_item_end') {
17859             body += this.tok();
17860           }
17861     
17862           return this.renderer.listitem(body);
17863         }
17864         case 'html': {
17865           var html = !this.token.pre && !this.options.pedantic
17866             ? this.inline.output(this.token.text)
17867             : this.token.text;
17868           return this.renderer.html(html);
17869         }
17870         case 'paragraph': {
17871           return this.renderer.paragraph(this.inline.output(this.token.text));
17872         }
17873         case 'text': {
17874           return this.renderer.paragraph(this.parseText());
17875         }
17876       }
17877     };
17878     
17879     /**
17880      * Helpers
17881      */
17882     
17883     function escape(html, encode) {
17884       return html
17885         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
17886         .replace(/</g, '&lt;')
17887         .replace(/>/g, '&gt;')
17888         .replace(/"/g, '&quot;')
17889         .replace(/'/g, '&#39;');
17890     }
17891     
17892     function unescape(html) {
17893         // explicitly match decimal, hex, and named HTML entities 
17894       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
17895         n = n.toLowerCase();
17896         if (n === 'colon') { return ':'; }
17897         if (n.charAt(0) === '#') {
17898           return n.charAt(1) === 'x'
17899             ? String.fromCharCode(parseInt(n.substring(2), 16))
17900             : String.fromCharCode(+n.substring(1));
17901         }
17902         return '';
17903       });
17904     }
17905     
17906     function replace(regex, opt) {
17907       regex = regex.source;
17908       opt = opt || '';
17909       return function self(name, val) {
17910         if (!name) { return new RegExp(regex, opt); }
17911         val = val.source || val;
17912         val = val.replace(/(^|[^\[])\^/g, '$1');
17913         regex = regex.replace(name, val);
17914         return self;
17915       };
17916     }
17917     
17918     function noop() {}
17919     noop.exec = noop;
17920     
17921     function merge(obj) {
17922       var i = 1
17923         , target
17924         , key;
17925     
17926       for (; i < arguments.length; i++) {
17927         target = arguments[i];
17928         for (key in target) {
17929           if (Object.prototype.hasOwnProperty.call(target, key)) {
17930             obj[key] = target[key];
17931           }
17932         }
17933       }
17934     
17935       return obj;
17936     }
17937     
17938     
17939     /**
17940      * Marked
17941      */
17942     
17943     function marked(src, opt, callback) {
17944       if (callback || typeof opt === 'function') {
17945         if (!callback) {
17946           callback = opt;
17947           opt = null;
17948         }
17949     
17950         opt = merge({}, marked.defaults, opt || {});
17951     
17952         var highlight = opt.highlight
17953           , tokens
17954           , pending
17955           , i = 0;
17956     
17957         try {
17958           tokens = Lexer.lex(src, opt)
17959         } catch (e) {
17960           return callback(e);
17961         }
17962     
17963         pending = tokens.length;
17964     
17965         var done = function(err) {
17966           if (err) {
17967             opt.highlight = highlight;
17968             return callback(err);
17969           }
17970     
17971           var out;
17972     
17973           try {
17974             out = Parser.parse(tokens, opt);
17975           } catch (e) {
17976             err = e;
17977           }
17978     
17979           opt.highlight = highlight;
17980     
17981           return err
17982             ? callback(err)
17983             : callback(null, out);
17984         };
17985     
17986         if (!highlight || highlight.length < 3) {
17987           return done();
17988         }
17989     
17990         delete opt.highlight;
17991     
17992         if (!pending) { return done(); }
17993     
17994         for (; i < tokens.length; i++) {
17995           (function(token) {
17996             if (token.type !== 'code') {
17997               return --pending || done();
17998             }
17999             return highlight(token.text, token.lang, function(err, code) {
18000               if (err) { return done(err); }
18001               if (code == null || code === token.text) {
18002                 return --pending || done();
18003               }
18004               token.text = code;
18005               token.escaped = true;
18006               --pending || done();
18007             });
18008           })(tokens[i]);
18009         }
18010     
18011         return;
18012       }
18013       try {
18014         if (opt) { opt = merge({}, marked.defaults, opt); }
18015         return Parser.parse(Lexer.lex(src, opt), opt);
18016       } catch (e) {
18017         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18018         if ((opt || marked.defaults).silent) {
18019           return '<p>An error occured:</p><pre>'
18020             + escape(e.message + '', true)
18021             + '</pre>';
18022         }
18023         throw e;
18024       }
18025     }
18026     
18027     /**
18028      * Options
18029      */
18030     
18031     marked.options =
18032     marked.setOptions = function(opt) {
18033       merge(marked.defaults, opt);
18034       return marked;
18035     };
18036     
18037     marked.defaults = {
18038       gfm: true,
18039       tables: true,
18040       breaks: false,
18041       pedantic: false,
18042       sanitize: false,
18043       sanitizer: null,
18044       mangle: true,
18045       smartLists: false,
18046       silent: false,
18047       highlight: null,
18048       langPrefix: 'lang-',
18049       smartypants: false,
18050       headerPrefix: '',
18051       renderer: new Renderer,
18052       xhtml: false
18053     };
18054     
18055     /**
18056      * Expose
18057      */
18058     
18059     marked.Parser = Parser;
18060     marked.parser = Parser.parse;
18061     
18062     marked.Renderer = Renderer;
18063     
18064     marked.Lexer = Lexer;
18065     marked.lexer = Lexer.lex;
18066     
18067     marked.InlineLexer = InlineLexer;
18068     marked.inlineLexer = InlineLexer.output;
18069     
18070     marked.parse = marked;
18071     
18072     Roo.Markdown.marked = marked;
18073
18074 })();/*
18075  * Based on:
18076  * Ext JS Library 1.1.1
18077  * Copyright(c) 2006-2007, Ext JS, LLC.
18078  *
18079  * Originally Released Under LGPL - original licence link has changed is not relivant.
18080  *
18081  * Fork - LGPL
18082  * <script type="text/javascript">
18083  */
18084
18085
18086
18087 /*
18088  * These classes are derivatives of the similarly named classes in the YUI Library.
18089  * The original license:
18090  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18091  * Code licensed under the BSD License:
18092  * http://developer.yahoo.net/yui/license.txt
18093  */
18094
18095 (function() {
18096
18097 var Event=Roo.EventManager;
18098 var Dom=Roo.lib.Dom;
18099
18100 /**
18101  * @class Roo.dd.DragDrop
18102  * @extends Roo.util.Observable
18103  * Defines the interface and base operation of items that that can be
18104  * dragged or can be drop targets.  It was designed to be extended, overriding
18105  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18106  * Up to three html elements can be associated with a DragDrop instance:
18107  * <ul>
18108  * <li>linked element: the element that is passed into the constructor.
18109  * This is the element which defines the boundaries for interaction with
18110  * other DragDrop objects.</li>
18111  * <li>handle element(s): The drag operation only occurs if the element that
18112  * was clicked matches a handle element.  By default this is the linked
18113  * element, but there are times that you will want only a portion of the
18114  * linked element to initiate the drag operation, and the setHandleElId()
18115  * method provides a way to define this.</li>
18116  * <li>drag element: this represents the element that would be moved along
18117  * with the cursor during a drag operation.  By default, this is the linked
18118  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18119  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18120  * </li>
18121  * </ul>
18122  * This class should not be instantiated until the onload event to ensure that
18123  * the associated elements are available.
18124  * The following would define a DragDrop obj that would interact with any
18125  * other DragDrop obj in the "group1" group:
18126  * <pre>
18127  *  dd = new Roo.dd.DragDrop("div1", "group1");
18128  * </pre>
18129  * Since none of the event handlers have been implemented, nothing would
18130  * actually happen if you were to run the code above.  Normally you would
18131  * override this class or one of the default implementations, but you can
18132  * also override the methods you want on an instance of the class...
18133  * <pre>
18134  *  dd.onDragDrop = function(e, id) {
18135  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18136  *  }
18137  * </pre>
18138  * @constructor
18139  * @param {String} id of the element that is linked to this instance
18140  * @param {String} sGroup the group of related DragDrop objects
18141  * @param {object} config an object containing configurable attributes
18142  *                Valid properties for DragDrop:
18143  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18144  */
18145 Roo.dd.DragDrop = function(id, sGroup, config) {
18146     if (id) {
18147         this.init(id, sGroup, config);
18148     }
18149     
18150 };
18151
18152 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18153
18154     /**
18155      * The id of the element associated with this object.  This is what we
18156      * refer to as the "linked element" because the size and position of
18157      * this element is used to determine when the drag and drop objects have
18158      * interacted.
18159      * @property id
18160      * @type String
18161      */
18162     id: null,
18163
18164     /**
18165      * Configuration attributes passed into the constructor
18166      * @property config
18167      * @type object
18168      */
18169     config: null,
18170
18171     /**
18172      * The id of the element that will be dragged.  By default this is same
18173      * as the linked element , but could be changed to another element. Ex:
18174      * Roo.dd.DDProxy
18175      * @property dragElId
18176      * @type String
18177      * @private
18178      */
18179     dragElId: null,
18180
18181     /**
18182      * the id of the element that initiates the drag operation.  By default
18183      * this is the linked element, but could be changed to be a child of this
18184      * element.  This lets us do things like only starting the drag when the
18185      * header element within the linked html element is clicked.
18186      * @property handleElId
18187      * @type String
18188      * @private
18189      */
18190     handleElId: null,
18191
18192     /**
18193      * An associative array of HTML tags that will be ignored if clicked.
18194      * @property invalidHandleTypes
18195      * @type {string: string}
18196      */
18197     invalidHandleTypes: null,
18198
18199     /**
18200      * An associative array of ids for elements that will be ignored if clicked
18201      * @property invalidHandleIds
18202      * @type {string: string}
18203      */
18204     invalidHandleIds: null,
18205
18206     /**
18207      * An indexted array of css class names for elements that will be ignored
18208      * if clicked.
18209      * @property invalidHandleClasses
18210      * @type string[]
18211      */
18212     invalidHandleClasses: null,
18213
18214     /**
18215      * The linked element's absolute X position at the time the drag was
18216      * started
18217      * @property startPageX
18218      * @type int
18219      * @private
18220      */
18221     startPageX: 0,
18222
18223     /**
18224      * The linked element's absolute X position at the time the drag was
18225      * started
18226      * @property startPageY
18227      * @type int
18228      * @private
18229      */
18230     startPageY: 0,
18231
18232     /**
18233      * The group defines a logical collection of DragDrop objects that are
18234      * related.  Instances only get events when interacting with other
18235      * DragDrop object in the same group.  This lets us define multiple
18236      * groups using a single DragDrop subclass if we want.
18237      * @property groups
18238      * @type {string: string}
18239      */
18240     groups: null,
18241
18242     /**
18243      * Individual drag/drop instances can be locked.  This will prevent
18244      * onmousedown start drag.
18245      * @property locked
18246      * @type boolean
18247      * @private
18248      */
18249     locked: false,
18250
18251     /**
18252      * Lock this instance
18253      * @method lock
18254      */
18255     lock: function() { this.locked = true; },
18256
18257     /**
18258      * Unlock this instace
18259      * @method unlock
18260      */
18261     unlock: function() { this.locked = false; },
18262
18263     /**
18264      * By default, all insances can be a drop target.  This can be disabled by
18265      * setting isTarget to false.
18266      * @method isTarget
18267      * @type boolean
18268      */
18269     isTarget: true,
18270
18271     /**
18272      * The padding configured for this drag and drop object for calculating
18273      * the drop zone intersection with this object.
18274      * @method padding
18275      * @type int[]
18276      */
18277     padding: null,
18278
18279     /**
18280      * Cached reference to the linked element
18281      * @property _domRef
18282      * @private
18283      */
18284     _domRef: null,
18285
18286     /**
18287      * Internal typeof flag
18288      * @property __ygDragDrop
18289      * @private
18290      */
18291     __ygDragDrop: true,
18292
18293     /**
18294      * Set to true when horizontal contraints are applied
18295      * @property constrainX
18296      * @type boolean
18297      * @private
18298      */
18299     constrainX: false,
18300
18301     /**
18302      * Set to true when vertical contraints are applied
18303      * @property constrainY
18304      * @type boolean
18305      * @private
18306      */
18307     constrainY: false,
18308
18309     /**
18310      * The left constraint
18311      * @property minX
18312      * @type int
18313      * @private
18314      */
18315     minX: 0,
18316
18317     /**
18318      * The right constraint
18319      * @property maxX
18320      * @type int
18321      * @private
18322      */
18323     maxX: 0,
18324
18325     /**
18326      * The up constraint
18327      * @property minY
18328      * @type int
18329      * @type int
18330      * @private
18331      */
18332     minY: 0,
18333
18334     /**
18335      * The down constraint
18336      * @property maxY
18337      * @type int
18338      * @private
18339      */
18340     maxY: 0,
18341
18342     /**
18343      * Maintain offsets when we resetconstraints.  Set to true when you want
18344      * the position of the element relative to its parent to stay the same
18345      * when the page changes
18346      *
18347      * @property maintainOffset
18348      * @type boolean
18349      */
18350     maintainOffset: false,
18351
18352     /**
18353      * Array of pixel locations the element will snap to if we specified a
18354      * horizontal graduation/interval.  This array is generated automatically
18355      * when you define a tick interval.
18356      * @property xTicks
18357      * @type int[]
18358      */
18359     xTicks: null,
18360
18361     /**
18362      * Array of pixel locations the element will snap to if we specified a
18363      * vertical graduation/interval.  This array is generated automatically
18364      * when you define a tick interval.
18365      * @property yTicks
18366      * @type int[]
18367      */
18368     yTicks: null,
18369
18370     /**
18371      * By default the drag and drop instance will only respond to the primary
18372      * button click (left button for a right-handed mouse).  Set to true to
18373      * allow drag and drop to start with any mouse click that is propogated
18374      * by the browser
18375      * @property primaryButtonOnly
18376      * @type boolean
18377      */
18378     primaryButtonOnly: true,
18379
18380     /**
18381      * The availabe property is false until the linked dom element is accessible.
18382      * @property available
18383      * @type boolean
18384      */
18385     available: false,
18386
18387     /**
18388      * By default, drags can only be initiated if the mousedown occurs in the
18389      * region the linked element is.  This is done in part to work around a
18390      * bug in some browsers that mis-report the mousedown if the previous
18391      * mouseup happened outside of the window.  This property is set to true
18392      * if outer handles are defined.
18393      *
18394      * @property hasOuterHandles
18395      * @type boolean
18396      * @default false
18397      */
18398     hasOuterHandles: false,
18399
18400     /**
18401      * Code that executes immediately before the startDrag event
18402      * @method b4StartDrag
18403      * @private
18404      */
18405     b4StartDrag: function(x, y) { },
18406
18407     /**
18408      * Abstract method called after a drag/drop object is clicked
18409      * and the drag or mousedown time thresholds have beeen met.
18410      * @method startDrag
18411      * @param {int} X click location
18412      * @param {int} Y click location
18413      */
18414     startDrag: function(x, y) { /* override this */ },
18415
18416     /**
18417      * Code that executes immediately before the onDrag event
18418      * @method b4Drag
18419      * @private
18420      */
18421     b4Drag: function(e) { },
18422
18423     /**
18424      * Abstract method called during the onMouseMove event while dragging an
18425      * object.
18426      * @method onDrag
18427      * @param {Event} e the mousemove event
18428      */
18429     onDrag: function(e) { /* override this */ },
18430
18431     /**
18432      * Abstract method called when this element fist begins hovering over
18433      * another DragDrop obj
18434      * @method onDragEnter
18435      * @param {Event} e the mousemove event
18436      * @param {String|DragDrop[]} id In POINT mode, the element
18437      * id this is hovering over.  In INTERSECT mode, an array of one or more
18438      * dragdrop items being hovered over.
18439      */
18440     onDragEnter: function(e, id) { /* override this */ },
18441
18442     /**
18443      * Code that executes immediately before the onDragOver event
18444      * @method b4DragOver
18445      * @private
18446      */
18447     b4DragOver: function(e) { },
18448
18449     /**
18450      * Abstract method called when this element is hovering over another
18451      * DragDrop obj
18452      * @method onDragOver
18453      * @param {Event} e the mousemove event
18454      * @param {String|DragDrop[]} id In POINT mode, the element
18455      * id this is hovering over.  In INTERSECT mode, an array of dd items
18456      * being hovered over.
18457      */
18458     onDragOver: function(e, id) { /* override this */ },
18459
18460     /**
18461      * Code that executes immediately before the onDragOut event
18462      * @method b4DragOut
18463      * @private
18464      */
18465     b4DragOut: function(e) { },
18466
18467     /**
18468      * Abstract method called when we are no longer hovering over an element
18469      * @method onDragOut
18470      * @param {Event} e the mousemove event
18471      * @param {String|DragDrop[]} id In POINT mode, the element
18472      * id this was hovering over.  In INTERSECT mode, an array of dd items
18473      * that the mouse is no longer over.
18474      */
18475     onDragOut: function(e, id) { /* override this */ },
18476
18477     /**
18478      * Code that executes immediately before the onDragDrop event
18479      * @method b4DragDrop
18480      * @private
18481      */
18482     b4DragDrop: function(e) { },
18483
18484     /**
18485      * Abstract method called when this item is dropped on another DragDrop
18486      * obj
18487      * @method onDragDrop
18488      * @param {Event} e the mouseup event
18489      * @param {String|DragDrop[]} id In POINT mode, the element
18490      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18491      * was dropped on.
18492      */
18493     onDragDrop: function(e, id) { /* override this */ },
18494
18495     /**
18496      * Abstract method called when this item is dropped on an area with no
18497      * drop target
18498      * @method onInvalidDrop
18499      * @param {Event} e the mouseup event
18500      */
18501     onInvalidDrop: function(e) { /* override this */ },
18502
18503     /**
18504      * Code that executes immediately before the endDrag event
18505      * @method b4EndDrag
18506      * @private
18507      */
18508     b4EndDrag: function(e) { },
18509
18510     /**
18511      * Fired when we are done dragging the object
18512      * @method endDrag
18513      * @param {Event} e the mouseup event
18514      */
18515     endDrag: function(e) { /* override this */ },
18516
18517     /**
18518      * Code executed immediately before the onMouseDown event
18519      * @method b4MouseDown
18520      * @param {Event} e the mousedown event
18521      * @private
18522      */
18523     b4MouseDown: function(e) {  },
18524
18525     /**
18526      * Event handler that fires when a drag/drop obj gets a mousedown
18527      * @method onMouseDown
18528      * @param {Event} e the mousedown event
18529      */
18530     onMouseDown: function(e) { /* override this */ },
18531
18532     /**
18533      * Event handler that fires when a drag/drop obj gets a mouseup
18534      * @method onMouseUp
18535      * @param {Event} e the mouseup event
18536      */
18537     onMouseUp: function(e) { /* override this */ },
18538
18539     /**
18540      * Override the onAvailable method to do what is needed after the initial
18541      * position was determined.
18542      * @method onAvailable
18543      */
18544     onAvailable: function () {
18545     },
18546
18547     /*
18548      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18549      * @type Object
18550      */
18551     defaultPadding : {left:0, right:0, top:0, bottom:0},
18552
18553     /*
18554      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18555  *
18556  * Usage:
18557  <pre><code>
18558  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18559                 { dragElId: "existingProxyDiv" });
18560  dd.startDrag = function(){
18561      this.constrainTo("parent-id");
18562  };
18563  </code></pre>
18564  * Or you can initalize it using the {@link Roo.Element} object:
18565  <pre><code>
18566  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18567      startDrag : function(){
18568          this.constrainTo("parent-id");
18569      }
18570  });
18571  </code></pre>
18572      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18573      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18574      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18575      * an object containing the sides to pad. For example: {right:10, bottom:10}
18576      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18577      */
18578     constrainTo : function(constrainTo, pad, inContent){
18579         if(typeof pad == "number"){
18580             pad = {left: pad, right:pad, top:pad, bottom:pad};
18581         }
18582         pad = pad || this.defaultPadding;
18583         var b = Roo.get(this.getEl()).getBox();
18584         var ce = Roo.get(constrainTo);
18585         var s = ce.getScroll();
18586         var c, cd = ce.dom;
18587         if(cd == document.body){
18588             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18589         }else{
18590             xy = ce.getXY();
18591             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18592         }
18593
18594
18595         var topSpace = b.y - c.y;
18596         var leftSpace = b.x - c.x;
18597
18598         this.resetConstraints();
18599         this.setXConstraint(leftSpace - (pad.left||0), // left
18600                 c.width - leftSpace - b.width - (pad.right||0) //right
18601         );
18602         this.setYConstraint(topSpace - (pad.top||0), //top
18603                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18604         );
18605     },
18606
18607     /**
18608      * Returns a reference to the linked element
18609      * @method getEl
18610      * @return {HTMLElement} the html element
18611      */
18612     getEl: function() {
18613         if (!this._domRef) {
18614             this._domRef = Roo.getDom(this.id);
18615         }
18616
18617         return this._domRef;
18618     },
18619
18620     /**
18621      * Returns a reference to the actual element to drag.  By default this is
18622      * the same as the html element, but it can be assigned to another
18623      * element. An example of this can be found in Roo.dd.DDProxy
18624      * @method getDragEl
18625      * @return {HTMLElement} the html element
18626      */
18627     getDragEl: function() {
18628         return Roo.getDom(this.dragElId);
18629     },
18630
18631     /**
18632      * Sets up the DragDrop object.  Must be called in the constructor of any
18633      * Roo.dd.DragDrop subclass
18634      * @method init
18635      * @param id the id of the linked element
18636      * @param {String} sGroup the group of related items
18637      * @param {object} config configuration attributes
18638      */
18639     init: function(id, sGroup, config) {
18640         this.initTarget(id, sGroup, config);
18641         if (!Roo.isTouch) {
18642             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18643         }
18644         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18645         // Event.on(this.id, "selectstart", Event.preventDefault);
18646     },
18647
18648     /**
18649      * Initializes Targeting functionality only... the object does not
18650      * get a mousedown handler.
18651      * @method initTarget
18652      * @param id the id of the linked element
18653      * @param {String} sGroup the group of related items
18654      * @param {object} config configuration attributes
18655      */
18656     initTarget: function(id, sGroup, config) {
18657
18658         // configuration attributes
18659         this.config = config || {};
18660
18661         // create a local reference to the drag and drop manager
18662         this.DDM = Roo.dd.DDM;
18663         // initialize the groups array
18664         this.groups = {};
18665
18666         // assume that we have an element reference instead of an id if the
18667         // parameter is not a string
18668         if (typeof id !== "string") {
18669             id = Roo.id(id);
18670         }
18671
18672         // set the id
18673         this.id = id;
18674
18675         // add to an interaction group
18676         this.addToGroup((sGroup) ? sGroup : "default");
18677
18678         // We don't want to register this as the handle with the manager
18679         // so we just set the id rather than calling the setter.
18680         this.handleElId = id;
18681
18682         // the linked element is the element that gets dragged by default
18683         this.setDragElId(id);
18684
18685         // by default, clicked anchors will not start drag operations.
18686         this.invalidHandleTypes = { A: "A" };
18687         this.invalidHandleIds = {};
18688         this.invalidHandleClasses = [];
18689
18690         this.applyConfig();
18691
18692         this.handleOnAvailable();
18693     },
18694
18695     /**
18696      * Applies the configuration parameters that were passed into the constructor.
18697      * This is supposed to happen at each level through the inheritance chain.  So
18698      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18699      * DragDrop in order to get all of the parameters that are available in
18700      * each object.
18701      * @method applyConfig
18702      */
18703     applyConfig: function() {
18704
18705         // configurable properties:
18706         //    padding, isTarget, maintainOffset, primaryButtonOnly
18707         this.padding           = this.config.padding || [0, 0, 0, 0];
18708         this.isTarget          = (this.config.isTarget !== false);
18709         this.maintainOffset    = (this.config.maintainOffset);
18710         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18711
18712     },
18713
18714     /**
18715      * Executed when the linked element is available
18716      * @method handleOnAvailable
18717      * @private
18718      */
18719     handleOnAvailable: function() {
18720         this.available = true;
18721         this.resetConstraints();
18722         this.onAvailable();
18723     },
18724
18725      /**
18726      * Configures the padding for the target zone in px.  Effectively expands
18727      * (or reduces) the virtual object size for targeting calculations.
18728      * Supports css-style shorthand; if only one parameter is passed, all sides
18729      * will have that padding, and if only two are passed, the top and bottom
18730      * will have the first param, the left and right the second.
18731      * @method setPadding
18732      * @param {int} iTop    Top pad
18733      * @param {int} iRight  Right pad
18734      * @param {int} iBot    Bot pad
18735      * @param {int} iLeft   Left pad
18736      */
18737     setPadding: function(iTop, iRight, iBot, iLeft) {
18738         // this.padding = [iLeft, iRight, iTop, iBot];
18739         if (!iRight && 0 !== iRight) {
18740             this.padding = [iTop, iTop, iTop, iTop];
18741         } else if (!iBot && 0 !== iBot) {
18742             this.padding = [iTop, iRight, iTop, iRight];
18743         } else {
18744             this.padding = [iTop, iRight, iBot, iLeft];
18745         }
18746     },
18747
18748     /**
18749      * Stores the initial placement of the linked element.
18750      * @method setInitialPosition
18751      * @param {int} diffX   the X offset, default 0
18752      * @param {int} diffY   the Y offset, default 0
18753      */
18754     setInitPosition: function(diffX, diffY) {
18755         var el = this.getEl();
18756
18757         if (!this.DDM.verifyEl(el)) {
18758             return;
18759         }
18760
18761         var dx = diffX || 0;
18762         var dy = diffY || 0;
18763
18764         var p = Dom.getXY( el );
18765
18766         this.initPageX = p[0] - dx;
18767         this.initPageY = p[1] - dy;
18768
18769         this.lastPageX = p[0];
18770         this.lastPageY = p[1];
18771
18772
18773         this.setStartPosition(p);
18774     },
18775
18776     /**
18777      * Sets the start position of the element.  This is set when the obj
18778      * is initialized, the reset when a drag is started.
18779      * @method setStartPosition
18780      * @param pos current position (from previous lookup)
18781      * @private
18782      */
18783     setStartPosition: function(pos) {
18784         var p = pos || Dom.getXY( this.getEl() );
18785         this.deltaSetXY = null;
18786
18787         this.startPageX = p[0];
18788         this.startPageY = p[1];
18789     },
18790
18791     /**
18792      * Add this instance to a group of related drag/drop objects.  All
18793      * instances belong to at least one group, and can belong to as many
18794      * groups as needed.
18795      * @method addToGroup
18796      * @param sGroup {string} the name of the group
18797      */
18798     addToGroup: function(sGroup) {
18799         this.groups[sGroup] = true;
18800         this.DDM.regDragDrop(this, sGroup);
18801     },
18802
18803     /**
18804      * Remove's this instance from the supplied interaction group
18805      * @method removeFromGroup
18806      * @param {string}  sGroup  The group to drop
18807      */
18808     removeFromGroup: function(sGroup) {
18809         if (this.groups[sGroup]) {
18810             delete this.groups[sGroup];
18811         }
18812
18813         this.DDM.removeDDFromGroup(this, sGroup);
18814     },
18815
18816     /**
18817      * Allows you to specify that an element other than the linked element
18818      * will be moved with the cursor during a drag
18819      * @method setDragElId
18820      * @param id {string} the id of the element that will be used to initiate the drag
18821      */
18822     setDragElId: function(id) {
18823         this.dragElId = id;
18824     },
18825
18826     /**
18827      * Allows you to specify a child of the linked element that should be
18828      * used to initiate the drag operation.  An example of this would be if
18829      * you have a content div with text and links.  Clicking anywhere in the
18830      * content area would normally start the drag operation.  Use this method
18831      * to specify that an element inside of the content div is the element
18832      * that starts the drag operation.
18833      * @method setHandleElId
18834      * @param id {string} the id of the element that will be used to
18835      * initiate the drag.
18836      */
18837     setHandleElId: function(id) {
18838         if (typeof id !== "string") {
18839             id = Roo.id(id);
18840         }
18841         this.handleElId = id;
18842         this.DDM.regHandle(this.id, id);
18843     },
18844
18845     /**
18846      * Allows you to set an element outside of the linked element as a drag
18847      * handle
18848      * @method setOuterHandleElId
18849      * @param id the id of the element that will be used to initiate the drag
18850      */
18851     setOuterHandleElId: function(id) {
18852         if (typeof id !== "string") {
18853             id = Roo.id(id);
18854         }
18855         Event.on(id, "mousedown",
18856                 this.handleMouseDown, this);
18857         this.setHandleElId(id);
18858
18859         this.hasOuterHandles = true;
18860     },
18861
18862     /**
18863      * Remove all drag and drop hooks for this element
18864      * @method unreg
18865      */
18866     unreg: function() {
18867         Event.un(this.id, "mousedown",
18868                 this.handleMouseDown);
18869         Event.un(this.id, "touchstart",
18870                 this.handleMouseDown);
18871         this._domRef = null;
18872         this.DDM._remove(this);
18873     },
18874
18875     destroy : function(){
18876         this.unreg();
18877     },
18878
18879     /**
18880      * Returns true if this instance is locked, or the drag drop mgr is locked
18881      * (meaning that all drag/drop is disabled on the page.)
18882      * @method isLocked
18883      * @return {boolean} true if this obj or all drag/drop is locked, else
18884      * false
18885      */
18886     isLocked: function() {
18887         return (this.DDM.isLocked() || this.locked);
18888     },
18889
18890     /**
18891      * Fired when this object is clicked
18892      * @method handleMouseDown
18893      * @param {Event} e
18894      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18895      * @private
18896      */
18897     handleMouseDown: function(e, oDD){
18898      
18899         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18900             //Roo.log('not touch/ button !=0');
18901             return;
18902         }
18903         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18904             return; // double touch..
18905         }
18906         
18907
18908         if (this.isLocked()) {
18909             //Roo.log('locked');
18910             return;
18911         }
18912
18913         this.DDM.refreshCache(this.groups);
18914 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18915         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18916         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18917             //Roo.log('no outer handes or not over target');
18918                 // do nothing.
18919         } else {
18920 //            Roo.log('check validator');
18921             if (this.clickValidator(e)) {
18922 //                Roo.log('validate success');
18923                 // set the initial element position
18924                 this.setStartPosition();
18925
18926
18927                 this.b4MouseDown(e);
18928                 this.onMouseDown(e);
18929
18930                 this.DDM.handleMouseDown(e, this);
18931
18932                 this.DDM.stopEvent(e);
18933             } else {
18934
18935
18936             }
18937         }
18938     },
18939
18940     clickValidator: function(e) {
18941         var target = e.getTarget();
18942         return ( this.isValidHandleChild(target) &&
18943                     (this.id == this.handleElId ||
18944                         this.DDM.handleWasClicked(target, this.id)) );
18945     },
18946
18947     /**
18948      * Allows you to specify a tag name that should not start a drag operation
18949      * when clicked.  This is designed to facilitate embedding links within a
18950      * drag handle that do something other than start the drag.
18951      * @method addInvalidHandleType
18952      * @param {string} tagName the type of element to exclude
18953      */
18954     addInvalidHandleType: function(tagName) {
18955         var type = tagName.toUpperCase();
18956         this.invalidHandleTypes[type] = type;
18957     },
18958
18959     /**
18960      * Lets you to specify an element id for a child of a drag handle
18961      * that should not initiate a drag
18962      * @method addInvalidHandleId
18963      * @param {string} id the element id of the element you wish to ignore
18964      */
18965     addInvalidHandleId: function(id) {
18966         if (typeof id !== "string") {
18967             id = Roo.id(id);
18968         }
18969         this.invalidHandleIds[id] = id;
18970     },
18971
18972     /**
18973      * Lets you specify a css class of elements that will not initiate a drag
18974      * @method addInvalidHandleClass
18975      * @param {string} cssClass the class of the elements you wish to ignore
18976      */
18977     addInvalidHandleClass: function(cssClass) {
18978         this.invalidHandleClasses.push(cssClass);
18979     },
18980
18981     /**
18982      * Unsets an excluded tag name set by addInvalidHandleType
18983      * @method removeInvalidHandleType
18984      * @param {string} tagName the type of element to unexclude
18985      */
18986     removeInvalidHandleType: function(tagName) {
18987         var type = tagName.toUpperCase();
18988         // this.invalidHandleTypes[type] = null;
18989         delete this.invalidHandleTypes[type];
18990     },
18991
18992     /**
18993      * Unsets an invalid handle id
18994      * @method removeInvalidHandleId
18995      * @param {string} id the id of the element to re-enable
18996      */
18997     removeInvalidHandleId: function(id) {
18998         if (typeof id !== "string") {
18999             id = Roo.id(id);
19000         }
19001         delete this.invalidHandleIds[id];
19002     },
19003
19004     /**
19005      * Unsets an invalid css class
19006      * @method removeInvalidHandleClass
19007      * @param {string} cssClass the class of the element(s) you wish to
19008      * re-enable
19009      */
19010     removeInvalidHandleClass: function(cssClass) {
19011         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19012             if (this.invalidHandleClasses[i] == cssClass) {
19013                 delete this.invalidHandleClasses[i];
19014             }
19015         }
19016     },
19017
19018     /**
19019      * Checks the tag exclusion list to see if this click should be ignored
19020      * @method isValidHandleChild
19021      * @param {HTMLElement} node the HTMLElement to evaluate
19022      * @return {boolean} true if this is a valid tag type, false if not
19023      */
19024     isValidHandleChild: function(node) {
19025
19026         var valid = true;
19027         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19028         var nodeName;
19029         try {
19030             nodeName = node.nodeName.toUpperCase();
19031         } catch(e) {
19032             nodeName = node.nodeName;
19033         }
19034         valid = valid && !this.invalidHandleTypes[nodeName];
19035         valid = valid && !this.invalidHandleIds[node.id];
19036
19037         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19038             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19039         }
19040
19041
19042         return valid;
19043
19044     },
19045
19046     /**
19047      * Create the array of horizontal tick marks if an interval was specified
19048      * in setXConstraint().
19049      * @method setXTicks
19050      * @private
19051      */
19052     setXTicks: function(iStartX, iTickSize) {
19053         this.xTicks = [];
19054         this.xTickSize = iTickSize;
19055
19056         var tickMap = {};
19057
19058         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19059             if (!tickMap[i]) {
19060                 this.xTicks[this.xTicks.length] = i;
19061                 tickMap[i] = true;
19062             }
19063         }
19064
19065         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19066             if (!tickMap[i]) {
19067                 this.xTicks[this.xTicks.length] = i;
19068                 tickMap[i] = true;
19069             }
19070         }
19071
19072         this.xTicks.sort(this.DDM.numericSort) ;
19073     },
19074
19075     /**
19076      * Create the array of vertical tick marks if an interval was specified in
19077      * setYConstraint().
19078      * @method setYTicks
19079      * @private
19080      */
19081     setYTicks: function(iStartY, iTickSize) {
19082         this.yTicks = [];
19083         this.yTickSize = iTickSize;
19084
19085         var tickMap = {};
19086
19087         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19088             if (!tickMap[i]) {
19089                 this.yTicks[this.yTicks.length] = i;
19090                 tickMap[i] = true;
19091             }
19092         }
19093
19094         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19095             if (!tickMap[i]) {
19096                 this.yTicks[this.yTicks.length] = i;
19097                 tickMap[i] = true;
19098             }
19099         }
19100
19101         this.yTicks.sort(this.DDM.numericSort) ;
19102     },
19103
19104     /**
19105      * By default, the element can be dragged any place on the screen.  Use
19106      * this method to limit the horizontal travel of the element.  Pass in
19107      * 0,0 for the parameters if you want to lock the drag to the y axis.
19108      * @method setXConstraint
19109      * @param {int} iLeft the number of pixels the element can move to the left
19110      * @param {int} iRight the number of pixels the element can move to the
19111      * right
19112      * @param {int} iTickSize optional parameter for specifying that the
19113      * element
19114      * should move iTickSize pixels at a time.
19115      */
19116     setXConstraint: function(iLeft, iRight, iTickSize) {
19117         this.leftConstraint = iLeft;
19118         this.rightConstraint = iRight;
19119
19120         this.minX = this.initPageX - iLeft;
19121         this.maxX = this.initPageX + iRight;
19122         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19123
19124         this.constrainX = true;
19125     },
19126
19127     /**
19128      * Clears any constraints applied to this instance.  Also clears ticks
19129      * since they can't exist independent of a constraint at this time.
19130      * @method clearConstraints
19131      */
19132     clearConstraints: function() {
19133         this.constrainX = false;
19134         this.constrainY = false;
19135         this.clearTicks();
19136     },
19137
19138     /**
19139      * Clears any tick interval defined for this instance
19140      * @method clearTicks
19141      */
19142     clearTicks: function() {
19143         this.xTicks = null;
19144         this.yTicks = null;
19145         this.xTickSize = 0;
19146         this.yTickSize = 0;
19147     },
19148
19149     /**
19150      * By default, the element can be dragged any place on the screen.  Set
19151      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19152      * parameters if you want to lock the drag to the x axis.
19153      * @method setYConstraint
19154      * @param {int} iUp the number of pixels the element can move up
19155      * @param {int} iDown the number of pixels the element can move down
19156      * @param {int} iTickSize optional parameter for specifying that the
19157      * element should move iTickSize pixels at a time.
19158      */
19159     setYConstraint: function(iUp, iDown, iTickSize) {
19160         this.topConstraint = iUp;
19161         this.bottomConstraint = iDown;
19162
19163         this.minY = this.initPageY - iUp;
19164         this.maxY = this.initPageY + iDown;
19165         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19166
19167         this.constrainY = true;
19168
19169     },
19170
19171     /**
19172      * resetConstraints must be called if you manually reposition a dd element.
19173      * @method resetConstraints
19174      * @param {boolean} maintainOffset
19175      */
19176     resetConstraints: function() {
19177
19178
19179         // Maintain offsets if necessary
19180         if (this.initPageX || this.initPageX === 0) {
19181             // figure out how much this thing has moved
19182             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19183             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19184
19185             this.setInitPosition(dx, dy);
19186
19187         // This is the first time we have detected the element's position
19188         } else {
19189             this.setInitPosition();
19190         }
19191
19192         if (this.constrainX) {
19193             this.setXConstraint( this.leftConstraint,
19194                                  this.rightConstraint,
19195                                  this.xTickSize        );
19196         }
19197
19198         if (this.constrainY) {
19199             this.setYConstraint( this.topConstraint,
19200                                  this.bottomConstraint,
19201                                  this.yTickSize         );
19202         }
19203     },
19204
19205     /**
19206      * Normally the drag element is moved pixel by pixel, but we can specify
19207      * that it move a number of pixels at a time.  This method resolves the
19208      * location when we have it set up like this.
19209      * @method getTick
19210      * @param {int} val where we want to place the object
19211      * @param {int[]} tickArray sorted array of valid points
19212      * @return {int} the closest tick
19213      * @private
19214      */
19215     getTick: function(val, tickArray) {
19216
19217         if (!tickArray) {
19218             // If tick interval is not defined, it is effectively 1 pixel,
19219             // so we return the value passed to us.
19220             return val;
19221         } else if (tickArray[0] >= val) {
19222             // The value is lower than the first tick, so we return the first
19223             // tick.
19224             return tickArray[0];
19225         } else {
19226             for (var i=0, len=tickArray.length; i<len; ++i) {
19227                 var next = i + 1;
19228                 if (tickArray[next] && tickArray[next] >= val) {
19229                     var diff1 = val - tickArray[i];
19230                     var diff2 = tickArray[next] - val;
19231                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19232                 }
19233             }
19234
19235             // The value is larger than the last tick, so we return the last
19236             // tick.
19237             return tickArray[tickArray.length - 1];
19238         }
19239     },
19240
19241     /**
19242      * toString method
19243      * @method toString
19244      * @return {string} string representation of the dd obj
19245      */
19246     toString: function() {
19247         return ("DragDrop " + this.id);
19248     }
19249
19250 });
19251
19252 })();
19253 /*
19254  * Based on:
19255  * Ext JS Library 1.1.1
19256  * Copyright(c) 2006-2007, Ext JS, LLC.
19257  *
19258  * Originally Released Under LGPL - original licence link has changed is not relivant.
19259  *
19260  * Fork - LGPL
19261  * <script type="text/javascript">
19262  */
19263
19264
19265 /**
19266  * The drag and drop utility provides a framework for building drag and drop
19267  * applications.  In addition to enabling drag and drop for specific elements,
19268  * the drag and drop elements are tracked by the manager class, and the
19269  * interactions between the various elements are tracked during the drag and
19270  * the implementing code is notified about these important moments.
19271  */
19272
19273 // Only load the library once.  Rewriting the manager class would orphan
19274 // existing drag and drop instances.
19275 if (!Roo.dd.DragDropMgr) {
19276
19277 /**
19278  * @class Roo.dd.DragDropMgr
19279  * DragDropMgr is a singleton that tracks the element interaction for
19280  * all DragDrop items in the window.  Generally, you will not call
19281  * this class directly, but it does have helper methods that could
19282  * be useful in your DragDrop implementations.
19283  * @singleton
19284  */
19285 Roo.dd.DragDropMgr = function() {
19286
19287     var Event = Roo.EventManager;
19288
19289     return {
19290
19291         /**
19292          * Two dimensional Array of registered DragDrop objects.  The first
19293          * dimension is the DragDrop item group, the second the DragDrop
19294          * object.
19295          * @property ids
19296          * @type {string: string}
19297          * @private
19298          * @static
19299          */
19300         ids: {},
19301
19302         /**
19303          * Array of element ids defined as drag handles.  Used to determine
19304          * if the element that generated the mousedown event is actually the
19305          * handle and not the html element itself.
19306          * @property handleIds
19307          * @type {string: string}
19308          * @private
19309          * @static
19310          */
19311         handleIds: {},
19312
19313         /**
19314          * the DragDrop object that is currently being dragged
19315          * @property dragCurrent
19316          * @type DragDrop
19317          * @private
19318          * @static
19319          **/
19320         dragCurrent: null,
19321
19322         /**
19323          * the DragDrop object(s) that are being hovered over
19324          * @property dragOvers
19325          * @type Array
19326          * @private
19327          * @static
19328          */
19329         dragOvers: {},
19330
19331         /**
19332          * the X distance between the cursor and the object being dragged
19333          * @property deltaX
19334          * @type int
19335          * @private
19336          * @static
19337          */
19338         deltaX: 0,
19339
19340         /**
19341          * the Y distance between the cursor and the object being dragged
19342          * @property deltaY
19343          * @type int
19344          * @private
19345          * @static
19346          */
19347         deltaY: 0,
19348
19349         /**
19350          * Flag to determine if we should prevent the default behavior of the
19351          * events we define. By default this is true, but this can be set to
19352          * false if you need the default behavior (not recommended)
19353          * @property preventDefault
19354          * @type boolean
19355          * @static
19356          */
19357         preventDefault: true,
19358
19359         /**
19360          * Flag to determine if we should stop the propagation of the events
19361          * we generate. This is true by default but you may want to set it to
19362          * false if the html element contains other features that require the
19363          * mouse click.
19364          * @property stopPropagation
19365          * @type boolean
19366          * @static
19367          */
19368         stopPropagation: true,
19369
19370         /**
19371          * Internal flag that is set to true when drag and drop has been
19372          * intialized
19373          * @property initialized
19374          * @private
19375          * @static
19376          */
19377         initalized: false,
19378
19379         /**
19380          * All drag and drop can be disabled.
19381          * @property locked
19382          * @private
19383          * @static
19384          */
19385         locked: false,
19386
19387         /**
19388          * Called the first time an element is registered.
19389          * @method init
19390          * @private
19391          * @static
19392          */
19393         init: function() {
19394             this.initialized = true;
19395         },
19396
19397         /**
19398          * In point mode, drag and drop interaction is defined by the
19399          * location of the cursor during the drag/drop
19400          * @property POINT
19401          * @type int
19402          * @static
19403          */
19404         POINT: 0,
19405
19406         /**
19407          * In intersect mode, drag and drop interactio nis defined by the
19408          * overlap of two or more drag and drop objects.
19409          * @property INTERSECT
19410          * @type int
19411          * @static
19412          */
19413         INTERSECT: 1,
19414
19415         /**
19416          * The current drag and drop mode.  Default: POINT
19417          * @property mode
19418          * @type int
19419          * @static
19420          */
19421         mode: 0,
19422
19423         /**
19424          * Runs method on all drag and drop objects
19425          * @method _execOnAll
19426          * @private
19427          * @static
19428          */
19429         _execOnAll: function(sMethod, args) {
19430             for (var i in this.ids) {
19431                 for (var j in this.ids[i]) {
19432                     var oDD = this.ids[i][j];
19433                     if (! this.isTypeOfDD(oDD)) {
19434                         continue;
19435                     }
19436                     oDD[sMethod].apply(oDD, args);
19437                 }
19438             }
19439         },
19440
19441         /**
19442          * Drag and drop initialization.  Sets up the global event handlers
19443          * @method _onLoad
19444          * @private
19445          * @static
19446          */
19447         _onLoad: function() {
19448
19449             this.init();
19450
19451             if (!Roo.isTouch) {
19452                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19453                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19454             }
19455             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19456             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19457             
19458             Event.on(window,   "unload",    this._onUnload, this, true);
19459             Event.on(window,   "resize",    this._onResize, this, true);
19460             // Event.on(window,   "mouseout",    this._test);
19461
19462         },
19463
19464         /**
19465          * Reset constraints on all drag and drop objs
19466          * @method _onResize
19467          * @private
19468          * @static
19469          */
19470         _onResize: function(e) {
19471             this._execOnAll("resetConstraints", []);
19472         },
19473
19474         /**
19475          * Lock all drag and drop functionality
19476          * @method lock
19477          * @static
19478          */
19479         lock: function() { this.locked = true; },
19480
19481         /**
19482          * Unlock all drag and drop functionality
19483          * @method unlock
19484          * @static
19485          */
19486         unlock: function() { this.locked = false; },
19487
19488         /**
19489          * Is drag and drop locked?
19490          * @method isLocked
19491          * @return {boolean} True if drag and drop is locked, false otherwise.
19492          * @static
19493          */
19494         isLocked: function() { return this.locked; },
19495
19496         /**
19497          * Location cache that is set for all drag drop objects when a drag is
19498          * initiated, cleared when the drag is finished.
19499          * @property locationCache
19500          * @private
19501          * @static
19502          */
19503         locationCache: {},
19504
19505         /**
19506          * Set useCache to false if you want to force object the lookup of each
19507          * drag and drop linked element constantly during a drag.
19508          * @property useCache
19509          * @type boolean
19510          * @static
19511          */
19512         useCache: true,
19513
19514         /**
19515          * The number of pixels that the mouse needs to move after the
19516          * mousedown before the drag is initiated.  Default=3;
19517          * @property clickPixelThresh
19518          * @type int
19519          * @static
19520          */
19521         clickPixelThresh: 3,
19522
19523         /**
19524          * The number of milliseconds after the mousedown event to initiate the
19525          * drag if we don't get a mouseup event. Default=1000
19526          * @property clickTimeThresh
19527          * @type int
19528          * @static
19529          */
19530         clickTimeThresh: 350,
19531
19532         /**
19533          * Flag that indicates that either the drag pixel threshold or the
19534          * mousdown time threshold has been met
19535          * @property dragThreshMet
19536          * @type boolean
19537          * @private
19538          * @static
19539          */
19540         dragThreshMet: false,
19541
19542         /**
19543          * Timeout used for the click time threshold
19544          * @property clickTimeout
19545          * @type Object
19546          * @private
19547          * @static
19548          */
19549         clickTimeout: null,
19550
19551         /**
19552          * The X position of the mousedown event stored for later use when a
19553          * drag threshold is met.
19554          * @property startX
19555          * @type int
19556          * @private
19557          * @static
19558          */
19559         startX: 0,
19560
19561         /**
19562          * The Y position of the mousedown event stored for later use when a
19563          * drag threshold is met.
19564          * @property startY
19565          * @type int
19566          * @private
19567          * @static
19568          */
19569         startY: 0,
19570
19571         /**
19572          * Each DragDrop instance must be registered with the DragDropMgr.
19573          * This is executed in DragDrop.init()
19574          * @method regDragDrop
19575          * @param {DragDrop} oDD the DragDrop object to register
19576          * @param {String} sGroup the name of the group this element belongs to
19577          * @static
19578          */
19579         regDragDrop: function(oDD, sGroup) {
19580             if (!this.initialized) { this.init(); }
19581
19582             if (!this.ids[sGroup]) {
19583                 this.ids[sGroup] = {};
19584             }
19585             this.ids[sGroup][oDD.id] = oDD;
19586         },
19587
19588         /**
19589          * Removes the supplied dd instance from the supplied group. Executed
19590          * by DragDrop.removeFromGroup, so don't call this function directly.
19591          * @method removeDDFromGroup
19592          * @private
19593          * @static
19594          */
19595         removeDDFromGroup: function(oDD, sGroup) {
19596             if (!this.ids[sGroup]) {
19597                 this.ids[sGroup] = {};
19598             }
19599
19600             var obj = this.ids[sGroup];
19601             if (obj && obj[oDD.id]) {
19602                 delete obj[oDD.id];
19603             }
19604         },
19605
19606         /**
19607          * Unregisters a drag and drop item.  This is executed in
19608          * DragDrop.unreg, use that method instead of calling this directly.
19609          * @method _remove
19610          * @private
19611          * @static
19612          */
19613         _remove: function(oDD) {
19614             for (var g in oDD.groups) {
19615                 if (g && this.ids[g][oDD.id]) {
19616                     delete this.ids[g][oDD.id];
19617                 }
19618             }
19619             delete this.handleIds[oDD.id];
19620         },
19621
19622         /**
19623          * Each DragDrop handle element must be registered.  This is done
19624          * automatically when executing DragDrop.setHandleElId()
19625          * @method regHandle
19626          * @param {String} sDDId the DragDrop id this element is a handle for
19627          * @param {String} sHandleId the id of the element that is the drag
19628          * handle
19629          * @static
19630          */
19631         regHandle: function(sDDId, sHandleId) {
19632             if (!this.handleIds[sDDId]) {
19633                 this.handleIds[sDDId] = {};
19634             }
19635             this.handleIds[sDDId][sHandleId] = sHandleId;
19636         },
19637
19638         /**
19639          * Utility function to determine if a given element has been
19640          * registered as a drag drop item.
19641          * @method isDragDrop
19642          * @param {String} id the element id to check
19643          * @return {boolean} true if this element is a DragDrop item,
19644          * false otherwise
19645          * @static
19646          */
19647         isDragDrop: function(id) {
19648             return ( this.getDDById(id) ) ? true : false;
19649         },
19650
19651         /**
19652          * Returns the drag and drop instances that are in all groups the
19653          * passed in instance belongs to.
19654          * @method getRelated
19655          * @param {DragDrop} p_oDD the obj to get related data for
19656          * @param {boolean} bTargetsOnly if true, only return targetable objs
19657          * @return {DragDrop[]} the related instances
19658          * @static
19659          */
19660         getRelated: function(p_oDD, bTargetsOnly) {
19661             var oDDs = [];
19662             for (var i in p_oDD.groups) {
19663                 for (j in this.ids[i]) {
19664                     var dd = this.ids[i][j];
19665                     if (! this.isTypeOfDD(dd)) {
19666                         continue;
19667                     }
19668                     if (!bTargetsOnly || dd.isTarget) {
19669                         oDDs[oDDs.length] = dd;
19670                     }
19671                 }
19672             }
19673
19674             return oDDs;
19675         },
19676
19677         /**
19678          * Returns true if the specified dd target is a legal target for
19679          * the specifice drag obj
19680          * @method isLegalTarget
19681          * @param {DragDrop} the drag obj
19682          * @param {DragDrop} the target
19683          * @return {boolean} true if the target is a legal target for the
19684          * dd obj
19685          * @static
19686          */
19687         isLegalTarget: function (oDD, oTargetDD) {
19688             var targets = this.getRelated(oDD, true);
19689             for (var i=0, len=targets.length;i<len;++i) {
19690                 if (targets[i].id == oTargetDD.id) {
19691                     return true;
19692                 }
19693             }
19694
19695             return false;
19696         },
19697
19698         /**
19699          * My goal is to be able to transparently determine if an object is
19700          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19701          * returns "object", oDD.constructor.toString() always returns
19702          * "DragDrop" and not the name of the subclass.  So for now it just
19703          * evaluates a well-known variable in DragDrop.
19704          * @method isTypeOfDD
19705          * @param {Object} the object to evaluate
19706          * @return {boolean} true if typeof oDD = DragDrop
19707          * @static
19708          */
19709         isTypeOfDD: function (oDD) {
19710             return (oDD && oDD.__ygDragDrop);
19711         },
19712
19713         /**
19714          * Utility function to determine if a given element has been
19715          * registered as a drag drop handle for the given Drag Drop object.
19716          * @method isHandle
19717          * @param {String} id the element id to check
19718          * @return {boolean} true if this element is a DragDrop handle, false
19719          * otherwise
19720          * @static
19721          */
19722         isHandle: function(sDDId, sHandleId) {
19723             return ( this.handleIds[sDDId] &&
19724                             this.handleIds[sDDId][sHandleId] );
19725         },
19726
19727         /**
19728          * Returns the DragDrop instance for a given id
19729          * @method getDDById
19730          * @param {String} id the id of the DragDrop object
19731          * @return {DragDrop} the drag drop object, null if it is not found
19732          * @static
19733          */
19734         getDDById: function(id) {
19735             for (var i in this.ids) {
19736                 if (this.ids[i][id]) {
19737                     return this.ids[i][id];
19738                 }
19739             }
19740             return null;
19741         },
19742
19743         /**
19744          * Fired after a registered DragDrop object gets the mousedown event.
19745          * Sets up the events required to track the object being dragged
19746          * @method handleMouseDown
19747          * @param {Event} e the event
19748          * @param oDD the DragDrop object being dragged
19749          * @private
19750          * @static
19751          */
19752         handleMouseDown: function(e, oDD) {
19753             if(Roo.QuickTips){
19754                 Roo.QuickTips.disable();
19755             }
19756             this.currentTarget = e.getTarget();
19757
19758             this.dragCurrent = oDD;
19759
19760             var el = oDD.getEl();
19761
19762             // track start position
19763             this.startX = e.getPageX();
19764             this.startY = e.getPageY();
19765
19766             this.deltaX = this.startX - el.offsetLeft;
19767             this.deltaY = this.startY - el.offsetTop;
19768
19769             this.dragThreshMet = false;
19770
19771             this.clickTimeout = setTimeout(
19772                     function() {
19773                         var DDM = Roo.dd.DDM;
19774                         DDM.startDrag(DDM.startX, DDM.startY);
19775                     },
19776                     this.clickTimeThresh );
19777         },
19778
19779         /**
19780          * Fired when either the drag pixel threshol or the mousedown hold
19781          * time threshold has been met.
19782          * @method startDrag
19783          * @param x {int} the X position of the original mousedown
19784          * @param y {int} the Y position of the original mousedown
19785          * @static
19786          */
19787         startDrag: function(x, y) {
19788             clearTimeout(this.clickTimeout);
19789             if (this.dragCurrent) {
19790                 this.dragCurrent.b4StartDrag(x, y);
19791                 this.dragCurrent.startDrag(x, y);
19792             }
19793             this.dragThreshMet = true;
19794         },
19795
19796         /**
19797          * Internal function to handle the mouseup event.  Will be invoked
19798          * from the context of the document.
19799          * @method handleMouseUp
19800          * @param {Event} e the event
19801          * @private
19802          * @static
19803          */
19804         handleMouseUp: function(e) {
19805
19806             if(Roo.QuickTips){
19807                 Roo.QuickTips.enable();
19808             }
19809             if (! this.dragCurrent) {
19810                 return;
19811             }
19812
19813             clearTimeout(this.clickTimeout);
19814
19815             if (this.dragThreshMet) {
19816                 this.fireEvents(e, true);
19817             } else {
19818             }
19819
19820             this.stopDrag(e);
19821
19822             this.stopEvent(e);
19823         },
19824
19825         /**
19826          * Utility to stop event propagation and event default, if these
19827          * features are turned on.
19828          * @method stopEvent
19829          * @param {Event} e the event as returned by this.getEvent()
19830          * @static
19831          */
19832         stopEvent: function(e){
19833             if(this.stopPropagation) {
19834                 e.stopPropagation();
19835             }
19836
19837             if (this.preventDefault) {
19838                 e.preventDefault();
19839             }
19840         },
19841
19842         /**
19843          * Internal function to clean up event handlers after the drag
19844          * operation is complete
19845          * @method stopDrag
19846          * @param {Event} e the event
19847          * @private
19848          * @static
19849          */
19850         stopDrag: function(e) {
19851             // Fire the drag end event for the item that was dragged
19852             if (this.dragCurrent) {
19853                 if (this.dragThreshMet) {
19854                     this.dragCurrent.b4EndDrag(e);
19855                     this.dragCurrent.endDrag(e);
19856                 }
19857
19858                 this.dragCurrent.onMouseUp(e);
19859             }
19860
19861             this.dragCurrent = null;
19862             this.dragOvers = {};
19863         },
19864
19865         /**
19866          * Internal function to handle the mousemove event.  Will be invoked
19867          * from the context of the html element.
19868          *
19869          * @TODO figure out what we can do about mouse events lost when the
19870          * user drags objects beyond the window boundary.  Currently we can
19871          * detect this in internet explorer by verifying that the mouse is
19872          * down during the mousemove event.  Firefox doesn't give us the
19873          * button state on the mousemove event.
19874          * @method handleMouseMove
19875          * @param {Event} e the event
19876          * @private
19877          * @static
19878          */
19879         handleMouseMove: function(e) {
19880             if (! this.dragCurrent) {
19881                 return true;
19882             }
19883
19884             // var button = e.which || e.button;
19885
19886             // check for IE mouseup outside of page boundary
19887             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19888                 this.stopEvent(e);
19889                 return this.handleMouseUp(e);
19890             }
19891
19892             if (!this.dragThreshMet) {
19893                 var diffX = Math.abs(this.startX - e.getPageX());
19894                 var diffY = Math.abs(this.startY - e.getPageY());
19895                 if (diffX > this.clickPixelThresh ||
19896                             diffY > this.clickPixelThresh) {
19897                     this.startDrag(this.startX, this.startY);
19898                 }
19899             }
19900
19901             if (this.dragThreshMet) {
19902                 this.dragCurrent.b4Drag(e);
19903                 this.dragCurrent.onDrag(e);
19904                 if(!this.dragCurrent.moveOnly){
19905                     this.fireEvents(e, false);
19906                 }
19907             }
19908
19909             this.stopEvent(e);
19910
19911             return true;
19912         },
19913
19914         /**
19915          * Iterates over all of the DragDrop elements to find ones we are
19916          * hovering over or dropping on
19917          * @method fireEvents
19918          * @param {Event} e the event
19919          * @param {boolean} isDrop is this a drop op or a mouseover op?
19920          * @private
19921          * @static
19922          */
19923         fireEvents: function(e, isDrop) {
19924             var dc = this.dragCurrent;
19925
19926             // If the user did the mouse up outside of the window, we could
19927             // get here even though we have ended the drag.
19928             if (!dc || dc.isLocked()) {
19929                 return;
19930             }
19931
19932             var pt = e.getPoint();
19933
19934             // cache the previous dragOver array
19935             var oldOvers = [];
19936
19937             var outEvts   = [];
19938             var overEvts  = [];
19939             var dropEvts  = [];
19940             var enterEvts = [];
19941
19942             // Check to see if the object(s) we were hovering over is no longer
19943             // being hovered over so we can fire the onDragOut event
19944             for (var i in this.dragOvers) {
19945
19946                 var ddo = this.dragOvers[i];
19947
19948                 if (! this.isTypeOfDD(ddo)) {
19949                     continue;
19950                 }
19951
19952                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19953                     outEvts.push( ddo );
19954                 }
19955
19956                 oldOvers[i] = true;
19957                 delete this.dragOvers[i];
19958             }
19959
19960             for (var sGroup in dc.groups) {
19961
19962                 if ("string" != typeof sGroup) {
19963                     continue;
19964                 }
19965
19966                 for (i in this.ids[sGroup]) {
19967                     var oDD = this.ids[sGroup][i];
19968                     if (! this.isTypeOfDD(oDD)) {
19969                         continue;
19970                     }
19971
19972                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
19973                         if (this.isOverTarget(pt, oDD, this.mode)) {
19974                             // look for drop interactions
19975                             if (isDrop) {
19976                                 dropEvts.push( oDD );
19977                             // look for drag enter and drag over interactions
19978                             } else {
19979
19980                                 // initial drag over: dragEnter fires
19981                                 if (!oldOvers[oDD.id]) {
19982                                     enterEvts.push( oDD );
19983                                 // subsequent drag overs: dragOver fires
19984                                 } else {
19985                                     overEvts.push( oDD );
19986                                 }
19987
19988                                 this.dragOvers[oDD.id] = oDD;
19989                             }
19990                         }
19991                     }
19992                 }
19993             }
19994
19995             if (this.mode) {
19996                 if (outEvts.length) {
19997                     dc.b4DragOut(e, outEvts);
19998                     dc.onDragOut(e, outEvts);
19999                 }
20000
20001                 if (enterEvts.length) {
20002                     dc.onDragEnter(e, enterEvts);
20003                 }
20004
20005                 if (overEvts.length) {
20006                     dc.b4DragOver(e, overEvts);
20007                     dc.onDragOver(e, overEvts);
20008                 }
20009
20010                 if (dropEvts.length) {
20011                     dc.b4DragDrop(e, dropEvts);
20012                     dc.onDragDrop(e, dropEvts);
20013                 }
20014
20015             } else {
20016                 // fire dragout events
20017                 var len = 0;
20018                 for (i=0, len=outEvts.length; i<len; ++i) {
20019                     dc.b4DragOut(e, outEvts[i].id);
20020                     dc.onDragOut(e, outEvts[i].id);
20021                 }
20022
20023                 // fire enter events
20024                 for (i=0,len=enterEvts.length; i<len; ++i) {
20025                     // dc.b4DragEnter(e, oDD.id);
20026                     dc.onDragEnter(e, enterEvts[i].id);
20027                 }
20028
20029                 // fire over events
20030                 for (i=0,len=overEvts.length; i<len; ++i) {
20031                     dc.b4DragOver(e, overEvts[i].id);
20032                     dc.onDragOver(e, overEvts[i].id);
20033                 }
20034
20035                 // fire drop events
20036                 for (i=0, len=dropEvts.length; i<len; ++i) {
20037                     dc.b4DragDrop(e, dropEvts[i].id);
20038                     dc.onDragDrop(e, dropEvts[i].id);
20039                 }
20040
20041             }
20042
20043             // notify about a drop that did not find a target
20044             if (isDrop && !dropEvts.length) {
20045                 dc.onInvalidDrop(e);
20046             }
20047
20048         },
20049
20050         /**
20051          * Helper function for getting the best match from the list of drag
20052          * and drop objects returned by the drag and drop events when we are
20053          * in INTERSECT mode.  It returns either the first object that the
20054          * cursor is over, or the object that has the greatest overlap with
20055          * the dragged element.
20056          * @method getBestMatch
20057          * @param  {DragDrop[]} dds The array of drag and drop objects
20058          * targeted
20059          * @return {DragDrop}       The best single match
20060          * @static
20061          */
20062         getBestMatch: function(dds) {
20063             var winner = null;
20064             // Return null if the input is not what we expect
20065             //if (!dds || !dds.length || dds.length == 0) {
20066                // winner = null;
20067             // If there is only one item, it wins
20068             //} else if (dds.length == 1) {
20069
20070             var len = dds.length;
20071
20072             if (len == 1) {
20073                 winner = dds[0];
20074             } else {
20075                 // Loop through the targeted items
20076                 for (var i=0; i<len; ++i) {
20077                     var dd = dds[i];
20078                     // If the cursor is over the object, it wins.  If the
20079                     // cursor is over multiple matches, the first one we come
20080                     // to wins.
20081                     if (dd.cursorIsOver) {
20082                         winner = dd;
20083                         break;
20084                     // Otherwise the object with the most overlap wins
20085                     } else {
20086                         if (!winner ||
20087                             winner.overlap.getArea() < dd.overlap.getArea()) {
20088                             winner = dd;
20089                         }
20090                     }
20091                 }
20092             }
20093
20094             return winner;
20095         },
20096
20097         /**
20098          * Refreshes the cache of the top-left and bottom-right points of the
20099          * drag and drop objects in the specified group(s).  This is in the
20100          * format that is stored in the drag and drop instance, so typical
20101          * usage is:
20102          * <code>
20103          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20104          * </code>
20105          * Alternatively:
20106          * <code>
20107          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20108          * </code>
20109          * @TODO this really should be an indexed array.  Alternatively this
20110          * method could accept both.
20111          * @method refreshCache
20112          * @param {Object} groups an associative array of groups to refresh
20113          * @static
20114          */
20115         refreshCache: function(groups) {
20116             for (var sGroup in groups) {
20117                 if ("string" != typeof sGroup) {
20118                     continue;
20119                 }
20120                 for (var i in this.ids[sGroup]) {
20121                     var oDD = this.ids[sGroup][i];
20122
20123                     if (this.isTypeOfDD(oDD)) {
20124                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20125                         var loc = this.getLocation(oDD);
20126                         if (loc) {
20127                             this.locationCache[oDD.id] = loc;
20128                         } else {
20129                             delete this.locationCache[oDD.id];
20130                             // this will unregister the drag and drop object if
20131                             // the element is not in a usable state
20132                             // oDD.unreg();
20133                         }
20134                     }
20135                 }
20136             }
20137         },
20138
20139         /**
20140          * This checks to make sure an element exists and is in the DOM.  The
20141          * main purpose is to handle cases where innerHTML is used to remove
20142          * drag and drop objects from the DOM.  IE provides an 'unspecified
20143          * error' when trying to access the offsetParent of such an element
20144          * @method verifyEl
20145          * @param {HTMLElement} el the element to check
20146          * @return {boolean} true if the element looks usable
20147          * @static
20148          */
20149         verifyEl: function(el) {
20150             if (el) {
20151                 var parent;
20152                 if(Roo.isIE){
20153                     try{
20154                         parent = el.offsetParent;
20155                     }catch(e){}
20156                 }else{
20157                     parent = el.offsetParent;
20158                 }
20159                 if (parent) {
20160                     return true;
20161                 }
20162             }
20163
20164             return false;
20165         },
20166
20167         /**
20168          * Returns a Region object containing the drag and drop element's position
20169          * and size, including the padding configured for it
20170          * @method getLocation
20171          * @param {DragDrop} oDD the drag and drop object to get the
20172          *                       location for
20173          * @return {Roo.lib.Region} a Region object representing the total area
20174          *                             the element occupies, including any padding
20175          *                             the instance is configured for.
20176          * @static
20177          */
20178         getLocation: function(oDD) {
20179             if (! this.isTypeOfDD(oDD)) {
20180                 return null;
20181             }
20182
20183             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20184
20185             try {
20186                 pos= Roo.lib.Dom.getXY(el);
20187             } catch (e) { }
20188
20189             if (!pos) {
20190                 return null;
20191             }
20192
20193             x1 = pos[0];
20194             x2 = x1 + el.offsetWidth;
20195             y1 = pos[1];
20196             y2 = y1 + el.offsetHeight;
20197
20198             t = y1 - oDD.padding[0];
20199             r = x2 + oDD.padding[1];
20200             b = y2 + oDD.padding[2];
20201             l = x1 - oDD.padding[3];
20202
20203             return new Roo.lib.Region( t, r, b, l );
20204         },
20205
20206         /**
20207          * Checks the cursor location to see if it over the target
20208          * @method isOverTarget
20209          * @param {Roo.lib.Point} pt The point to evaluate
20210          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20211          * @return {boolean} true if the mouse is over the target
20212          * @private
20213          * @static
20214          */
20215         isOverTarget: function(pt, oTarget, intersect) {
20216             // use cache if available
20217             var loc = this.locationCache[oTarget.id];
20218             if (!loc || !this.useCache) {
20219                 loc = this.getLocation(oTarget);
20220                 this.locationCache[oTarget.id] = loc;
20221
20222             }
20223
20224             if (!loc) {
20225                 return false;
20226             }
20227
20228             oTarget.cursorIsOver = loc.contains( pt );
20229
20230             // DragDrop is using this as a sanity check for the initial mousedown
20231             // in this case we are done.  In POINT mode, if the drag obj has no
20232             // contraints, we are also done. Otherwise we need to evaluate the
20233             // location of the target as related to the actual location of the
20234             // dragged element.
20235             var dc = this.dragCurrent;
20236             if (!dc || !dc.getTargetCoord ||
20237                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20238                 return oTarget.cursorIsOver;
20239             }
20240
20241             oTarget.overlap = null;
20242
20243             // Get the current location of the drag element, this is the
20244             // location of the mouse event less the delta that represents
20245             // where the original mousedown happened on the element.  We
20246             // need to consider constraints and ticks as well.
20247             var pos = dc.getTargetCoord(pt.x, pt.y);
20248
20249             var el = dc.getDragEl();
20250             var curRegion = new Roo.lib.Region( pos.y,
20251                                                    pos.x + el.offsetWidth,
20252                                                    pos.y + el.offsetHeight,
20253                                                    pos.x );
20254
20255             var overlap = curRegion.intersect(loc);
20256
20257             if (overlap) {
20258                 oTarget.overlap = overlap;
20259                 return (intersect) ? true : oTarget.cursorIsOver;
20260             } else {
20261                 return false;
20262             }
20263         },
20264
20265         /**
20266          * unload event handler
20267          * @method _onUnload
20268          * @private
20269          * @static
20270          */
20271         _onUnload: function(e, me) {
20272             Roo.dd.DragDropMgr.unregAll();
20273         },
20274
20275         /**
20276          * Cleans up the drag and drop events and objects.
20277          * @method unregAll
20278          * @private
20279          * @static
20280          */
20281         unregAll: function() {
20282
20283             if (this.dragCurrent) {
20284                 this.stopDrag();
20285                 this.dragCurrent = null;
20286             }
20287
20288             this._execOnAll("unreg", []);
20289
20290             for (i in this.elementCache) {
20291                 delete this.elementCache[i];
20292             }
20293
20294             this.elementCache = {};
20295             this.ids = {};
20296         },
20297
20298         /**
20299          * A cache of DOM elements
20300          * @property elementCache
20301          * @private
20302          * @static
20303          */
20304         elementCache: {},
20305
20306         /**
20307          * Get the wrapper for the DOM element specified
20308          * @method getElWrapper
20309          * @param {String} id the id of the element to get
20310          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20311          * @private
20312          * @deprecated This wrapper isn't that useful
20313          * @static
20314          */
20315         getElWrapper: function(id) {
20316             var oWrapper = this.elementCache[id];
20317             if (!oWrapper || !oWrapper.el) {
20318                 oWrapper = this.elementCache[id] =
20319                     new this.ElementWrapper(Roo.getDom(id));
20320             }
20321             return oWrapper;
20322         },
20323
20324         /**
20325          * Returns the actual DOM element
20326          * @method getElement
20327          * @param {String} id the id of the elment to get
20328          * @return {Object} The element
20329          * @deprecated use Roo.getDom instead
20330          * @static
20331          */
20332         getElement: function(id) {
20333             return Roo.getDom(id);
20334         },
20335
20336         /**
20337          * Returns the style property for the DOM element (i.e.,
20338          * document.getElById(id).style)
20339          * @method getCss
20340          * @param {String} id the id of the elment to get
20341          * @return {Object} The style property of the element
20342          * @deprecated use Roo.getDom instead
20343          * @static
20344          */
20345         getCss: function(id) {
20346             var el = Roo.getDom(id);
20347             return (el) ? el.style : null;
20348         },
20349
20350         /**
20351          * Inner class for cached elements
20352          * @class DragDropMgr.ElementWrapper
20353          * @for DragDropMgr
20354          * @private
20355          * @deprecated
20356          */
20357         ElementWrapper: function(el) {
20358                 /**
20359                  * The element
20360                  * @property el
20361                  */
20362                 this.el = el || null;
20363                 /**
20364                  * The element id
20365                  * @property id
20366                  */
20367                 this.id = this.el && el.id;
20368                 /**
20369                  * A reference to the style property
20370                  * @property css
20371                  */
20372                 this.css = this.el && el.style;
20373             },
20374
20375         /**
20376          * Returns the X position of an html element
20377          * @method getPosX
20378          * @param el the element for which to get the position
20379          * @return {int} the X coordinate
20380          * @for DragDropMgr
20381          * @deprecated use Roo.lib.Dom.getX instead
20382          * @static
20383          */
20384         getPosX: function(el) {
20385             return Roo.lib.Dom.getX(el);
20386         },
20387
20388         /**
20389          * Returns the Y position of an html element
20390          * @method getPosY
20391          * @param el the element for which to get the position
20392          * @return {int} the Y coordinate
20393          * @deprecated use Roo.lib.Dom.getY instead
20394          * @static
20395          */
20396         getPosY: function(el) {
20397             return Roo.lib.Dom.getY(el);
20398         },
20399
20400         /**
20401          * Swap two nodes.  In IE, we use the native method, for others we
20402          * emulate the IE behavior
20403          * @method swapNode
20404          * @param n1 the first node to swap
20405          * @param n2 the other node to swap
20406          * @static
20407          */
20408         swapNode: function(n1, n2) {
20409             if (n1.swapNode) {
20410                 n1.swapNode(n2);
20411             } else {
20412                 var p = n2.parentNode;
20413                 var s = n2.nextSibling;
20414
20415                 if (s == n1) {
20416                     p.insertBefore(n1, n2);
20417                 } else if (n2 == n1.nextSibling) {
20418                     p.insertBefore(n2, n1);
20419                 } else {
20420                     n1.parentNode.replaceChild(n2, n1);
20421                     p.insertBefore(n1, s);
20422                 }
20423             }
20424         },
20425
20426         /**
20427          * Returns the current scroll position
20428          * @method getScroll
20429          * @private
20430          * @static
20431          */
20432         getScroll: function () {
20433             var t, l, dde=document.documentElement, db=document.body;
20434             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20435                 t = dde.scrollTop;
20436                 l = dde.scrollLeft;
20437             } else if (db) {
20438                 t = db.scrollTop;
20439                 l = db.scrollLeft;
20440             } else {
20441
20442             }
20443             return { top: t, left: l };
20444         },
20445
20446         /**
20447          * Returns the specified element style property
20448          * @method getStyle
20449          * @param {HTMLElement} el          the element
20450          * @param {string}      styleProp   the style property
20451          * @return {string} The value of the style property
20452          * @deprecated use Roo.lib.Dom.getStyle
20453          * @static
20454          */
20455         getStyle: function(el, styleProp) {
20456             return Roo.fly(el).getStyle(styleProp);
20457         },
20458
20459         /**
20460          * Gets the scrollTop
20461          * @method getScrollTop
20462          * @return {int} the document's scrollTop
20463          * @static
20464          */
20465         getScrollTop: function () { return this.getScroll().top; },
20466
20467         /**
20468          * Gets the scrollLeft
20469          * @method getScrollLeft
20470          * @return {int} the document's scrollTop
20471          * @static
20472          */
20473         getScrollLeft: function () { return this.getScroll().left; },
20474
20475         /**
20476          * Sets the x/y position of an element to the location of the
20477          * target element.
20478          * @method moveToEl
20479          * @param {HTMLElement} moveEl      The element to move
20480          * @param {HTMLElement} targetEl    The position reference element
20481          * @static
20482          */
20483         moveToEl: function (moveEl, targetEl) {
20484             var aCoord = Roo.lib.Dom.getXY(targetEl);
20485             Roo.lib.Dom.setXY(moveEl, aCoord);
20486         },
20487
20488         /**
20489          * Numeric array sort function
20490          * @method numericSort
20491          * @static
20492          */
20493         numericSort: function(a, b) { return (a - b); },
20494
20495         /**
20496          * Internal counter
20497          * @property _timeoutCount
20498          * @private
20499          * @static
20500          */
20501         _timeoutCount: 0,
20502
20503         /**
20504          * Trying to make the load order less important.  Without this we get
20505          * an error if this file is loaded before the Event Utility.
20506          * @method _addListeners
20507          * @private
20508          * @static
20509          */
20510         _addListeners: function() {
20511             var DDM = Roo.dd.DDM;
20512             if ( Roo.lib.Event && document ) {
20513                 DDM._onLoad();
20514             } else {
20515                 if (DDM._timeoutCount > 2000) {
20516                 } else {
20517                     setTimeout(DDM._addListeners, 10);
20518                     if (document && document.body) {
20519                         DDM._timeoutCount += 1;
20520                     }
20521                 }
20522             }
20523         },
20524
20525         /**
20526          * Recursively searches the immediate parent and all child nodes for
20527          * the handle element in order to determine wheter or not it was
20528          * clicked.
20529          * @method handleWasClicked
20530          * @param node the html element to inspect
20531          * @static
20532          */
20533         handleWasClicked: function(node, id) {
20534             if (this.isHandle(id, node.id)) {
20535                 return true;
20536             } else {
20537                 // check to see if this is a text node child of the one we want
20538                 var p = node.parentNode;
20539
20540                 while (p) {
20541                     if (this.isHandle(id, p.id)) {
20542                         return true;
20543                     } else {
20544                         p = p.parentNode;
20545                     }
20546                 }
20547             }
20548
20549             return false;
20550         }
20551
20552     };
20553
20554 }();
20555
20556 // shorter alias, save a few bytes
20557 Roo.dd.DDM = Roo.dd.DragDropMgr;
20558 Roo.dd.DDM._addListeners();
20559
20560 }/*
20561  * Based on:
20562  * Ext JS Library 1.1.1
20563  * Copyright(c) 2006-2007, Ext JS, LLC.
20564  *
20565  * Originally Released Under LGPL - original licence link has changed is not relivant.
20566  *
20567  * Fork - LGPL
20568  * <script type="text/javascript">
20569  */
20570
20571 /**
20572  * @class Roo.dd.DD
20573  * A DragDrop implementation where the linked element follows the
20574  * mouse cursor during a drag.
20575  * @extends Roo.dd.DragDrop
20576  * @constructor
20577  * @param {String} id the id of the linked element
20578  * @param {String} sGroup the group of related DragDrop items
20579  * @param {object} config an object containing configurable attributes
20580  *                Valid properties for DD:
20581  *                    scroll
20582  */
20583 Roo.dd.DD = function(id, sGroup, config) {
20584     if (id) {
20585         this.init(id, sGroup, config);
20586     }
20587 };
20588
20589 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20590
20591     /**
20592      * When set to true, the utility automatically tries to scroll the browser
20593      * window wehn a drag and drop element is dragged near the viewport boundary.
20594      * Defaults to true.
20595      * @property scroll
20596      * @type boolean
20597      */
20598     scroll: true,
20599
20600     /**
20601      * Sets the pointer offset to the distance between the linked element's top
20602      * left corner and the location the element was clicked
20603      * @method autoOffset
20604      * @param {int} iPageX the X coordinate of the click
20605      * @param {int} iPageY the Y coordinate of the click
20606      */
20607     autoOffset: function(iPageX, iPageY) {
20608         var x = iPageX - this.startPageX;
20609         var y = iPageY - this.startPageY;
20610         this.setDelta(x, y);
20611     },
20612
20613     /**
20614      * Sets the pointer offset.  You can call this directly to force the
20615      * offset to be in a particular location (e.g., pass in 0,0 to set it
20616      * to the center of the object)
20617      * @method setDelta
20618      * @param {int} iDeltaX the distance from the left
20619      * @param {int} iDeltaY the distance from the top
20620      */
20621     setDelta: function(iDeltaX, iDeltaY) {
20622         this.deltaX = iDeltaX;
20623         this.deltaY = iDeltaY;
20624     },
20625
20626     /**
20627      * Sets the drag element to the location of the mousedown or click event,
20628      * maintaining the cursor location relative to the location on the element
20629      * that was clicked.  Override this if you want to place the element in a
20630      * location other than where the cursor is.
20631      * @method setDragElPos
20632      * @param {int} iPageX the X coordinate of the mousedown or drag event
20633      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20634      */
20635     setDragElPos: function(iPageX, iPageY) {
20636         // the first time we do this, we are going to check to make sure
20637         // the element has css positioning
20638
20639         var el = this.getDragEl();
20640         this.alignElWithMouse(el, iPageX, iPageY);
20641     },
20642
20643     /**
20644      * Sets the element to the location of the mousedown or click event,
20645      * maintaining the cursor location relative to the location on the element
20646      * that was clicked.  Override this if you want to place the element in a
20647      * location other than where the cursor is.
20648      * @method alignElWithMouse
20649      * @param {HTMLElement} el the element to move
20650      * @param {int} iPageX the X coordinate of the mousedown or drag event
20651      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20652      */
20653     alignElWithMouse: function(el, iPageX, iPageY) {
20654         var oCoord = this.getTargetCoord(iPageX, iPageY);
20655         var fly = el.dom ? el : Roo.fly(el);
20656         if (!this.deltaSetXY) {
20657             var aCoord = [oCoord.x, oCoord.y];
20658             fly.setXY(aCoord);
20659             var newLeft = fly.getLeft(true);
20660             var newTop  = fly.getTop(true);
20661             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20662         } else {
20663             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20664         }
20665
20666         this.cachePosition(oCoord.x, oCoord.y);
20667         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20668         return oCoord;
20669     },
20670
20671     /**
20672      * Saves the most recent position so that we can reset the constraints and
20673      * tick marks on-demand.  We need to know this so that we can calculate the
20674      * number of pixels the element is offset from its original position.
20675      * @method cachePosition
20676      * @param iPageX the current x position (optional, this just makes it so we
20677      * don't have to look it up again)
20678      * @param iPageY the current y position (optional, this just makes it so we
20679      * don't have to look it up again)
20680      */
20681     cachePosition: function(iPageX, iPageY) {
20682         if (iPageX) {
20683             this.lastPageX = iPageX;
20684             this.lastPageY = iPageY;
20685         } else {
20686             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20687             this.lastPageX = aCoord[0];
20688             this.lastPageY = aCoord[1];
20689         }
20690     },
20691
20692     /**
20693      * Auto-scroll the window if the dragged object has been moved beyond the
20694      * visible window boundary.
20695      * @method autoScroll
20696      * @param {int} x the drag element's x position
20697      * @param {int} y the drag element's y position
20698      * @param {int} h the height of the drag element
20699      * @param {int} w the width of the drag element
20700      * @private
20701      */
20702     autoScroll: function(x, y, h, w) {
20703
20704         if (this.scroll) {
20705             // The client height
20706             var clientH = Roo.lib.Dom.getViewWidth();
20707
20708             // The client width
20709             var clientW = Roo.lib.Dom.getViewHeight();
20710
20711             // The amt scrolled down
20712             var st = this.DDM.getScrollTop();
20713
20714             // The amt scrolled right
20715             var sl = this.DDM.getScrollLeft();
20716
20717             // Location of the bottom of the element
20718             var bot = h + y;
20719
20720             // Location of the right of the element
20721             var right = w + x;
20722
20723             // The distance from the cursor to the bottom of the visible area,
20724             // adjusted so that we don't scroll if the cursor is beyond the
20725             // element drag constraints
20726             var toBot = (clientH + st - y - this.deltaY);
20727
20728             // The distance from the cursor to the right of the visible area
20729             var toRight = (clientW + sl - x - this.deltaX);
20730
20731
20732             // How close to the edge the cursor must be before we scroll
20733             // var thresh = (document.all) ? 100 : 40;
20734             var thresh = 40;
20735
20736             // How many pixels to scroll per autoscroll op.  This helps to reduce
20737             // clunky scrolling. IE is more sensitive about this ... it needs this
20738             // value to be higher.
20739             var scrAmt = (document.all) ? 80 : 30;
20740
20741             // Scroll down if we are near the bottom of the visible page and the
20742             // obj extends below the crease
20743             if ( bot > clientH && toBot < thresh ) {
20744                 window.scrollTo(sl, st + scrAmt);
20745             }
20746
20747             // Scroll up if the window is scrolled down and the top of the object
20748             // goes above the top border
20749             if ( y < st && st > 0 && y - st < thresh ) {
20750                 window.scrollTo(sl, st - scrAmt);
20751             }
20752
20753             // Scroll right if the obj is beyond the right border and the cursor is
20754             // near the border.
20755             if ( right > clientW && toRight < thresh ) {
20756                 window.scrollTo(sl + scrAmt, st);
20757             }
20758
20759             // Scroll left if the window has been scrolled to the right and the obj
20760             // extends past the left border
20761             if ( x < sl && sl > 0 && x - sl < thresh ) {
20762                 window.scrollTo(sl - scrAmt, st);
20763             }
20764         }
20765     },
20766
20767     /**
20768      * Finds the location the element should be placed if we want to move
20769      * it to where the mouse location less the click offset would place us.
20770      * @method getTargetCoord
20771      * @param {int} iPageX the X coordinate of the click
20772      * @param {int} iPageY the Y coordinate of the click
20773      * @return an object that contains the coordinates (Object.x and Object.y)
20774      * @private
20775      */
20776     getTargetCoord: function(iPageX, iPageY) {
20777
20778
20779         var x = iPageX - this.deltaX;
20780         var y = iPageY - this.deltaY;
20781
20782         if (this.constrainX) {
20783             if (x < this.minX) { x = this.minX; }
20784             if (x > this.maxX) { x = this.maxX; }
20785         }
20786
20787         if (this.constrainY) {
20788             if (y < this.minY) { y = this.minY; }
20789             if (y > this.maxY) { y = this.maxY; }
20790         }
20791
20792         x = this.getTick(x, this.xTicks);
20793         y = this.getTick(y, this.yTicks);
20794
20795
20796         return {x:x, y:y};
20797     },
20798
20799     /*
20800      * Sets up config options specific to this class. Overrides
20801      * Roo.dd.DragDrop, but all versions of this method through the
20802      * inheritance chain are called
20803      */
20804     applyConfig: function() {
20805         Roo.dd.DD.superclass.applyConfig.call(this);
20806         this.scroll = (this.config.scroll !== false);
20807     },
20808
20809     /*
20810      * Event that fires prior to the onMouseDown event.  Overrides
20811      * Roo.dd.DragDrop.
20812      */
20813     b4MouseDown: function(e) {
20814         // this.resetConstraints();
20815         this.autoOffset(e.getPageX(),
20816                             e.getPageY());
20817     },
20818
20819     /*
20820      * Event that fires prior to the onDrag event.  Overrides
20821      * Roo.dd.DragDrop.
20822      */
20823     b4Drag: function(e) {
20824         this.setDragElPos(e.getPageX(),
20825                             e.getPageY());
20826     },
20827
20828     toString: function() {
20829         return ("DD " + this.id);
20830     }
20831
20832     //////////////////////////////////////////////////////////////////////////
20833     // Debugging ygDragDrop events that can be overridden
20834     //////////////////////////////////////////////////////////////////////////
20835     /*
20836     startDrag: function(x, y) {
20837     },
20838
20839     onDrag: function(e) {
20840     },
20841
20842     onDragEnter: function(e, id) {
20843     },
20844
20845     onDragOver: function(e, id) {
20846     },
20847
20848     onDragOut: function(e, id) {
20849     },
20850
20851     onDragDrop: function(e, id) {
20852     },
20853
20854     endDrag: function(e) {
20855     }
20856
20857     */
20858
20859 });/*
20860  * Based on:
20861  * Ext JS Library 1.1.1
20862  * Copyright(c) 2006-2007, Ext JS, LLC.
20863  *
20864  * Originally Released Under LGPL - original licence link has changed is not relivant.
20865  *
20866  * Fork - LGPL
20867  * <script type="text/javascript">
20868  */
20869
20870 /**
20871  * @class Roo.dd.DDProxy
20872  * A DragDrop implementation that inserts an empty, bordered div into
20873  * the document that follows the cursor during drag operations.  At the time of
20874  * the click, the frame div is resized to the dimensions of the linked html
20875  * element, and moved to the exact location of the linked element.
20876  *
20877  * References to the "frame" element refer to the single proxy element that
20878  * was created to be dragged in place of all DDProxy elements on the
20879  * page.
20880  *
20881  * @extends Roo.dd.DD
20882  * @constructor
20883  * @param {String} id the id of the linked html element
20884  * @param {String} sGroup the group of related DragDrop objects
20885  * @param {object} config an object containing configurable attributes
20886  *                Valid properties for DDProxy in addition to those in DragDrop:
20887  *                   resizeFrame, centerFrame, dragElId
20888  */
20889 Roo.dd.DDProxy = function(id, sGroup, config) {
20890     if (id) {
20891         this.init(id, sGroup, config);
20892         this.initFrame();
20893     }
20894 };
20895
20896 /**
20897  * The default drag frame div id
20898  * @property Roo.dd.DDProxy.dragElId
20899  * @type String
20900  * @static
20901  */
20902 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20903
20904 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20905
20906     /**
20907      * By default we resize the drag frame to be the same size as the element
20908      * we want to drag (this is to get the frame effect).  We can turn it off
20909      * if we want a different behavior.
20910      * @property resizeFrame
20911      * @type boolean
20912      */
20913     resizeFrame: true,
20914
20915     /**
20916      * By default the frame is positioned exactly where the drag element is, so
20917      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20918      * you do not have constraints on the obj is to have the drag frame centered
20919      * around the cursor.  Set centerFrame to true for this effect.
20920      * @property centerFrame
20921      * @type boolean
20922      */
20923     centerFrame: false,
20924
20925     /**
20926      * Creates the proxy element if it does not yet exist
20927      * @method createFrame
20928      */
20929     createFrame: function() {
20930         var self = this;
20931         var body = document.body;
20932
20933         if (!body || !body.firstChild) {
20934             setTimeout( function() { self.createFrame(); }, 50 );
20935             return;
20936         }
20937
20938         var div = this.getDragEl();
20939
20940         if (!div) {
20941             div    = document.createElement("div");
20942             div.id = this.dragElId;
20943             var s  = div.style;
20944
20945             s.position   = "absolute";
20946             s.visibility = "hidden";
20947             s.cursor     = "move";
20948             s.border     = "2px solid #aaa";
20949             s.zIndex     = 999;
20950
20951             // appendChild can blow up IE if invoked prior to the window load event
20952             // while rendering a table.  It is possible there are other scenarios
20953             // that would cause this to happen as well.
20954             body.insertBefore(div, body.firstChild);
20955         }
20956     },
20957
20958     /**
20959      * Initialization for the drag frame element.  Must be called in the
20960      * constructor of all subclasses
20961      * @method initFrame
20962      */
20963     initFrame: function() {
20964         this.createFrame();
20965     },
20966
20967     applyConfig: function() {
20968         Roo.dd.DDProxy.superclass.applyConfig.call(this);
20969
20970         this.resizeFrame = (this.config.resizeFrame !== false);
20971         this.centerFrame = (this.config.centerFrame);
20972         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
20973     },
20974
20975     /**
20976      * Resizes the drag frame to the dimensions of the clicked object, positions
20977      * it over the object, and finally displays it
20978      * @method showFrame
20979      * @param {int} iPageX X click position
20980      * @param {int} iPageY Y click position
20981      * @private
20982      */
20983     showFrame: function(iPageX, iPageY) {
20984         var el = this.getEl();
20985         var dragEl = this.getDragEl();
20986         var s = dragEl.style;
20987
20988         this._resizeProxy();
20989
20990         if (this.centerFrame) {
20991             this.setDelta( Math.round(parseInt(s.width,  10)/2),
20992                            Math.round(parseInt(s.height, 10)/2) );
20993         }
20994
20995         this.setDragElPos(iPageX, iPageY);
20996
20997         Roo.fly(dragEl).show();
20998     },
20999
21000     /**
21001      * The proxy is automatically resized to the dimensions of the linked
21002      * element when a drag is initiated, unless resizeFrame is set to false
21003      * @method _resizeProxy
21004      * @private
21005      */
21006     _resizeProxy: function() {
21007         if (this.resizeFrame) {
21008             var el = this.getEl();
21009             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21010         }
21011     },
21012
21013     // overrides Roo.dd.DragDrop
21014     b4MouseDown: function(e) {
21015         var x = e.getPageX();
21016         var y = e.getPageY();
21017         this.autoOffset(x, y);
21018         this.setDragElPos(x, y);
21019     },
21020
21021     // overrides Roo.dd.DragDrop
21022     b4StartDrag: function(x, y) {
21023         // show the drag frame
21024         this.showFrame(x, y);
21025     },
21026
21027     // overrides Roo.dd.DragDrop
21028     b4EndDrag: function(e) {
21029         Roo.fly(this.getDragEl()).hide();
21030     },
21031
21032     // overrides Roo.dd.DragDrop
21033     // By default we try to move the element to the last location of the frame.
21034     // This is so that the default behavior mirrors that of Roo.dd.DD.
21035     endDrag: function(e) {
21036
21037         var lel = this.getEl();
21038         var del = this.getDragEl();
21039
21040         // Show the drag frame briefly so we can get its position
21041         del.style.visibility = "";
21042
21043         this.beforeMove();
21044         // Hide the linked element before the move to get around a Safari
21045         // rendering bug.
21046         lel.style.visibility = "hidden";
21047         Roo.dd.DDM.moveToEl(lel, del);
21048         del.style.visibility = "hidden";
21049         lel.style.visibility = "";
21050
21051         this.afterDrag();
21052     },
21053
21054     beforeMove : function(){
21055
21056     },
21057
21058     afterDrag : function(){
21059
21060     },
21061
21062     toString: function() {
21063         return ("DDProxy " + this.id);
21064     }
21065
21066 });
21067 /*
21068  * Based on:
21069  * Ext JS Library 1.1.1
21070  * Copyright(c) 2006-2007, Ext JS, LLC.
21071  *
21072  * Originally Released Under LGPL - original licence link has changed is not relivant.
21073  *
21074  * Fork - LGPL
21075  * <script type="text/javascript">
21076  */
21077
21078  /**
21079  * @class Roo.dd.DDTarget
21080  * A DragDrop implementation that does not move, but can be a drop
21081  * target.  You would get the same result by simply omitting implementation
21082  * for the event callbacks, but this way we reduce the processing cost of the
21083  * event listener and the callbacks.
21084  * @extends Roo.dd.DragDrop
21085  * @constructor
21086  * @param {String} id the id of the element that is a drop target
21087  * @param {String} sGroup the group of related DragDrop objects
21088  * @param {object} config an object containing configurable attributes
21089  *                 Valid properties for DDTarget in addition to those in
21090  *                 DragDrop:
21091  *                    none
21092  */
21093 Roo.dd.DDTarget = function(id, sGroup, config) {
21094     if (id) {
21095         this.initTarget(id, sGroup, config);
21096     }
21097     if (config.listeners || config.events) { 
21098        Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21099             listeners : config.listeners || {}, 
21100             events : config.events || {} 
21101         });    
21102     }
21103 };
21104
21105 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21106 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21107     toString: function() {
21108         return ("DDTarget " + this.id);
21109     }
21110 });
21111 /*
21112  * Based on:
21113  * Ext JS Library 1.1.1
21114  * Copyright(c) 2006-2007, Ext JS, LLC.
21115  *
21116  * Originally Released Under LGPL - original licence link has changed is not relivant.
21117  *
21118  * Fork - LGPL
21119  * <script type="text/javascript">
21120  */
21121  
21122
21123 /**
21124  * @class Roo.dd.ScrollManager
21125  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21126  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21127  * @singleton
21128  */
21129 Roo.dd.ScrollManager = function(){
21130     var ddm = Roo.dd.DragDropMgr;
21131     var els = {};
21132     var dragEl = null;
21133     var proc = {};
21134     
21135     
21136     
21137     var onStop = function(e){
21138         dragEl = null;
21139         clearProc();
21140     };
21141     
21142     var triggerRefresh = function(){
21143         if(ddm.dragCurrent){
21144              ddm.refreshCache(ddm.dragCurrent.groups);
21145         }
21146     };
21147     
21148     var doScroll = function(){
21149         if(ddm.dragCurrent){
21150             var dds = Roo.dd.ScrollManager;
21151             if(!dds.animate){
21152                 if(proc.el.scroll(proc.dir, dds.increment)){
21153                     triggerRefresh();
21154                 }
21155             }else{
21156                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21157             }
21158         }
21159     };
21160     
21161     var clearProc = function(){
21162         if(proc.id){
21163             clearInterval(proc.id);
21164         }
21165         proc.id = 0;
21166         proc.el = null;
21167         proc.dir = "";
21168     };
21169     
21170     var startProc = function(el, dir){
21171          Roo.log('scroll startproc');
21172         clearProc();
21173         proc.el = el;
21174         proc.dir = dir;
21175         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21176     };
21177     
21178     var onFire = function(e, isDrop){
21179        
21180         if(isDrop || !ddm.dragCurrent){ return; }
21181         var dds = Roo.dd.ScrollManager;
21182         if(!dragEl || dragEl != ddm.dragCurrent){
21183             dragEl = ddm.dragCurrent;
21184             // refresh regions on drag start
21185             dds.refreshCache();
21186         }
21187         
21188         var xy = Roo.lib.Event.getXY(e);
21189         var pt = new Roo.lib.Point(xy[0], xy[1]);
21190         for(var id in els){
21191             var el = els[id], r = el._region;
21192             if(r && r.contains(pt) && el.isScrollable()){
21193                 if(r.bottom - pt.y <= dds.thresh){
21194                     if(proc.el != el){
21195                         startProc(el, "down");
21196                     }
21197                     return;
21198                 }else if(r.right - pt.x <= dds.thresh){
21199                     if(proc.el != el){
21200                         startProc(el, "left");
21201                     }
21202                     return;
21203                 }else if(pt.y - r.top <= dds.thresh){
21204                     if(proc.el != el){
21205                         startProc(el, "up");
21206                     }
21207                     return;
21208                 }else if(pt.x - r.left <= dds.thresh){
21209                     if(proc.el != el){
21210                         startProc(el, "right");
21211                     }
21212                     return;
21213                 }
21214             }
21215         }
21216         clearProc();
21217     };
21218     
21219     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21220     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21221     
21222     return {
21223         /**
21224          * Registers new overflow element(s) to auto scroll
21225          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21226          */
21227         register : function(el){
21228             if(el instanceof Array){
21229                 for(var i = 0, len = el.length; i < len; i++) {
21230                         this.register(el[i]);
21231                 }
21232             }else{
21233                 el = Roo.get(el);
21234                 els[el.id] = el;
21235             }
21236             Roo.dd.ScrollManager.els = els;
21237         },
21238         
21239         /**
21240          * Unregisters overflow element(s) so they are no longer scrolled
21241          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21242          */
21243         unregister : function(el){
21244             if(el instanceof Array){
21245                 for(var i = 0, len = el.length; i < len; i++) {
21246                         this.unregister(el[i]);
21247                 }
21248             }else{
21249                 el = Roo.get(el);
21250                 delete els[el.id];
21251             }
21252         },
21253         
21254         /**
21255          * The number of pixels from the edge of a container the pointer needs to be to 
21256          * trigger scrolling (defaults to 25)
21257          * @type Number
21258          */
21259         thresh : 25,
21260         
21261         /**
21262          * The number of pixels to scroll in each scroll increment (defaults to 50)
21263          * @type Number
21264          */
21265         increment : 100,
21266         
21267         /**
21268          * The frequency of scrolls in milliseconds (defaults to 500)
21269          * @type Number
21270          */
21271         frequency : 500,
21272         
21273         /**
21274          * True to animate the scroll (defaults to true)
21275          * @type Boolean
21276          */
21277         animate: true,
21278         
21279         /**
21280          * The animation duration in seconds - 
21281          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21282          * @type Number
21283          */
21284         animDuration: .4,
21285         
21286         /**
21287          * Manually trigger a cache refresh.
21288          */
21289         refreshCache : function(){
21290             for(var id in els){
21291                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21292                     els[id]._region = els[id].getRegion();
21293                 }
21294             }
21295         }
21296     };
21297 }();/*
21298  * Based on:
21299  * Ext JS Library 1.1.1
21300  * Copyright(c) 2006-2007, Ext JS, LLC.
21301  *
21302  * Originally Released Under LGPL - original licence link has changed is not relivant.
21303  *
21304  * Fork - LGPL
21305  * <script type="text/javascript">
21306  */
21307  
21308
21309 /**
21310  * @class Roo.dd.Registry
21311  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21312  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21313  * @singleton
21314  */
21315 Roo.dd.Registry = function(){
21316     var elements = {}; 
21317     var handles = {}; 
21318     var autoIdSeed = 0;
21319
21320     var getId = function(el, autogen){
21321         if(typeof el == "string"){
21322             return el;
21323         }
21324         var id = el.id;
21325         if(!id && autogen !== false){
21326             id = "roodd-" + (++autoIdSeed);
21327             el.id = id;
21328         }
21329         return id;
21330     };
21331     
21332     return {
21333     /**
21334      * Register a drag drop element
21335      * @param {String|HTMLElement} element The id or DOM node to register
21336      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21337      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21338      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21339      * populated in the data object (if applicable):
21340      * <pre>
21341 Value      Description<br />
21342 ---------  ------------------------------------------<br />
21343 handles    Array of DOM nodes that trigger dragging<br />
21344            for the element being registered<br />
21345 isHandle   True if the element passed in triggers<br />
21346            dragging itself, else false
21347 </pre>
21348      */
21349         register : function(el, data){
21350             data = data || {};
21351             if(typeof el == "string"){
21352                 el = document.getElementById(el);
21353             }
21354             data.ddel = el;
21355             elements[getId(el)] = data;
21356             if(data.isHandle !== false){
21357                 handles[data.ddel.id] = data;
21358             }
21359             if(data.handles){
21360                 var hs = data.handles;
21361                 for(var i = 0, len = hs.length; i < len; i++){
21362                         handles[getId(hs[i])] = data;
21363                 }
21364             }
21365         },
21366
21367     /**
21368      * Unregister a drag drop element
21369      * @param {String|HTMLElement}  element The id or DOM node to unregister
21370      */
21371         unregister : function(el){
21372             var id = getId(el, false);
21373             var data = elements[id];
21374             if(data){
21375                 delete elements[id];
21376                 if(data.handles){
21377                     var hs = data.handles;
21378                     for(var i = 0, len = hs.length; i < len; i++){
21379                         delete handles[getId(hs[i], false)];
21380                     }
21381                 }
21382             }
21383         },
21384
21385     /**
21386      * Returns the handle registered for a DOM Node by id
21387      * @param {String|HTMLElement} id The DOM node or id to look up
21388      * @return {Object} handle The custom handle data
21389      */
21390         getHandle : function(id){
21391             if(typeof id != "string"){ // must be element?
21392                 id = id.id;
21393             }
21394             return handles[id];
21395         },
21396
21397     /**
21398      * Returns the handle that is registered for the DOM node that is the target of the event
21399      * @param {Event} e The event
21400      * @return {Object} handle The custom handle data
21401      */
21402         getHandleFromEvent : function(e){
21403             var t = Roo.lib.Event.getTarget(e);
21404             return t ? handles[t.id] : null;
21405         },
21406
21407     /**
21408      * Returns a custom data object that is registered for a DOM node by id
21409      * @param {String|HTMLElement} id The DOM node or id to look up
21410      * @return {Object} data The custom data
21411      */
21412         getTarget : function(id){
21413             if(typeof id != "string"){ // must be element?
21414                 id = id.id;
21415             }
21416             return elements[id];
21417         },
21418
21419     /**
21420      * Returns a custom data object that is registered for the DOM node that is the target of the event
21421      * @param {Event} e The event
21422      * @return {Object} data The custom data
21423      */
21424         getTargetFromEvent : function(e){
21425             var t = Roo.lib.Event.getTarget(e);
21426             return t ? elements[t.id] || handles[t.id] : null;
21427         }
21428     };
21429 }();/*
21430  * Based on:
21431  * Ext JS Library 1.1.1
21432  * Copyright(c) 2006-2007, Ext JS, LLC.
21433  *
21434  * Originally Released Under LGPL - original licence link has changed is not relivant.
21435  *
21436  * Fork - LGPL
21437  * <script type="text/javascript">
21438  */
21439  
21440
21441 /**
21442  * @class Roo.dd.StatusProxy
21443  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21444  * default drag proxy used by all Roo.dd components.
21445  * @constructor
21446  * @param {Object} config
21447  */
21448 Roo.dd.StatusProxy = function(config){
21449     Roo.apply(this, config);
21450     this.id = this.id || Roo.id();
21451     this.el = new Roo.Layer({
21452         dh: {
21453             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21454                 {tag: "div", cls: "x-dd-drop-icon"},
21455                 {tag: "div", cls: "x-dd-drag-ghost"}
21456             ]
21457         }, 
21458         shadow: !config || config.shadow !== false
21459     });
21460     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21461     this.dropStatus = this.dropNotAllowed;
21462 };
21463
21464 Roo.dd.StatusProxy.prototype = {
21465     /**
21466      * @cfg {String} dropAllowed
21467      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21468      */
21469     dropAllowed : "x-dd-drop-ok",
21470     /**
21471      * @cfg {String} dropNotAllowed
21472      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21473      */
21474     dropNotAllowed : "x-dd-drop-nodrop",
21475
21476     /**
21477      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21478      * over the current target element.
21479      * @param {String} cssClass The css class for the new drop status indicator image
21480      */
21481     setStatus : function(cssClass){
21482         cssClass = cssClass || this.dropNotAllowed;
21483         if(this.dropStatus != cssClass){
21484             this.el.replaceClass(this.dropStatus, cssClass);
21485             this.dropStatus = cssClass;
21486         }
21487     },
21488
21489     /**
21490      * Resets the status indicator to the default dropNotAllowed value
21491      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21492      */
21493     reset : function(clearGhost){
21494         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21495         this.dropStatus = this.dropNotAllowed;
21496         if(clearGhost){
21497             this.ghost.update("");
21498         }
21499     },
21500
21501     /**
21502      * Updates the contents of the ghost element
21503      * @param {String} html The html that will replace the current innerHTML of the ghost element
21504      */
21505     update : function(html){
21506         if(typeof html == "string"){
21507             this.ghost.update(html);
21508         }else{
21509             this.ghost.update("");
21510             html.style.margin = "0";
21511             this.ghost.dom.appendChild(html);
21512         }
21513         // ensure float = none set?? cant remember why though.
21514         var el = this.ghost.dom.firstChild;
21515                 if(el){
21516                         Roo.fly(el).setStyle('float', 'none');
21517                 }
21518     },
21519     
21520     /**
21521      * Returns the underlying proxy {@link Roo.Layer}
21522      * @return {Roo.Layer} el
21523     */
21524     getEl : function(){
21525         return this.el;
21526     },
21527
21528     /**
21529      * Returns the ghost element
21530      * @return {Roo.Element} el
21531      */
21532     getGhost : function(){
21533         return this.ghost;
21534     },
21535
21536     /**
21537      * Hides the proxy
21538      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21539      */
21540     hide : function(clear){
21541         this.el.hide();
21542         if(clear){
21543             this.reset(true);
21544         }
21545     },
21546
21547     /**
21548      * Stops the repair animation if it's currently running
21549      */
21550     stop : function(){
21551         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21552             this.anim.stop();
21553         }
21554     },
21555
21556     /**
21557      * Displays this proxy
21558      */
21559     show : function(){
21560         this.el.show();
21561     },
21562
21563     /**
21564      * Force the Layer to sync its shadow and shim positions to the element
21565      */
21566     sync : function(){
21567         this.el.sync();
21568     },
21569
21570     /**
21571      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21572      * invalid drop operation by the item being dragged.
21573      * @param {Array} xy The XY position of the element ([x, y])
21574      * @param {Function} callback The function to call after the repair is complete
21575      * @param {Object} scope The scope in which to execute the callback
21576      */
21577     repair : function(xy, callback, scope){
21578         this.callback = callback;
21579         this.scope = scope;
21580         if(xy && this.animRepair !== false){
21581             this.el.addClass("x-dd-drag-repair");
21582             this.el.hideUnders(true);
21583             this.anim = this.el.shift({
21584                 duration: this.repairDuration || .5,
21585                 easing: 'easeOut',
21586                 xy: xy,
21587                 stopFx: true,
21588                 callback: this.afterRepair,
21589                 scope: this
21590             });
21591         }else{
21592             this.afterRepair();
21593         }
21594     },
21595
21596     // private
21597     afterRepair : function(){
21598         this.hide(true);
21599         if(typeof this.callback == "function"){
21600             this.callback.call(this.scope || this);
21601         }
21602         this.callback = null;
21603         this.scope = null;
21604     }
21605 };/*
21606  * Based on:
21607  * Ext JS Library 1.1.1
21608  * Copyright(c) 2006-2007, Ext JS, LLC.
21609  *
21610  * Originally Released Under LGPL - original licence link has changed is not relivant.
21611  *
21612  * Fork - LGPL
21613  * <script type="text/javascript">
21614  */
21615
21616 /**
21617  * @class Roo.dd.DragSource
21618  * @extends Roo.dd.DDProxy
21619  * A simple class that provides the basic implementation needed to make any element draggable.
21620  * @constructor
21621  * @param {String/HTMLElement/Element} el The container element
21622  * @param {Object} config
21623  */
21624 Roo.dd.DragSource = function(el, config){
21625     this.el = Roo.get(el);
21626     this.dragData = {};
21627     
21628     Roo.apply(this, config);
21629     
21630     if(!this.proxy){
21631         this.proxy = new Roo.dd.StatusProxy();
21632     }
21633
21634     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21635           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21636     
21637     this.dragging = false;
21638 };
21639
21640 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21641     /**
21642      * @cfg {String} dropAllowed
21643      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21644      */
21645     dropAllowed : "x-dd-drop-ok",
21646     /**
21647      * @cfg {String} dropNotAllowed
21648      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21649      */
21650     dropNotAllowed : "x-dd-drop-nodrop",
21651
21652     /**
21653      * Returns the data object associated with this drag source
21654      * @return {Object} data An object containing arbitrary data
21655      */
21656     getDragData : function(e){
21657         return this.dragData;
21658     },
21659
21660     // private
21661     onDragEnter : function(e, id){
21662         var target = Roo.dd.DragDropMgr.getDDById(id);
21663         this.cachedTarget = target;
21664         if(this.beforeDragEnter(target, e, id) !== false){
21665             if(target.isNotifyTarget){
21666                 var status = target.notifyEnter(this, e, this.dragData);
21667                 this.proxy.setStatus(status);
21668             }else{
21669                 this.proxy.setStatus(this.dropAllowed);
21670             }
21671             
21672             if(this.afterDragEnter){
21673                 /**
21674                  * An empty function by default, but provided so that you can perform a custom action
21675                  * when the dragged item enters the drop target by providing an implementation.
21676                  * @param {Roo.dd.DragDrop} target The drop target
21677                  * @param {Event} e The event object
21678                  * @param {String} id The id of the dragged element
21679                  * @method afterDragEnter
21680                  */
21681                 this.afterDragEnter(target, e, id);
21682             }
21683         }
21684     },
21685
21686     /**
21687      * An empty function by default, but provided so that you can perform a custom action
21688      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21689      * @param {Roo.dd.DragDrop} target The drop target
21690      * @param {Event} e The event object
21691      * @param {String} id The id of the dragged element
21692      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21693      */
21694     beforeDragEnter : function(target, e, id){
21695         return true;
21696     },
21697
21698     // private
21699     alignElWithMouse: function() {
21700         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21701         this.proxy.sync();
21702     },
21703
21704     // private
21705     onDragOver : function(e, id){
21706         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21707         if(this.beforeDragOver(target, e, id) !== false){
21708             if(target.isNotifyTarget){
21709                 var status = target.notifyOver(this, e, this.dragData);
21710                 this.proxy.setStatus(status);
21711             }
21712
21713             if(this.afterDragOver){
21714                 /**
21715                  * An empty function by default, but provided so that you can perform a custom action
21716                  * while the dragged item is over the drop target by providing an implementation.
21717                  * @param {Roo.dd.DragDrop} target The drop target
21718                  * @param {Event} e The event object
21719                  * @param {String} id The id of the dragged element
21720                  * @method afterDragOver
21721                  */
21722                 this.afterDragOver(target, e, id);
21723             }
21724         }
21725     },
21726
21727     /**
21728      * An empty function by default, but provided so that you can perform a custom action
21729      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21730      * @param {Roo.dd.DragDrop} target The drop target
21731      * @param {Event} e The event object
21732      * @param {String} id The id of the dragged element
21733      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21734      */
21735     beforeDragOver : function(target, e, id){
21736         return true;
21737     },
21738
21739     // private
21740     onDragOut : function(e, id){
21741         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21742         if(this.beforeDragOut(target, e, id) !== false){
21743             if(target.isNotifyTarget){
21744                 target.notifyOut(this, e, this.dragData);
21745             }
21746             this.proxy.reset();
21747             if(this.afterDragOut){
21748                 /**
21749                  * An empty function by default, but provided so that you can perform a custom action
21750                  * after the dragged item is dragged out of the target without dropping.
21751                  * @param {Roo.dd.DragDrop} target The drop target
21752                  * @param {Event} e The event object
21753                  * @param {String} id The id of the dragged element
21754                  * @method afterDragOut
21755                  */
21756                 this.afterDragOut(target, e, id);
21757             }
21758         }
21759         this.cachedTarget = null;
21760     },
21761
21762     /**
21763      * An empty function by default, but provided so that you can perform a custom action before the dragged
21764      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21765      * @param {Roo.dd.DragDrop} target The drop target
21766      * @param {Event} e The event object
21767      * @param {String} id The id of the dragged element
21768      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21769      */
21770     beforeDragOut : function(target, e, id){
21771         return true;
21772     },
21773     
21774     // private
21775     onDragDrop : function(e, id){
21776         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21777         if(this.beforeDragDrop(target, e, id) !== false){
21778             if(target.isNotifyTarget){
21779                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21780                     this.onValidDrop(target, e, id);
21781                 }else{
21782                     this.onInvalidDrop(target, e, id);
21783                 }
21784             }else{
21785                 this.onValidDrop(target, e, id);
21786             }
21787             
21788             if(this.afterDragDrop){
21789                 /**
21790                  * An empty function by default, but provided so that you can perform a custom action
21791                  * after a valid drag drop has occurred by providing an implementation.
21792                  * @param {Roo.dd.DragDrop} target The drop target
21793                  * @param {Event} e The event object
21794                  * @param {String} id The id of the dropped element
21795                  * @method afterDragDrop
21796                  */
21797                 this.afterDragDrop(target, e, id);
21798             }
21799         }
21800         delete this.cachedTarget;
21801     },
21802
21803     /**
21804      * An empty function by default, but provided so that you can perform a custom action before the dragged
21805      * item is dropped onto the target and optionally cancel the onDragDrop.
21806      * @param {Roo.dd.DragDrop} target The drop target
21807      * @param {Event} e The event object
21808      * @param {String} id The id of the dragged element
21809      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21810      */
21811     beforeDragDrop : function(target, e, id){
21812         return true;
21813     },
21814
21815     // private
21816     onValidDrop : function(target, e, id){
21817         this.hideProxy();
21818         if(this.afterValidDrop){
21819             /**
21820              * An empty function by default, but provided so that you can perform a custom action
21821              * after a valid drop has occurred by providing an implementation.
21822              * @param {Object} target The target DD 
21823              * @param {Event} e The event object
21824              * @param {String} id The id of the dropped element
21825              * @method afterInvalidDrop
21826              */
21827             this.afterValidDrop(target, e, id);
21828         }
21829     },
21830
21831     // private
21832     getRepairXY : function(e, data){
21833         return this.el.getXY();  
21834     },
21835
21836     // private
21837     onInvalidDrop : function(target, e, id){
21838         this.beforeInvalidDrop(target, e, id);
21839         if(this.cachedTarget){
21840             if(this.cachedTarget.isNotifyTarget){
21841                 this.cachedTarget.notifyOut(this, e, this.dragData);
21842             }
21843             this.cacheTarget = null;
21844         }
21845         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21846
21847         if(this.afterInvalidDrop){
21848             /**
21849              * An empty function by default, but provided so that you can perform a custom action
21850              * after an invalid drop has occurred by providing an implementation.
21851              * @param {Event} e The event object
21852              * @param {String} id The id of the dropped element
21853              * @method afterInvalidDrop
21854              */
21855             this.afterInvalidDrop(e, id);
21856         }
21857     },
21858
21859     // private
21860     afterRepair : function(){
21861         if(Roo.enableFx){
21862             this.el.highlight(this.hlColor || "c3daf9");
21863         }
21864         this.dragging = false;
21865     },
21866
21867     /**
21868      * An empty function by default, but provided so that you can perform a custom action after an invalid
21869      * drop has occurred.
21870      * @param {Roo.dd.DragDrop} target The drop target
21871      * @param {Event} e The event object
21872      * @param {String} id The id of the dragged element
21873      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21874      */
21875     beforeInvalidDrop : function(target, e, id){
21876         return true;
21877     },
21878
21879     // private
21880     handleMouseDown : function(e){
21881         if(this.dragging) {
21882             return;
21883         }
21884         var data = this.getDragData(e);
21885         if(data && this.onBeforeDrag(data, e) !== false){
21886             this.dragData = data;
21887             this.proxy.stop();
21888             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21889         } 
21890     },
21891
21892     /**
21893      * An empty function by default, but provided so that you can perform a custom action before the initial
21894      * drag event begins and optionally cancel it.
21895      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21896      * @param {Event} e The event object
21897      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21898      */
21899     onBeforeDrag : function(data, e){
21900         return true;
21901     },
21902
21903     /**
21904      * An empty function by default, but provided so that you can perform a custom action once the initial
21905      * drag event has begun.  The drag cannot be canceled from this function.
21906      * @param {Number} x The x position of the click on the dragged object
21907      * @param {Number} y The y position of the click on the dragged object
21908      */
21909     onStartDrag : Roo.emptyFn,
21910
21911     // private - YUI override
21912     startDrag : function(x, y){
21913         this.proxy.reset();
21914         this.dragging = true;
21915         this.proxy.update("");
21916         this.onInitDrag(x, y);
21917         this.proxy.show();
21918     },
21919
21920     // private
21921     onInitDrag : function(x, y){
21922         var clone = this.el.dom.cloneNode(true);
21923         clone.id = Roo.id(); // prevent duplicate ids
21924         this.proxy.update(clone);
21925         this.onStartDrag(x, y);
21926         return true;
21927     },
21928
21929     /**
21930      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21931      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21932      */
21933     getProxy : function(){
21934         return this.proxy;  
21935     },
21936
21937     /**
21938      * Hides the drag source's {@link Roo.dd.StatusProxy}
21939      */
21940     hideProxy : function(){
21941         this.proxy.hide();  
21942         this.proxy.reset(true);
21943         this.dragging = false;
21944     },
21945
21946     // private
21947     triggerCacheRefresh : function(){
21948         Roo.dd.DDM.refreshCache(this.groups);
21949     },
21950
21951     // private - override to prevent hiding
21952     b4EndDrag: function(e) {
21953     },
21954
21955     // private - override to prevent moving
21956     endDrag : function(e){
21957         this.onEndDrag(this.dragData, e);
21958     },
21959
21960     // private
21961     onEndDrag : function(data, e){
21962     },
21963     
21964     // private - pin to cursor
21965     autoOffset : function(x, y) {
21966         this.setDelta(-12, -20);
21967     }    
21968 });/*
21969  * Based on:
21970  * Ext JS Library 1.1.1
21971  * Copyright(c) 2006-2007, Ext JS, LLC.
21972  *
21973  * Originally Released Under LGPL - original licence link has changed is not relivant.
21974  *
21975  * Fork - LGPL
21976  * <script type="text/javascript">
21977  */
21978
21979
21980 /**
21981  * @class Roo.dd.DropTarget
21982  * @extends Roo.dd.DDTarget
21983  * A simple class that provides the basic implementation needed to make any element a drop target that can have
21984  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
21985  * @constructor
21986  * @param {String/HTMLElement/Element} el The container element
21987  * @param {Object} config
21988  */
21989 Roo.dd.DropTarget = function(el, config){
21990     this.el = Roo.get(el);
21991     
21992     var listeners = false; ;
21993     if (config && config.listeners) {
21994         listeners= config.listeners;
21995         delete config.listeners;
21996     }
21997     Roo.apply(this, config);
21998     
21999     if(this.containerScroll){
22000         Roo.dd.ScrollManager.register(this.el);
22001     }
22002     this.addEvents( {
22003          /**
22004          * @scope Roo.dd.DropTarget
22005          */
22006          
22007          /**
22008          * @event enter
22009          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22010          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22011          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22012          * 
22013          * IMPORTANT : it should set this.overClass and this.dropAllowed
22014          * 
22015          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22016          * @param {Event} e The event
22017          * @param {Object} data An object containing arbitrary data supplied by the drag source
22018          */
22019         "enter" : true,
22020         
22021          /**
22022          * @event over
22023          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22024          * This method will be called on every mouse movement while the drag source is over the drop target.
22025          * This default implementation simply returns the dropAllowed config value.
22026          * 
22027          * IMPORTANT : it should set this.dropAllowed
22028          * 
22029          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22030          * @param {Event} e The event
22031          * @param {Object} data An object containing arbitrary data supplied by the drag source
22032          
22033          */
22034         "over" : true,
22035         /**
22036          * @event out
22037          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22038          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22039          * overClass (if any) from the drop element.
22040          * 
22041          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22042          * @param {Event} e The event
22043          * @param {Object} data An object containing arbitrary data supplied by the drag source
22044          */
22045          "out" : true,
22046          
22047         /**
22048          * @event drop
22049          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22050          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22051          * implementation that does something to process the drop event and returns true so that the drag source's
22052          * repair action does not run.
22053          * 
22054          * IMPORTANT : it should set this.success
22055          * 
22056          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22057          * @param {Event} e The event
22058          * @param {Object} data An object containing arbitrary data supplied by the drag source
22059         */
22060          "drop" : true
22061     });
22062             
22063      
22064     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22065         this.el.dom, 
22066         this.ddGroup || this.group,
22067         {
22068             isTarget: true,
22069             listeners : listeners || {} 
22070            
22071         
22072         }
22073     );
22074
22075 };
22076
22077 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22078     /**
22079      * @cfg {String} overClass
22080      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22081      */
22082      /**
22083      * @cfg {String} ddGroup
22084      * The drag drop group to handle drop events for
22085      */
22086      
22087     /**
22088      * @cfg {String} dropAllowed
22089      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22090      */
22091     dropAllowed : "x-dd-drop-ok",
22092     /**
22093      * @cfg {String} dropNotAllowed
22094      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22095      */
22096     dropNotAllowed : "x-dd-drop-nodrop",
22097     /**
22098      * @cfg {boolean} success
22099      * set this after drop listener.. 
22100      */
22101     success : false,
22102     /**
22103      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22104      * if the drop point is valid for over/enter..
22105      */
22106     valid : false,
22107     // private
22108     isTarget : true,
22109
22110     // private
22111     isNotifyTarget : true,
22112     
22113     /**
22114      * @hide
22115      */
22116     notifyEnter : function(dd, e, data)
22117     {
22118         this.valid = true;
22119         this.fireEvent('enter', dd, e, data);
22120         if(this.overClass){
22121             this.el.addClass(this.overClass);
22122         }
22123         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22124             this.valid ? this.dropAllowed : this.dropNotAllowed
22125         );
22126     },
22127
22128     /**
22129      * @hide
22130      */
22131     notifyOver : function(dd, e, data)
22132     {
22133         this.valid = true;
22134         this.fireEvent('over', dd, e, data);
22135         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22136             this.valid ? this.dropAllowed : this.dropNotAllowed
22137         );
22138     },
22139
22140     /**
22141      * @hide
22142      */
22143     notifyOut : function(dd, e, data)
22144     {
22145         this.fireEvent('out', dd, e, data);
22146         if(this.overClass){
22147             this.el.removeClass(this.overClass);
22148         }
22149     },
22150
22151     /**
22152      * @hide
22153      */
22154     notifyDrop : function(dd, e, data)
22155     {
22156         this.success = false;
22157         this.fireEvent('drop', dd, e, data);
22158         return this.success;
22159     }
22160 });/*
22161  * Based on:
22162  * Ext JS Library 1.1.1
22163  * Copyright(c) 2006-2007, Ext JS, LLC.
22164  *
22165  * Originally Released Under LGPL - original licence link has changed is not relivant.
22166  *
22167  * Fork - LGPL
22168  * <script type="text/javascript">
22169  */
22170
22171
22172 /**
22173  * @class Roo.dd.DragZone
22174  * @extends Roo.dd.DragSource
22175  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22176  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22177  * @constructor
22178  * @param {String/HTMLElement/Element} el The container element
22179  * @param {Object} config
22180  */
22181 Roo.dd.DragZone = function(el, config){
22182     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22183     if(this.containerScroll){
22184         Roo.dd.ScrollManager.register(this.el);
22185     }
22186 };
22187
22188 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22189     /**
22190      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22191      * for auto scrolling during drag operations.
22192      */
22193     /**
22194      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22195      * method after a failed drop (defaults to "c3daf9" - light blue)
22196      */
22197
22198     /**
22199      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22200      * for a valid target to drag based on the mouse down. Override this method
22201      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22202      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22203      * @param {EventObject} e The mouse down event
22204      * @return {Object} The dragData
22205      */
22206     getDragData : function(e){
22207         return Roo.dd.Registry.getHandleFromEvent(e);
22208     },
22209     
22210     /**
22211      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22212      * this.dragData.ddel
22213      * @param {Number} x The x position of the click on the dragged object
22214      * @param {Number} y The y position of the click on the dragged object
22215      * @return {Boolean} true to continue the drag, false to cancel
22216      */
22217     onInitDrag : function(x, y){
22218         this.proxy.update(this.dragData.ddel.cloneNode(true));
22219         this.onStartDrag(x, y);
22220         return true;
22221     },
22222     
22223     /**
22224      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22225      */
22226     afterRepair : function(){
22227         if(Roo.enableFx){
22228             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22229         }
22230         this.dragging = false;
22231     },
22232
22233     /**
22234      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22235      * the XY of this.dragData.ddel
22236      * @param {EventObject} e The mouse up event
22237      * @return {Array} The xy location (e.g. [100, 200])
22238      */
22239     getRepairXY : function(e){
22240         return Roo.Element.fly(this.dragData.ddel).getXY();  
22241     }
22242 });/*
22243  * Based on:
22244  * Ext JS Library 1.1.1
22245  * Copyright(c) 2006-2007, Ext JS, LLC.
22246  *
22247  * Originally Released Under LGPL - original licence link has changed is not relivant.
22248  *
22249  * Fork - LGPL
22250  * <script type="text/javascript">
22251  */
22252 /**
22253  * @class Roo.dd.DropZone
22254  * @extends Roo.dd.DropTarget
22255  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22256  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22257  * @constructor
22258  * @param {String/HTMLElement/Element} el The container element
22259  * @param {Object} config
22260  */
22261 Roo.dd.DropZone = function(el, config){
22262     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22263 };
22264
22265 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22266     /**
22267      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22268      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22269      * provide your own custom lookup.
22270      * @param {Event} e The event
22271      * @return {Object} data The custom data
22272      */
22273     getTargetFromEvent : function(e){
22274         return Roo.dd.Registry.getTargetFromEvent(e);
22275     },
22276
22277     /**
22278      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22279      * that it has registered.  This method has no default implementation and should be overridden to provide
22280      * node-specific processing if necessary.
22281      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22282      * {@link #getTargetFromEvent} for this node)
22283      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22284      * @param {Event} e The event
22285      * @param {Object} data An object containing arbitrary data supplied by the drag source
22286      */
22287     onNodeEnter : function(n, dd, e, data){
22288         
22289     },
22290
22291     /**
22292      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22293      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22294      * overridden to provide the proper feedback.
22295      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22296      * {@link #getTargetFromEvent} for this node)
22297      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22298      * @param {Event} e The event
22299      * @param {Object} data An object containing arbitrary data supplied by the drag source
22300      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22301      * underlying {@link Roo.dd.StatusProxy} can be updated
22302      */
22303     onNodeOver : function(n, dd, e, data){
22304         return this.dropAllowed;
22305     },
22306
22307     /**
22308      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22309      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22310      * node-specific processing if necessary.
22311      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22312      * {@link #getTargetFromEvent} for this node)
22313      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22314      * @param {Event} e The event
22315      * @param {Object} data An object containing arbitrary data supplied by the drag source
22316      */
22317     onNodeOut : function(n, dd, e, data){
22318         
22319     },
22320
22321     /**
22322      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22323      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22324      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22325      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22326      * {@link #getTargetFromEvent} for this node)
22327      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22328      * @param {Event} e The event
22329      * @param {Object} data An object containing arbitrary data supplied by the drag source
22330      * @return {Boolean} True if the drop was valid, else false
22331      */
22332     onNodeDrop : function(n, dd, e, data){
22333         return false;
22334     },
22335
22336     /**
22337      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22338      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22339      * it should be overridden to provide the proper feedback if necessary.
22340      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22341      * @param {Event} e The event
22342      * @param {Object} data An object containing arbitrary data supplied by the drag source
22343      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22344      * underlying {@link Roo.dd.StatusProxy} can be updated
22345      */
22346     onContainerOver : function(dd, e, data){
22347         return this.dropNotAllowed;
22348     },
22349
22350     /**
22351      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22352      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22353      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22354      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22355      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22356      * @param {Event} e The event
22357      * @param {Object} data An object containing arbitrary data supplied by the drag source
22358      * @return {Boolean} True if the drop was valid, else false
22359      */
22360     onContainerDrop : function(dd, e, data){
22361         return false;
22362     },
22363
22364     /**
22365      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22366      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22367      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22368      * you should override this method and provide a custom implementation.
22369      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22370      * @param {Event} e The event
22371      * @param {Object} data An object containing arbitrary data supplied by the drag source
22372      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22373      * underlying {@link Roo.dd.StatusProxy} can be updated
22374      */
22375     notifyEnter : function(dd, e, data){
22376         return this.dropNotAllowed;
22377     },
22378
22379     /**
22380      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22381      * This method will be called on every mouse movement while the drag source is over the drop zone.
22382      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22383      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22384      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22385      * registered node, it will call {@link #onContainerOver}.
22386      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22387      * @param {Event} e The event
22388      * @param {Object} data An object containing arbitrary data supplied by the drag source
22389      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22390      * underlying {@link Roo.dd.StatusProxy} can be updated
22391      */
22392     notifyOver : function(dd, e, data){
22393         var n = this.getTargetFromEvent(e);
22394         if(!n){ // not over valid drop target
22395             if(this.lastOverNode){
22396                 this.onNodeOut(this.lastOverNode, dd, e, data);
22397                 this.lastOverNode = null;
22398             }
22399             return this.onContainerOver(dd, e, data);
22400         }
22401         if(this.lastOverNode != n){
22402             if(this.lastOverNode){
22403                 this.onNodeOut(this.lastOverNode, dd, e, data);
22404             }
22405             this.onNodeEnter(n, dd, e, data);
22406             this.lastOverNode = n;
22407         }
22408         return this.onNodeOver(n, dd, e, data);
22409     },
22410
22411     /**
22412      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22413      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22414      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22415      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22416      * @param {Event} e The event
22417      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22418      */
22419     notifyOut : function(dd, e, data){
22420         if(this.lastOverNode){
22421             this.onNodeOut(this.lastOverNode, dd, e, data);
22422             this.lastOverNode = null;
22423         }
22424     },
22425
22426     /**
22427      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22428      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22429      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22430      * otherwise it will call {@link #onContainerDrop}.
22431      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22432      * @param {Event} e The event
22433      * @param {Object} data An object containing arbitrary data supplied by the drag source
22434      * @return {Boolean} True if the drop was valid, else false
22435      */
22436     notifyDrop : function(dd, e, data){
22437         if(this.lastOverNode){
22438             this.onNodeOut(this.lastOverNode, dd, e, data);
22439             this.lastOverNode = null;
22440         }
22441         var n = this.getTargetFromEvent(e);
22442         return n ?
22443             this.onNodeDrop(n, dd, e, data) :
22444             this.onContainerDrop(dd, e, data);
22445     },
22446
22447     // private
22448     triggerCacheRefresh : function(){
22449         Roo.dd.DDM.refreshCache(this.groups);
22450     }  
22451 });/*
22452  * Based on:
22453  * Ext JS Library 1.1.1
22454  * Copyright(c) 2006-2007, Ext JS, LLC.
22455  *
22456  * Originally Released Under LGPL - original licence link has changed is not relivant.
22457  *
22458  * Fork - LGPL
22459  * <script type="text/javascript">
22460  */
22461
22462
22463 /**
22464  * @class Roo.data.SortTypes
22465  * @singleton
22466  * Defines the default sorting (casting?) comparison functions used when sorting data.
22467  */
22468 Roo.data.SortTypes = {
22469     /**
22470      * Default sort that does nothing
22471      * @param {Mixed} s The value being converted
22472      * @return {Mixed} The comparison value
22473      */
22474     none : function(s){
22475         return s;
22476     },
22477     
22478     /**
22479      * The regular expression used to strip tags
22480      * @type {RegExp}
22481      * @property
22482      */
22483     stripTagsRE : /<\/?[^>]+>/gi,
22484     
22485     /**
22486      * Strips all HTML tags to sort on text only
22487      * @param {Mixed} s The value being converted
22488      * @return {String} The comparison value
22489      */
22490     asText : function(s){
22491         return String(s).replace(this.stripTagsRE, "");
22492     },
22493     
22494     /**
22495      * Strips all HTML tags to sort on text only - Case insensitive
22496      * @param {Mixed} s The value being converted
22497      * @return {String} The comparison value
22498      */
22499     asUCText : function(s){
22500         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22501     },
22502     
22503     /**
22504      * Case insensitive string
22505      * @param {Mixed} s The value being converted
22506      * @return {String} The comparison value
22507      */
22508     asUCString : function(s) {
22509         return String(s).toUpperCase();
22510     },
22511     
22512     /**
22513      * Date sorting
22514      * @param {Mixed} s The value being converted
22515      * @return {Number} The comparison value
22516      */
22517     asDate : function(s) {
22518         if(!s){
22519             return 0;
22520         }
22521         if(s instanceof Date){
22522             return s.getTime();
22523         }
22524         return Date.parse(String(s));
22525     },
22526     
22527     /**
22528      * Float sorting
22529      * @param {Mixed} s The value being converted
22530      * @return {Float} The comparison value
22531      */
22532     asFloat : function(s) {
22533         var val = parseFloat(String(s).replace(/,/g, ""));
22534         if(isNaN(val)) {
22535             val = 0;
22536         }
22537         return val;
22538     },
22539     
22540     /**
22541      * Integer sorting
22542      * @param {Mixed} s The value being converted
22543      * @return {Number} The comparison value
22544      */
22545     asInt : function(s) {
22546         var val = parseInt(String(s).replace(/,/g, ""));
22547         if(isNaN(val)) {
22548             val = 0;
22549         }
22550         return val;
22551     }
22552 };/*
22553  * Based on:
22554  * Ext JS Library 1.1.1
22555  * Copyright(c) 2006-2007, Ext JS, LLC.
22556  *
22557  * Originally Released Under LGPL - original licence link has changed is not relivant.
22558  *
22559  * Fork - LGPL
22560  * <script type="text/javascript">
22561  */
22562
22563 /**
22564 * @class Roo.data.Record
22565  * Instances of this class encapsulate both record <em>definition</em> information, and record
22566  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22567  * to access Records cached in an {@link Roo.data.Store} object.<br>
22568  * <p>
22569  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22570  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22571  * objects.<br>
22572  * <p>
22573  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22574  * @constructor
22575  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22576  * {@link #create}. The parameters are the same.
22577  * @param {Array} data An associative Array of data values keyed by the field name.
22578  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22579  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22580  * not specified an integer id is generated.
22581  */
22582 Roo.data.Record = function(data, id){
22583     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22584     this.data = data;
22585 };
22586
22587 /**
22588  * Generate a constructor for a specific record layout.
22589  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22590  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22591  * Each field definition object may contain the following properties: <ul>
22592  * <li><b>name</b> : String<p style="margin-left:1em">The name by which the field is referenced within the Record. This is referenced by,
22593  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22594  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22595  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22596  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22597  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22598  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22599  * this may be omitted.</p></li>
22600  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22601  * <ul><li>auto (Default, implies no conversion)</li>
22602  * <li>string</li>
22603  * <li>int</li>
22604  * <li>float</li>
22605  * <li>boolean</li>
22606  * <li>date</li></ul></p></li>
22607  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22608  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22609  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22610  * by the Reader into an object that will be stored in the Record. It is passed the
22611  * following parameters:<ul>
22612  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22613  * </ul></p></li>
22614  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22615  * </ul>
22616  * <br>usage:<br><pre><code>
22617 var TopicRecord = Roo.data.Record.create(
22618     {name: 'title', mapping: 'topic_title'},
22619     {name: 'author', mapping: 'username'},
22620     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22621     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22622     {name: 'lastPoster', mapping: 'user2'},
22623     {name: 'excerpt', mapping: 'post_text'}
22624 );
22625
22626 var myNewRecord = new TopicRecord({
22627     title: 'Do my job please',
22628     author: 'noobie',
22629     totalPosts: 1,
22630     lastPost: new Date(),
22631     lastPoster: 'Animal',
22632     excerpt: 'No way dude!'
22633 });
22634 myStore.add(myNewRecord);
22635 </code></pre>
22636  * @method create
22637  * @static
22638  */
22639 Roo.data.Record.create = function(o){
22640     var f = function(){
22641         f.superclass.constructor.apply(this, arguments);
22642     };
22643     Roo.extend(f, Roo.data.Record);
22644     var p = f.prototype;
22645     p.fields = new Roo.util.MixedCollection(false, function(field){
22646         return field.name;
22647     });
22648     for(var i = 0, len = o.length; i < len; i++){
22649         p.fields.add(new Roo.data.Field(o[i]));
22650     }
22651     f.getField = function(name){
22652         return p.fields.get(name);  
22653     };
22654     return f;
22655 };
22656
22657 Roo.data.Record.AUTO_ID = 1000;
22658 Roo.data.Record.EDIT = 'edit';
22659 Roo.data.Record.REJECT = 'reject';
22660 Roo.data.Record.COMMIT = 'commit';
22661
22662 Roo.data.Record.prototype = {
22663     /**
22664      * Readonly flag - true if this record has been modified.
22665      * @type Boolean
22666      */
22667     dirty : false,
22668     editing : false,
22669     error: null,
22670     modified: null,
22671
22672     // private
22673     join : function(store){
22674         this.store = store;
22675     },
22676
22677     /**
22678      * Set the named field to the specified value.
22679      * @param {String} name The name of the field to set.
22680      * @param {Object} value The value to set the field to.
22681      */
22682     set : function(name, value){
22683         if(this.data[name] == value){
22684             return;
22685         }
22686         this.dirty = true;
22687         if(!this.modified){
22688             this.modified = {};
22689         }
22690         if(typeof this.modified[name] == 'undefined'){
22691             this.modified[name] = this.data[name];
22692         }
22693         this.data[name] = value;
22694         if(!this.editing && this.store){
22695             this.store.afterEdit(this);
22696         }       
22697     },
22698
22699     /**
22700      * Get the value of the named field.
22701      * @param {String} name The name of the field to get the value of.
22702      * @return {Object} The value of the field.
22703      */
22704     get : function(name){
22705         return this.data[name]; 
22706     },
22707
22708     // private
22709     beginEdit : function(){
22710         this.editing = true;
22711         this.modified = {}; 
22712     },
22713
22714     // private
22715     cancelEdit : function(){
22716         this.editing = false;
22717         delete this.modified;
22718     },
22719
22720     // private
22721     endEdit : function(){
22722         this.editing = false;
22723         if(this.dirty && this.store){
22724             this.store.afterEdit(this);
22725         }
22726     },
22727
22728     /**
22729      * Usually called by the {@link Roo.data.Store} which owns the Record.
22730      * Rejects all changes made to the Record since either creation, or the last commit operation.
22731      * Modified fields are reverted to their original values.
22732      * <p>
22733      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22734      * of reject operations.
22735      */
22736     reject : function(){
22737         var m = this.modified;
22738         for(var n in m){
22739             if(typeof m[n] != "function"){
22740                 this.data[n] = m[n];
22741             }
22742         }
22743         this.dirty = false;
22744         delete this.modified;
22745         this.editing = false;
22746         if(this.store){
22747             this.store.afterReject(this);
22748         }
22749     },
22750
22751     /**
22752      * Usually called by the {@link Roo.data.Store} which owns the Record.
22753      * Commits all changes made to the Record since either creation, or the last commit operation.
22754      * <p>
22755      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22756      * of commit operations.
22757      */
22758     commit : function(){
22759         this.dirty = false;
22760         delete this.modified;
22761         this.editing = false;
22762         if(this.store){
22763             this.store.afterCommit(this);
22764         }
22765     },
22766
22767     // private
22768     hasError : function(){
22769         return this.error != null;
22770     },
22771
22772     // private
22773     clearError : function(){
22774         this.error = null;
22775     },
22776
22777     /**
22778      * Creates a copy of this record.
22779      * @param {String} id (optional) A new record id if you don't want to use this record's id
22780      * @return {Record}
22781      */
22782     copy : function(newId) {
22783         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22784     }
22785 };/*
22786  * Based on:
22787  * Ext JS Library 1.1.1
22788  * Copyright(c) 2006-2007, Ext JS, LLC.
22789  *
22790  * Originally Released Under LGPL - original licence link has changed is not relivant.
22791  *
22792  * Fork - LGPL
22793  * <script type="text/javascript">
22794  */
22795
22796
22797
22798 /**
22799  * @class Roo.data.Store
22800  * @extends Roo.util.Observable
22801  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22802  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22803  * <p>
22804  * A Store object uses an implementation of {@link Roo.data.DataProxy} to access a data object unless you call loadData() directly and pass in your data. The Store object
22805  * has no knowledge of the format of the data returned by the Proxy.<br>
22806  * <p>
22807  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22808  * instances from the data object. These records are cached and made available through accessor functions.
22809  * @constructor
22810  * Creates a new Store.
22811  * @param {Object} config A config object containing the objects needed for the Store to access data,
22812  * and read the data into Records.
22813  */
22814 Roo.data.Store = function(config){
22815     this.data = new Roo.util.MixedCollection(false);
22816     this.data.getKey = function(o){
22817         return o.id;
22818     };
22819     this.baseParams = {};
22820     // private
22821     this.paramNames = {
22822         "start" : "start",
22823         "limit" : "limit",
22824         "sort" : "sort",
22825         "dir" : "dir",
22826         "multisort" : "_multisort"
22827     };
22828
22829     if(config && config.data){
22830         this.inlineData = config.data;
22831         delete config.data;
22832     }
22833
22834     Roo.apply(this, config);
22835     
22836     if(this.reader){ // reader passed
22837         this.reader = Roo.factory(this.reader, Roo.data);
22838         this.reader.xmodule = this.xmodule || false;
22839         if(!this.recordType){
22840             this.recordType = this.reader.recordType;
22841         }
22842         if(this.reader.onMetaChange){
22843             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22844         }
22845     }
22846
22847     if(this.recordType){
22848         this.fields = this.recordType.prototype.fields;
22849     }
22850     this.modified = [];
22851
22852     this.addEvents({
22853         /**
22854          * @event datachanged
22855          * Fires when the data cache has changed, and a widget which is using this Store
22856          * as a Record cache should refresh its view.
22857          * @param {Store} this
22858          */
22859         datachanged : true,
22860         /**
22861          * @event metachange
22862          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22863          * @param {Store} this
22864          * @param {Object} meta The JSON metadata
22865          */
22866         metachange : true,
22867         /**
22868          * @event add
22869          * Fires when Records have been added to the Store
22870          * @param {Store} this
22871          * @param {Roo.data.Record[]} records The array of Records added
22872          * @param {Number} index The index at which the record(s) were added
22873          */
22874         add : true,
22875         /**
22876          * @event remove
22877          * Fires when a Record has been removed from the Store
22878          * @param {Store} this
22879          * @param {Roo.data.Record} record The Record that was removed
22880          * @param {Number} index The index at which the record was removed
22881          */
22882         remove : true,
22883         /**
22884          * @event update
22885          * Fires when a Record has been updated
22886          * @param {Store} this
22887          * @param {Roo.data.Record} record The Record that was updated
22888          * @param {String} operation The update operation being performed.  Value may be one of:
22889          * <pre><code>
22890  Roo.data.Record.EDIT
22891  Roo.data.Record.REJECT
22892  Roo.data.Record.COMMIT
22893          * </code></pre>
22894          */
22895         update : true,
22896         /**
22897          * @event clear
22898          * Fires when the data cache has been cleared.
22899          * @param {Store} this
22900          */
22901         clear : true,
22902         /**
22903          * @event beforeload
22904          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22905          * the load action will be canceled.
22906          * @param {Store} this
22907          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22908          */
22909         beforeload : true,
22910         /**
22911          * @event beforeloadadd
22912          * Fires after a new set of Records has been loaded.
22913          * @param {Store} this
22914          * @param {Roo.data.Record[]} records The Records that were loaded
22915          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22916          */
22917         beforeloadadd : true,
22918         /**
22919          * @event load
22920          * Fires after a new set of Records has been loaded, before they are added to the store.
22921          * @param {Store} this
22922          * @param {Roo.data.Record[]} records The Records that were loaded
22923          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22924          * @params {Object} return from reader
22925          */
22926         load : true,
22927         /**
22928          * @event loadexception
22929          * Fires if an exception occurs in the Proxy during loading.
22930          * Called with the signature of the Proxy's "loadexception" event.
22931          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22932          * 
22933          * @param {Proxy} 
22934          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22935          * @param {Object} load options 
22936          * @param {Object} jsonData from your request (normally this contains the Exception)
22937          */
22938         loadexception : true
22939     });
22940     
22941     if(this.proxy){
22942         this.proxy = Roo.factory(this.proxy, Roo.data);
22943         this.proxy.xmodule = this.xmodule || false;
22944         this.relayEvents(this.proxy,  ["loadexception"]);
22945     }
22946     this.sortToggle = {};
22947     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22948
22949     Roo.data.Store.superclass.constructor.call(this);
22950
22951     if(this.inlineData){
22952         this.loadData(this.inlineData);
22953         delete this.inlineData;
22954     }
22955 };
22956
22957 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22958      /**
22959     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22960     * without a remote query - used by combo/forms at present.
22961     */
22962     
22963     /**
22964     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22965     */
22966     /**
22967     * @cfg {Array} data Inline data to be loaded when the store is initialized.
22968     */
22969     /**
22970     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
22971     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
22972     */
22973     /**
22974     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
22975     * on any HTTP request
22976     */
22977     /**
22978     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
22979     */
22980     /**
22981     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
22982     */
22983     multiSort: false,
22984     /**
22985     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
22986     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
22987     */
22988     remoteSort : false,
22989
22990     /**
22991     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
22992      * loaded or when a record is removed. (defaults to false).
22993     */
22994     pruneModifiedRecords : false,
22995
22996     // private
22997     lastOptions : null,
22998
22999     /**
23000      * Add Records to the Store and fires the add event.
23001      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23002      */
23003     add : function(records){
23004         records = [].concat(records);
23005         for(var i = 0, len = records.length; i < len; i++){
23006             records[i].join(this);
23007         }
23008         var index = this.data.length;
23009         this.data.addAll(records);
23010         this.fireEvent("add", this, records, index);
23011     },
23012
23013     /**
23014      * Remove a Record from the Store and fires the remove event.
23015      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23016      */
23017     remove : function(record){
23018         var index = this.data.indexOf(record);
23019         this.data.removeAt(index);
23020  
23021         if(this.pruneModifiedRecords){
23022             this.modified.remove(record);
23023         }
23024         this.fireEvent("remove", this, record, index);
23025     },
23026
23027     /**
23028      * Remove all Records from the Store and fires the clear event.
23029      */
23030     removeAll : function(){
23031         this.data.clear();
23032         if(this.pruneModifiedRecords){
23033             this.modified = [];
23034         }
23035         this.fireEvent("clear", this);
23036     },
23037
23038     /**
23039      * Inserts Records to the Store at the given index and fires the add event.
23040      * @param {Number} index The start index at which to insert the passed Records.
23041      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23042      */
23043     insert : function(index, records){
23044         records = [].concat(records);
23045         for(var i = 0, len = records.length; i < len; i++){
23046             this.data.insert(index, records[i]);
23047             records[i].join(this);
23048         }
23049         this.fireEvent("add", this, records, index);
23050     },
23051
23052     /**
23053      * Get the index within the cache of the passed Record.
23054      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23055      * @return {Number} The index of the passed Record. Returns -1 if not found.
23056      */
23057     indexOf : function(record){
23058         return this.data.indexOf(record);
23059     },
23060
23061     /**
23062      * Get the index within the cache of the Record with the passed id.
23063      * @param {String} id The id of the Record to find.
23064      * @return {Number} The index of the Record. Returns -1 if not found.
23065      */
23066     indexOfId : function(id){
23067         return this.data.indexOfKey(id);
23068     },
23069
23070     /**
23071      * Get the Record with the specified id.
23072      * @param {String} id The id of the Record to find.
23073      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23074      */
23075     getById : function(id){
23076         return this.data.key(id);
23077     },
23078
23079     /**
23080      * Get the Record at the specified index.
23081      * @param {Number} index The index of the Record to find.
23082      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23083      */
23084     getAt : function(index){
23085         return this.data.itemAt(index);
23086     },
23087
23088     /**
23089      * Returns a range of Records between specified indices.
23090      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23091      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23092      * @return {Roo.data.Record[]} An array of Records
23093      */
23094     getRange : function(start, end){
23095         return this.data.getRange(start, end);
23096     },
23097
23098     // private
23099     storeOptions : function(o){
23100         o = Roo.apply({}, o);
23101         delete o.callback;
23102         delete o.scope;
23103         this.lastOptions = o;
23104     },
23105
23106     /**
23107      * Loads the Record cache from the configured Proxy using the configured Reader.
23108      * <p>
23109      * If using remote paging, then the first load call must specify the <em>start</em>
23110      * and <em>limit</em> properties in the options.params property to establish the initial
23111      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23112      * <p>
23113      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23114      * and this call will return before the new data has been loaded. Perform any post-processing
23115      * in a callback function, or in a "load" event handler.</strong>
23116      * <p>
23117      * @param {Object} options An object containing properties which control loading options:<ul>
23118      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23119      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23120      * passed the following arguments:<ul>
23121      * <li>r : Roo.data.Record[]</li>
23122      * <li>options: Options object from the load call</li>
23123      * <li>success: Boolean success indicator</li></ul></li>
23124      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23125      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23126      * </ul>
23127      */
23128     load : function(options){
23129         options = options || {};
23130         if(this.fireEvent("beforeload", this, options) !== false){
23131             this.storeOptions(options);
23132             var p = Roo.apply(options.params || {}, this.baseParams);
23133             // if meta was not loaded from remote source.. try requesting it.
23134             if (!this.reader.metaFromRemote) {
23135                 p._requestMeta = 1;
23136             }
23137             if(this.sortInfo && this.remoteSort){
23138                 var pn = this.paramNames;
23139                 p[pn["sort"]] = this.sortInfo.field;
23140                 p[pn["dir"]] = this.sortInfo.direction;
23141             }
23142             if (this.multiSort) {
23143                 var pn = this.paramNames;
23144                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23145             }
23146             
23147             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23148         }
23149     },
23150
23151     /**
23152      * Reloads the Record cache from the configured Proxy using the configured Reader and
23153      * the options from the last load operation performed.
23154      * @param {Object} options (optional) An object containing properties which may override the options
23155      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23156      * the most recently used options are reused).
23157      */
23158     reload : function(options){
23159         this.load(Roo.applyIf(options||{}, this.lastOptions));
23160     },
23161
23162     // private
23163     // Called as a callback by the Reader during a load operation.
23164     loadRecords : function(o, options, success){
23165         if(!o || success === false){
23166             if(success !== false){
23167                 this.fireEvent("load", this, [], options, o);
23168             }
23169             if(options.callback){
23170                 options.callback.call(options.scope || this, [], options, false);
23171             }
23172             return;
23173         }
23174         // if data returned failure - throw an exception.
23175         if (o.success === false) {
23176             // show a message if no listener is registered.
23177             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23178                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23179             }
23180             // loadmask wil be hooked into this..
23181             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23182             return;
23183         }
23184         var r = o.records, t = o.totalRecords || r.length;
23185         
23186         this.fireEvent("beforeloadadd", this, r, options, o);
23187         
23188         if(!options || options.add !== true){
23189             if(this.pruneModifiedRecords){
23190                 this.modified = [];
23191             }
23192             for(var i = 0, len = r.length; i < len; i++){
23193                 r[i].join(this);
23194             }
23195             if(this.snapshot){
23196                 this.data = this.snapshot;
23197                 delete this.snapshot;
23198             }
23199             this.data.clear();
23200             this.data.addAll(r);
23201             this.totalLength = t;
23202             this.applySort();
23203             this.fireEvent("datachanged", this);
23204         }else{
23205             this.totalLength = Math.max(t, this.data.length+r.length);
23206             this.add(r);
23207         }
23208         
23209         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23210                 
23211             var e = new Roo.data.Record({});
23212
23213             e.set(this.parent.displayField, this.parent.emptyTitle);
23214             e.set(this.parent.valueField, '');
23215
23216             this.insert(0, e);
23217         }
23218             
23219         this.fireEvent("load", this, r, options, o);
23220         if(options.callback){
23221             options.callback.call(options.scope || this, r, options, true);
23222         }
23223     },
23224
23225
23226     /**
23227      * Loads data from a passed data block. A Reader which understands the format of the data
23228      * must have been configured in the constructor.
23229      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23230      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23231      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23232      */
23233     loadData : function(o, append){
23234         var r = this.reader.readRecords(o);
23235         this.loadRecords(r, {add: append}, true);
23236     },
23237
23238     /**
23239      * Gets the number of cached records.
23240      * <p>
23241      * <em>If using paging, this may not be the total size of the dataset. If the data object
23242      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23243      * the data set size</em>
23244      */
23245     getCount : function(){
23246         return this.data.length || 0;
23247     },
23248
23249     /**
23250      * Gets the total number of records in the dataset as returned by the server.
23251      * <p>
23252      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23253      * the dataset size</em>
23254      */
23255     getTotalCount : function(){
23256         return this.totalLength || 0;
23257     },
23258
23259     /**
23260      * Returns the sort state of the Store as an object with two properties:
23261      * <pre><code>
23262  field {String} The name of the field by which the Records are sorted
23263  direction {String} The sort order, "ASC" or "DESC"
23264      * </code></pre>
23265      */
23266     getSortState : function(){
23267         return this.sortInfo;
23268     },
23269
23270     // private
23271     applySort : function(){
23272         if(this.sortInfo && !this.remoteSort){
23273             var s = this.sortInfo, f = s.field;
23274             var st = this.fields.get(f).sortType;
23275             var fn = function(r1, r2){
23276                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23277                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23278             };
23279             this.data.sort(s.direction, fn);
23280             if(this.snapshot && this.snapshot != this.data){
23281                 this.snapshot.sort(s.direction, fn);
23282             }
23283         }
23284     },
23285
23286     /**
23287      * Sets the default sort column and order to be used by the next load operation.
23288      * @param {String} fieldName The name of the field to sort by.
23289      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23290      */
23291     setDefaultSort : function(field, dir){
23292         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23293     },
23294
23295     /**
23296      * Sort the Records.
23297      * If remote sorting is used, the sort is performed on the server, and the cache is
23298      * reloaded. If local sorting is used, the cache is sorted internally.
23299      * @param {String} fieldName The name of the field to sort by.
23300      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23301      */
23302     sort : function(fieldName, dir){
23303         var f = this.fields.get(fieldName);
23304         if(!dir){
23305             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23306             
23307             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23308                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23309             }else{
23310                 dir = f.sortDir;
23311             }
23312         }
23313         this.sortToggle[f.name] = dir;
23314         this.sortInfo = {field: f.name, direction: dir};
23315         if(!this.remoteSort){
23316             this.applySort();
23317             this.fireEvent("datachanged", this);
23318         }else{
23319             this.load(this.lastOptions);
23320         }
23321     },
23322
23323     /**
23324      * Calls the specified function for each of the Records in the cache.
23325      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23326      * Returning <em>false</em> aborts and exits the iteration.
23327      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23328      */
23329     each : function(fn, scope){
23330         this.data.each(fn, scope);
23331     },
23332
23333     /**
23334      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23335      * (e.g., during paging).
23336      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23337      */
23338     getModifiedRecords : function(){
23339         return this.modified;
23340     },
23341
23342     // private
23343     createFilterFn : function(property, value, anyMatch){
23344         if(!value.exec){ // not a regex
23345             value = String(value);
23346             if(value.length == 0){
23347                 return false;
23348             }
23349             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23350         }
23351         return function(r){
23352             return value.test(r.data[property]);
23353         };
23354     },
23355
23356     /**
23357      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23358      * @param {String} property A field on your records
23359      * @param {Number} start The record index to start at (defaults to 0)
23360      * @param {Number} end The last record index to include (defaults to length - 1)
23361      * @return {Number} The sum
23362      */
23363     sum : function(property, start, end){
23364         var rs = this.data.items, v = 0;
23365         start = start || 0;
23366         end = (end || end === 0) ? end : rs.length-1;
23367
23368         for(var i = start; i <= end; i++){
23369             v += (rs[i].data[property] || 0);
23370         }
23371         return v;
23372     },
23373
23374     /**
23375      * Filter the records by a specified property.
23376      * @param {String} field A field on your records
23377      * @param {String/RegExp} value Either a string that the field
23378      * should start with or a RegExp to test against the field
23379      * @param {Boolean} anyMatch True to match any part not just the beginning
23380      */
23381     filter : function(property, value, anyMatch){
23382         var fn = this.createFilterFn(property, value, anyMatch);
23383         return fn ? this.filterBy(fn) : this.clearFilter();
23384     },
23385
23386     /**
23387      * Filter by a function. The specified function will be called with each
23388      * record in this data source. If the function returns true the record is included,
23389      * otherwise it is filtered.
23390      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23391      * @param {Object} scope (optional) The scope of the function (defaults to this)
23392      */
23393     filterBy : function(fn, scope){
23394         this.snapshot = this.snapshot || this.data;
23395         this.data = this.queryBy(fn, scope||this);
23396         this.fireEvent("datachanged", this);
23397     },
23398
23399     /**
23400      * Query the records by a specified property.
23401      * @param {String} field A field on your records
23402      * @param {String/RegExp} value Either a string that the field
23403      * should start with or a RegExp to test against the field
23404      * @param {Boolean} anyMatch True to match any part not just the beginning
23405      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23406      */
23407     query : function(property, value, anyMatch){
23408         var fn = this.createFilterFn(property, value, anyMatch);
23409         return fn ? this.queryBy(fn) : this.data.clone();
23410     },
23411
23412     /**
23413      * Query by a function. The specified function will be called with each
23414      * record in this data source. If the function returns true the record is included
23415      * in the results.
23416      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23417      * @param {Object} scope (optional) The scope of the function (defaults to this)
23418       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23419      **/
23420     queryBy : function(fn, scope){
23421         var data = this.snapshot || this.data;
23422         return data.filterBy(fn, scope||this);
23423     },
23424
23425     /**
23426      * Collects unique values for a particular dataIndex from this store.
23427      * @param {String} dataIndex The property to collect
23428      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23429      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23430      * @return {Array} An array of the unique values
23431      **/
23432     collect : function(dataIndex, allowNull, bypassFilter){
23433         var d = (bypassFilter === true && this.snapshot) ?
23434                 this.snapshot.items : this.data.items;
23435         var v, sv, r = [], l = {};
23436         for(var i = 0, len = d.length; i < len; i++){
23437             v = d[i].data[dataIndex];
23438             sv = String(v);
23439             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23440                 l[sv] = true;
23441                 r[r.length] = v;
23442             }
23443         }
23444         return r;
23445     },
23446
23447     /**
23448      * Revert to a view of the Record cache with no filtering applied.
23449      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23450      */
23451     clearFilter : function(suppressEvent){
23452         if(this.snapshot && this.snapshot != this.data){
23453             this.data = this.snapshot;
23454             delete this.snapshot;
23455             if(suppressEvent !== true){
23456                 this.fireEvent("datachanged", this);
23457             }
23458         }
23459     },
23460
23461     // private
23462     afterEdit : function(record){
23463         if(this.modified.indexOf(record) == -1){
23464             this.modified.push(record);
23465         }
23466         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23467     },
23468     
23469     // private
23470     afterReject : function(record){
23471         this.modified.remove(record);
23472         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23473     },
23474
23475     // private
23476     afterCommit : function(record){
23477         this.modified.remove(record);
23478         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23479     },
23480
23481     /**
23482      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23483      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23484      */
23485     commitChanges : function(){
23486         var m = this.modified.slice(0);
23487         this.modified = [];
23488         for(var i = 0, len = m.length; i < len; i++){
23489             m[i].commit();
23490         }
23491     },
23492
23493     /**
23494      * Cancel outstanding changes on all changed records.
23495      */
23496     rejectChanges : function(){
23497         var m = this.modified.slice(0);
23498         this.modified = [];
23499         for(var i = 0, len = m.length; i < len; i++){
23500             m[i].reject();
23501         }
23502     },
23503
23504     onMetaChange : function(meta, rtype, o){
23505         this.recordType = rtype;
23506         this.fields = rtype.prototype.fields;
23507         delete this.snapshot;
23508         this.sortInfo = meta.sortInfo || this.sortInfo;
23509         this.modified = [];
23510         this.fireEvent('metachange', this, this.reader.meta);
23511     },
23512     
23513     moveIndex : function(data, type)
23514     {
23515         var index = this.indexOf(data);
23516         
23517         var newIndex = index + type;
23518         
23519         this.remove(data);
23520         
23521         this.insert(newIndex, data);
23522         
23523     }
23524 });/*
23525  * Based on:
23526  * Ext JS Library 1.1.1
23527  * Copyright(c) 2006-2007, Ext JS, LLC.
23528  *
23529  * Originally Released Under LGPL - original licence link has changed is not relivant.
23530  *
23531  * Fork - LGPL
23532  * <script type="text/javascript">
23533  */
23534
23535 /**
23536  * @class Roo.data.SimpleStore
23537  * @extends Roo.data.Store
23538  * Small helper class to make creating Stores from Array data easier.
23539  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23540  * @cfg {Array} fields An array of field definition objects, or field name strings.
23541  * @cfg {Array} data The multi-dimensional array of data
23542  * @constructor
23543  * @param {Object} config
23544  */
23545 Roo.data.SimpleStore = function(config){
23546     Roo.data.SimpleStore.superclass.constructor.call(this, {
23547         isLocal : true,
23548         reader: new Roo.data.ArrayReader({
23549                 id: config.id
23550             },
23551             Roo.data.Record.create(config.fields)
23552         ),
23553         proxy : new Roo.data.MemoryProxy(config.data)
23554     });
23555     this.load();
23556 };
23557 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23558  * Based on:
23559  * Ext JS Library 1.1.1
23560  * Copyright(c) 2006-2007, Ext JS, LLC.
23561  *
23562  * Originally Released Under LGPL - original licence link has changed is not relivant.
23563  *
23564  * Fork - LGPL
23565  * <script type="text/javascript">
23566  */
23567
23568 /**
23569 /**
23570  * @extends Roo.data.Store
23571  * @class Roo.data.JsonStore
23572  * Small helper class to make creating Stores for JSON data easier. <br/>
23573 <pre><code>
23574 var store = new Roo.data.JsonStore({
23575     url: 'get-images.php',
23576     root: 'images',
23577     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23578 });
23579 </code></pre>
23580  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23581  * JsonReader and HttpProxy (unless inline data is provided).</b>
23582  * @cfg {Array} fields An array of field definition objects, or field name strings.
23583  * @constructor
23584  * @param {Object} config
23585  */
23586 Roo.data.JsonStore = function(c){
23587     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23588         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23589         reader: new Roo.data.JsonReader(c, c.fields)
23590     }));
23591 };
23592 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23593  * Based on:
23594  * Ext JS Library 1.1.1
23595  * Copyright(c) 2006-2007, Ext JS, LLC.
23596  *
23597  * Originally Released Under LGPL - original licence link has changed is not relivant.
23598  *
23599  * Fork - LGPL
23600  * <script type="text/javascript">
23601  */
23602
23603  
23604 Roo.data.Field = function(config){
23605     if(typeof config == "string"){
23606         config = {name: config};
23607     }
23608     Roo.apply(this, config);
23609     
23610     if(!this.type){
23611         this.type = "auto";
23612     }
23613     
23614     var st = Roo.data.SortTypes;
23615     // named sortTypes are supported, here we look them up
23616     if(typeof this.sortType == "string"){
23617         this.sortType = st[this.sortType];
23618     }
23619     
23620     // set default sortType for strings and dates
23621     if(!this.sortType){
23622         switch(this.type){
23623             case "string":
23624                 this.sortType = st.asUCString;
23625                 break;
23626             case "date":
23627                 this.sortType = st.asDate;
23628                 break;
23629             default:
23630                 this.sortType = st.none;
23631         }
23632     }
23633
23634     // define once
23635     var stripRe = /[\$,%]/g;
23636
23637     // prebuilt conversion function for this field, instead of
23638     // switching every time we're reading a value
23639     if(!this.convert){
23640         var cv, dateFormat = this.dateFormat;
23641         switch(this.type){
23642             case "":
23643             case "auto":
23644             case undefined:
23645                 cv = function(v){ return v; };
23646                 break;
23647             case "string":
23648                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23649                 break;
23650             case "int":
23651                 cv = function(v){
23652                     return v !== undefined && v !== null && v !== '' ?
23653                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23654                     };
23655                 break;
23656             case "float":
23657                 cv = function(v){
23658                     return v !== undefined && v !== null && v !== '' ?
23659                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23660                     };
23661                 break;
23662             case "bool":
23663             case "boolean":
23664                 cv = function(v){ return v === true || v === "true" || v == 1; };
23665                 break;
23666             case "date":
23667                 cv = function(v){
23668                     if(!v){
23669                         return '';
23670                     }
23671                     if(v instanceof Date){
23672                         return v;
23673                     }
23674                     if(dateFormat){
23675                         if(dateFormat == "timestamp"){
23676                             return new Date(v*1000);
23677                         }
23678                         return Date.parseDate(v, dateFormat);
23679                     }
23680                     var parsed = Date.parse(v);
23681                     return parsed ? new Date(parsed) : null;
23682                 };
23683              break;
23684             
23685         }
23686         this.convert = cv;
23687     }
23688 };
23689
23690 Roo.data.Field.prototype = {
23691     dateFormat: null,
23692     defaultValue: "",
23693     mapping: null,
23694     sortType : null,
23695     sortDir : "ASC"
23696 };/*
23697  * Based on:
23698  * Ext JS Library 1.1.1
23699  * Copyright(c) 2006-2007, Ext JS, LLC.
23700  *
23701  * Originally Released Under LGPL - original licence link has changed is not relivant.
23702  *
23703  * Fork - LGPL
23704  * <script type="text/javascript">
23705  */
23706  
23707 // Base class for reading structured data from a data source.  This class is intended to be
23708 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23709
23710 /**
23711  * @class Roo.data.DataReader
23712  * Base class for reading structured data from a data source.  This class is intended to be
23713  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23714  */
23715
23716 Roo.data.DataReader = function(meta, recordType){
23717     
23718     this.meta = meta;
23719     
23720     this.recordType = recordType instanceof Array ? 
23721         Roo.data.Record.create(recordType) : recordType;
23722 };
23723
23724 Roo.data.DataReader.prototype = {
23725      /**
23726      * Create an empty record
23727      * @param {Object} data (optional) - overlay some values
23728      * @return {Roo.data.Record} record created.
23729      */
23730     newRow :  function(d) {
23731         var da =  {};
23732         this.recordType.prototype.fields.each(function(c) {
23733             switch( c.type) {
23734                 case 'int' : da[c.name] = 0; break;
23735                 case 'date' : da[c.name] = new Date(); break;
23736                 case 'float' : da[c.name] = 0.0; break;
23737                 case 'boolean' : da[c.name] = false; break;
23738                 default : da[c.name] = ""; break;
23739             }
23740             
23741         });
23742         return new this.recordType(Roo.apply(da, d));
23743     }
23744     
23745 };/*
23746  * Based on:
23747  * Ext JS Library 1.1.1
23748  * Copyright(c) 2006-2007, Ext JS, LLC.
23749  *
23750  * Originally Released Under LGPL - original licence link has changed is not relivant.
23751  *
23752  * Fork - LGPL
23753  * <script type="text/javascript">
23754  */
23755
23756 /**
23757  * @class Roo.data.DataProxy
23758  * @extends Roo.data.Observable
23759  * This class is an abstract base class for implementations which provide retrieval of
23760  * unformatted data objects.<br>
23761  * <p>
23762  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23763  * (of the appropriate type which knows how to parse the data object) to provide a block of
23764  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23765  * <p>
23766  * Custom implementations must implement the load method as described in
23767  * {@link Roo.data.HttpProxy#load}.
23768  */
23769 Roo.data.DataProxy = function(){
23770     this.addEvents({
23771         /**
23772          * @event beforeload
23773          * Fires before a network request is made to retrieve a data object.
23774          * @param {Object} This DataProxy object.
23775          * @param {Object} params The params parameter to the load function.
23776          */
23777         beforeload : true,
23778         /**
23779          * @event load
23780          * Fires before the load method's callback is called.
23781          * @param {Object} This DataProxy object.
23782          * @param {Object} o The data object.
23783          * @param {Object} arg The callback argument object passed to the load function.
23784          */
23785         load : true,
23786         /**
23787          * @event loadexception
23788          * Fires if an Exception occurs during data retrieval.
23789          * @param {Object} This DataProxy object.
23790          * @param {Object} o The data object.
23791          * @param {Object} arg The callback argument object passed to the load function.
23792          * @param {Object} e The Exception.
23793          */
23794         loadexception : true
23795     });
23796     Roo.data.DataProxy.superclass.constructor.call(this);
23797 };
23798
23799 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23800
23801     /**
23802      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23803      */
23804 /*
23805  * Based on:
23806  * Ext JS Library 1.1.1
23807  * Copyright(c) 2006-2007, Ext JS, LLC.
23808  *
23809  * Originally Released Under LGPL - original licence link has changed is not relivant.
23810  *
23811  * Fork - LGPL
23812  * <script type="text/javascript">
23813  */
23814 /**
23815  * @class Roo.data.MemoryProxy
23816  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23817  * to the Reader when its load method is called.
23818  * @constructor
23819  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23820  */
23821 Roo.data.MemoryProxy = function(data){
23822     if (data.data) {
23823         data = data.data;
23824     }
23825     Roo.data.MemoryProxy.superclass.constructor.call(this);
23826     this.data = data;
23827 };
23828
23829 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23830     
23831     /**
23832      * Load data from the requested source (in this case an in-memory
23833      * data object passed to the constructor), read the data object into
23834      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23835      * process that block using the passed callback.
23836      * @param {Object} params This parameter is not used by the MemoryProxy class.
23837      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23838      * object into a block of Roo.data.Records.
23839      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23840      * The function must be passed <ul>
23841      * <li>The Record block object</li>
23842      * <li>The "arg" argument from the load function</li>
23843      * <li>A boolean success indicator</li>
23844      * </ul>
23845      * @param {Object} scope The scope in which to call the callback
23846      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23847      */
23848     load : function(params, reader, callback, scope, arg){
23849         params = params || {};
23850         var result;
23851         try {
23852             result = reader.readRecords(params.data ? params.data :this.data);
23853         }catch(e){
23854             this.fireEvent("loadexception", this, arg, null, e);
23855             callback.call(scope, null, arg, false);
23856             return;
23857         }
23858         callback.call(scope, result, arg, true);
23859     },
23860     
23861     // private
23862     update : function(params, records){
23863         
23864     }
23865 });/*
23866  * Based on:
23867  * Ext JS Library 1.1.1
23868  * Copyright(c) 2006-2007, Ext JS, LLC.
23869  *
23870  * Originally Released Under LGPL - original licence link has changed is not relivant.
23871  *
23872  * Fork - LGPL
23873  * <script type="text/javascript">
23874  */
23875 /**
23876  * @class Roo.data.HttpProxy
23877  * @extends Roo.data.DataProxy
23878  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23879  * configured to reference a certain URL.<br><br>
23880  * <p>
23881  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23882  * from which the running page was served.<br><br>
23883  * <p>
23884  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23885  * <p>
23886  * Be aware that to enable the browser to parse an XML document, the server must set
23887  * the Content-Type header in the HTTP response to "text/xml".
23888  * @constructor
23889  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23890  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23891  * will be used to make the request.
23892  */
23893 Roo.data.HttpProxy = function(conn){
23894     Roo.data.HttpProxy.superclass.constructor.call(this);
23895     // is conn a conn config or a real conn?
23896     this.conn = conn;
23897     this.useAjax = !conn || !conn.events;
23898   
23899 };
23900
23901 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23902     // thse are take from connection...
23903     
23904     /**
23905      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23906      */
23907     /**
23908      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23909      * extra parameters to each request made by this object. (defaults to undefined)
23910      */
23911     /**
23912      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23913      *  to each request made by this object. (defaults to undefined)
23914      */
23915     /**
23916      * @cfg {String} method (Optional) The default HTTP method to be used for requests. (defaults to undefined; if not set but parms are present will use POST, otherwise GET)
23917      */
23918     /**
23919      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23920      */
23921      /**
23922      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23923      * @type Boolean
23924      */
23925   
23926
23927     /**
23928      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23929      * @type Boolean
23930      */
23931     /**
23932      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23933      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23934      * a finer-grained basis than the DataProxy events.
23935      */
23936     getConnection : function(){
23937         return this.useAjax ? Roo.Ajax : this.conn;
23938     },
23939
23940     /**
23941      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23942      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23943      * process that block using the passed callback.
23944      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23945      * for the request to the remote server.
23946      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23947      * object into a block of Roo.data.Records.
23948      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23949      * The function must be passed <ul>
23950      * <li>The Record block object</li>
23951      * <li>The "arg" argument from the load function</li>
23952      * <li>A boolean success indicator</li>
23953      * </ul>
23954      * @param {Object} scope The scope in which to call the callback
23955      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23956      */
23957     load : function(params, reader, callback, scope, arg){
23958         if(this.fireEvent("beforeload", this, params) !== false){
23959             var  o = {
23960                 params : params || {},
23961                 request: {
23962                     callback : callback,
23963                     scope : scope,
23964                     arg : arg
23965                 },
23966                 reader: reader,
23967                 callback : this.loadResponse,
23968                 scope: this
23969             };
23970             if(this.useAjax){
23971                 Roo.applyIf(o, this.conn);
23972                 if(this.activeRequest){
23973                     Roo.Ajax.abort(this.activeRequest);
23974                 }
23975                 this.activeRequest = Roo.Ajax.request(o);
23976             }else{
23977                 this.conn.request(o);
23978             }
23979         }else{
23980             callback.call(scope||this, null, arg, false);
23981         }
23982     },
23983
23984     // private
23985     loadResponse : function(o, success, response){
23986         delete this.activeRequest;
23987         if(!success){
23988             this.fireEvent("loadexception", this, o, response);
23989             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23990             return;
23991         }
23992         var result;
23993         try {
23994             result = o.reader.read(response);
23995         }catch(e){
23996             this.fireEvent("loadexception", this, o, response, e);
23997             o.request.callback.call(o.request.scope, null, o.request.arg, false);
23998             return;
23999         }
24000         
24001         this.fireEvent("load", this, o, o.request.arg);
24002         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24003     },
24004
24005     // private
24006     update : function(dataSet){
24007
24008     },
24009
24010     // private
24011     updateResponse : function(dataSet){
24012
24013     }
24014 });/*
24015  * Based on:
24016  * Ext JS Library 1.1.1
24017  * Copyright(c) 2006-2007, Ext JS, LLC.
24018  *
24019  * Originally Released Under LGPL - original licence link has changed is not relivant.
24020  *
24021  * Fork - LGPL
24022  * <script type="text/javascript">
24023  */
24024
24025 /**
24026  * @class Roo.data.ScriptTagProxy
24027  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24028  * other than the originating domain of the running page.<br><br>
24029  * <p>
24030  * <em>Note that if you are retrieving data from a page that is in a domain that is NOT the same as the originating domain
24031  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24032  * <p>
24033  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24034  * source code that is used as the source inside a &lt;script> tag.<br><br>
24035  * <p>
24036  * In order for the browser to process the returned data, the server must wrap the data object
24037  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24038  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24039  * depending on whether the callback name was passed:
24040  * <p>
24041  * <pre><code>
24042 boolean scriptTag = false;
24043 String cb = request.getParameter("callback");
24044 if (cb != null) {
24045     scriptTag = true;
24046     response.setContentType("text/javascript");
24047 } else {
24048     response.setContentType("application/x-json");
24049 }
24050 Writer out = response.getWriter();
24051 if (scriptTag) {
24052     out.write(cb + "(");
24053 }
24054 out.print(dataBlock.toJsonString());
24055 if (scriptTag) {
24056     out.write(");");
24057 }
24058 </pre></code>
24059  *
24060  * @constructor
24061  * @param {Object} config A configuration object.
24062  */
24063 Roo.data.ScriptTagProxy = function(config){
24064     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24065     Roo.apply(this, config);
24066     this.head = document.getElementsByTagName("head")[0];
24067 };
24068
24069 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24070
24071 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24072     /**
24073      * @cfg {String} url The URL from which to request the data object.
24074      */
24075     /**
24076      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24077      */
24078     timeout : 30000,
24079     /**
24080      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24081      * the server the name of the callback function set up by the load call to process the returned data object.
24082      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24083      * javascript output which calls this named function passing the data object as its only parameter.
24084      */
24085     callbackParam : "callback",
24086     /**
24087      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24088      * name to the request.
24089      */
24090     nocache : true,
24091
24092     /**
24093      * Load data from the configured URL, read the data object into
24094      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24095      * process that block using the passed callback.
24096      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24097      * for the request to the remote server.
24098      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24099      * object into a block of Roo.data.Records.
24100      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24101      * The function must be passed <ul>
24102      * <li>The Record block object</li>
24103      * <li>The "arg" argument from the load function</li>
24104      * <li>A boolean success indicator</li>
24105      * </ul>
24106      * @param {Object} scope The scope in which to call the callback
24107      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24108      */
24109     load : function(params, reader, callback, scope, arg){
24110         if(this.fireEvent("beforeload", this, params) !== false){
24111
24112             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24113
24114             var url = this.url;
24115             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24116             if(this.nocache){
24117                 url += "&_dc=" + (new Date().getTime());
24118             }
24119             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24120             var trans = {
24121                 id : transId,
24122                 cb : "stcCallback"+transId,
24123                 scriptId : "stcScript"+transId,
24124                 params : params,
24125                 arg : arg,
24126                 url : url,
24127                 callback : callback,
24128                 scope : scope,
24129                 reader : reader
24130             };
24131             var conn = this;
24132
24133             window[trans.cb] = function(o){
24134                 conn.handleResponse(o, trans);
24135             };
24136
24137             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24138
24139             if(this.autoAbort !== false){
24140                 this.abort();
24141             }
24142
24143             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24144
24145             var script = document.createElement("script");
24146             script.setAttribute("src", url);
24147             script.setAttribute("type", "text/javascript");
24148             script.setAttribute("id", trans.scriptId);
24149             this.head.appendChild(script);
24150
24151             this.trans = trans;
24152         }else{
24153             callback.call(scope||this, null, arg, false);
24154         }
24155     },
24156
24157     // private
24158     isLoading : function(){
24159         return this.trans ? true : false;
24160     },
24161
24162     /**
24163      * Abort the current server request.
24164      */
24165     abort : function(){
24166         if(this.isLoading()){
24167             this.destroyTrans(this.trans);
24168         }
24169     },
24170
24171     // private
24172     destroyTrans : function(trans, isLoaded){
24173         this.head.removeChild(document.getElementById(trans.scriptId));
24174         clearTimeout(trans.timeoutId);
24175         if(isLoaded){
24176             window[trans.cb] = undefined;
24177             try{
24178                 delete window[trans.cb];
24179             }catch(e){}
24180         }else{
24181             // if hasn't been loaded, wait for load to remove it to prevent script error
24182             window[trans.cb] = function(){
24183                 window[trans.cb] = undefined;
24184                 try{
24185                     delete window[trans.cb];
24186                 }catch(e){}
24187             };
24188         }
24189     },
24190
24191     // private
24192     handleResponse : function(o, trans){
24193         this.trans = false;
24194         this.destroyTrans(trans, true);
24195         var result;
24196         try {
24197             result = trans.reader.readRecords(o);
24198         }catch(e){
24199             this.fireEvent("loadexception", this, o, trans.arg, e);
24200             trans.callback.call(trans.scope||window, null, trans.arg, false);
24201             return;
24202         }
24203         this.fireEvent("load", this, o, trans.arg);
24204         trans.callback.call(trans.scope||window, result, trans.arg, true);
24205     },
24206
24207     // private
24208     handleFailure : function(trans){
24209         this.trans = false;
24210         this.destroyTrans(trans, false);
24211         this.fireEvent("loadexception", this, null, trans.arg);
24212         trans.callback.call(trans.scope||window, null, trans.arg, false);
24213     }
24214 });/*
24215  * Based on:
24216  * Ext JS Library 1.1.1
24217  * Copyright(c) 2006-2007, Ext JS, LLC.
24218  *
24219  * Originally Released Under LGPL - original licence link has changed is not relivant.
24220  *
24221  * Fork - LGPL
24222  * <script type="text/javascript">
24223  */
24224
24225 /**
24226  * @class Roo.data.JsonReader
24227  * @extends Roo.data.DataReader
24228  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24229  * based on mappings in a provided Roo.data.Record constructor.
24230  * 
24231  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24232  * in the reply previously. 
24233  * 
24234  * <p>
24235  * Example code:
24236  * <pre><code>
24237 var RecordDef = Roo.data.Record.create([
24238     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24239     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24240 ]);
24241 var myReader = new Roo.data.JsonReader({
24242     totalProperty: "results",    // The property which contains the total dataset size (optional)
24243     root: "rows",                // The property which contains an Array of row objects
24244     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24245 }, RecordDef);
24246 </code></pre>
24247  * <p>
24248  * This would consume a JSON file like this:
24249  * <pre><code>
24250 { 'results': 2, 'rows': [
24251     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24252     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24253 }
24254 </code></pre>
24255  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24256  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24257  * paged from the remote server.
24258  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24259  * @cfg {String} root name of the property which contains the Array of row objects.
24260  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24261  * @cfg {Array} fields Array of field definition objects
24262  * @constructor
24263  * Create a new JsonReader
24264  * @param {Object} meta Metadata configuration options
24265  * @param {Object} recordType Either an Array of field definition objects,
24266  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24267  */
24268 Roo.data.JsonReader = function(meta, recordType){
24269     
24270     meta = meta || {};
24271     // set some defaults:
24272     Roo.applyIf(meta, {
24273         totalProperty: 'total',
24274         successProperty : 'success',
24275         root : 'data',
24276         id : 'id'
24277     });
24278     
24279     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24280 };
24281 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24282     
24283     /**
24284      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24285      * Used by Store query builder to append _requestMeta to params.
24286      * 
24287      */
24288     metaFromRemote : false,
24289     /**
24290      * This method is only used by a DataProxy which has retrieved data from a remote server.
24291      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24292      * @return {Object} data A data block which is used by an Roo.data.Store object as
24293      * a cache of Roo.data.Records.
24294      */
24295     read : function(response){
24296         var json = response.responseText;
24297        
24298         var o = /* eval:var:o */ eval("("+json+")");
24299         if(!o) {
24300             throw {message: "JsonReader.read: Json object not found"};
24301         }
24302         
24303         if(o.metaData){
24304             
24305             delete this.ef;
24306             this.metaFromRemote = true;
24307             this.meta = o.metaData;
24308             this.recordType = Roo.data.Record.create(o.metaData.fields);
24309             this.onMetaChange(this.meta, this.recordType, o);
24310         }
24311         return this.readRecords(o);
24312     },
24313
24314     // private function a store will implement
24315     onMetaChange : function(meta, recordType, o){
24316
24317     },
24318
24319     /**
24320          * @ignore
24321          */
24322     simpleAccess: function(obj, subsc) {
24323         return obj[subsc];
24324     },
24325
24326         /**
24327          * @ignore
24328          */
24329     getJsonAccessor: function(){
24330         var re = /[\[\.]/;
24331         return function(expr) {
24332             try {
24333                 return(re.test(expr))
24334                     ? new Function("obj", "return obj." + expr)
24335                     : function(obj){
24336                         return obj[expr];
24337                     };
24338             } catch(e){}
24339             return Roo.emptyFn;
24340         };
24341     }(),
24342
24343     /**
24344      * Create a data block containing Roo.data.Records from an XML document.
24345      * @param {Object} o An object which contains an Array of row objects in the property specified
24346      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24347      * which contains the total size of the dataset.
24348      * @return {Object} data A data block which is used by an Roo.data.Store object as
24349      * a cache of Roo.data.Records.
24350      */
24351     readRecords : function(o){
24352         /**
24353          * After any data loads, the raw JSON data is available for further custom processing.
24354          * @type Object
24355          */
24356         this.o = o;
24357         var s = this.meta, Record = this.recordType,
24358             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24359
24360 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24361         if (!this.ef) {
24362             if(s.totalProperty) {
24363                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24364                 }
24365                 if(s.successProperty) {
24366                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24367                 }
24368                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24369                 if (s.id) {
24370                         var g = this.getJsonAccessor(s.id);
24371                         this.getId = function(rec) {
24372                                 var r = g(rec);  
24373                                 return (r === undefined || r === "") ? null : r;
24374                         };
24375                 } else {
24376                         this.getId = function(){return null;};
24377                 }
24378             this.ef = [];
24379             for(var jj = 0; jj < fl; jj++){
24380                 f = fi[jj];
24381                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24382                 this.ef[jj] = this.getJsonAccessor(map);
24383             }
24384         }
24385
24386         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24387         if(s.totalProperty){
24388             var vt = parseInt(this.getTotal(o), 10);
24389             if(!isNaN(vt)){
24390                 totalRecords = vt;
24391             }
24392         }
24393         if(s.successProperty){
24394             var vs = this.getSuccess(o);
24395             if(vs === false || vs === 'false'){
24396                 success = false;
24397             }
24398         }
24399         var records = [];
24400         for(var i = 0; i < c; i++){
24401                 var n = root[i];
24402             var values = {};
24403             var id = this.getId(n);
24404             for(var j = 0; j < fl; j++){
24405                 f = fi[j];
24406             var v = this.ef[j](n);
24407             if (!f.convert) {
24408                 Roo.log('missing convert for ' + f.name);
24409                 Roo.log(f);
24410                 continue;
24411             }
24412             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24413             }
24414             var record = new Record(values, id);
24415             record.json = n;
24416             records[i] = record;
24417         }
24418         return {
24419             raw : o,
24420             success : success,
24421             records : records,
24422             totalRecords : totalRecords
24423         };
24424     }
24425 });/*
24426  * Based on:
24427  * Ext JS Library 1.1.1
24428  * Copyright(c) 2006-2007, Ext JS, LLC.
24429  *
24430  * Originally Released Under LGPL - original licence link has changed is not relivant.
24431  *
24432  * Fork - LGPL
24433  * <script type="text/javascript">
24434  */
24435
24436 /**
24437  * @class Roo.data.XmlReader
24438  * @extends Roo.data.DataReader
24439  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24440  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24441  * <p>
24442  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24443  * header in the HTTP response must be set to "text/xml".</em>
24444  * <p>
24445  * Example code:
24446  * <pre><code>
24447 var RecordDef = Roo.data.Record.create([
24448    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24449    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24450 ]);
24451 var myReader = new Roo.data.XmlReader({
24452    totalRecords: "results", // The element which contains the total dataset size (optional)
24453    record: "row",           // The repeated element which contains row information
24454    id: "id"                 // The element within the row that provides an ID for the record (optional)
24455 }, RecordDef);
24456 </code></pre>
24457  * <p>
24458  * This would consume an XML file like this:
24459  * <pre><code>
24460 &lt;?xml?>
24461 &lt;dataset>
24462  &lt;results>2&lt;/results>
24463  &lt;row>
24464    &lt;id>1&lt;/id>
24465    &lt;name>Bill&lt;/name>
24466    &lt;occupation>Gardener&lt;/occupation>
24467  &lt;/row>
24468  &lt;row>
24469    &lt;id>2&lt;/id>
24470    &lt;name>Ben&lt;/name>
24471    &lt;occupation>Horticulturalist&lt;/occupation>
24472  &lt;/row>
24473 &lt;/dataset>
24474 </code></pre>
24475  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24476  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24477  * paged from the remote server.
24478  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24479  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24480  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24481  * a record identifier value.
24482  * @constructor
24483  * Create a new XmlReader
24484  * @param {Object} meta Metadata configuration options
24485  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24486  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24487  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24488  */
24489 Roo.data.XmlReader = function(meta, recordType){
24490     meta = meta || {};
24491     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24492 };
24493 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24494     /**
24495      * This method is only used by a DataProxy which has retrieved data from a remote server.
24496          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24497          * to contain a method called 'responseXML' that returns an XML document object.
24498      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24499      * a cache of Roo.data.Records.
24500      */
24501     read : function(response){
24502         var doc = response.responseXML;
24503         if(!doc) {
24504             throw {message: "XmlReader.read: XML Document not available"};
24505         }
24506         return this.readRecords(doc);
24507     },
24508
24509     /**
24510      * Create a data block containing Roo.data.Records from an XML document.
24511          * @param {Object} doc A parsed XML document.
24512      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24513      * a cache of Roo.data.Records.
24514      */
24515     readRecords : function(doc){
24516         /**
24517          * After any data loads/reads, the raw XML Document is available for further custom processing.
24518          * @type XMLDocument
24519          */
24520         this.xmlData = doc;
24521         var root = doc.documentElement || doc;
24522         var q = Roo.DomQuery;
24523         var recordType = this.recordType, fields = recordType.prototype.fields;
24524         var sid = this.meta.id;
24525         var totalRecords = 0, success = true;
24526         if(this.meta.totalRecords){
24527             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24528         }
24529         
24530         if(this.meta.success){
24531             var sv = q.selectValue(this.meta.success, root, true);
24532             success = sv !== false && sv !== 'false';
24533         }
24534         var records = [];
24535         var ns = q.select(this.meta.record, root);
24536         for(var i = 0, len = ns.length; i < len; i++) {
24537                 var n = ns[i];
24538                 var values = {};
24539                 var id = sid ? q.selectValue(sid, n) : undefined;
24540                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24541                     var f = fields.items[j];
24542                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24543                     v = f.convert(v);
24544                     values[f.name] = v;
24545                 }
24546                 var record = new recordType(values, id);
24547                 record.node = n;
24548                 records[records.length] = record;
24549             }
24550
24551             return {
24552                 success : success,
24553                 records : records,
24554                 totalRecords : totalRecords || records.length
24555             };
24556     }
24557 });/*
24558  * Based on:
24559  * Ext JS Library 1.1.1
24560  * Copyright(c) 2006-2007, Ext JS, LLC.
24561  *
24562  * Originally Released Under LGPL - original licence link has changed is not relivant.
24563  *
24564  * Fork - LGPL
24565  * <script type="text/javascript">
24566  */
24567
24568 /**
24569  * @class Roo.data.ArrayReader
24570  * @extends Roo.data.DataReader
24571  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24572  * Each element of that Array represents a row of data fields. The
24573  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24574  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24575  * <p>
24576  * Example code:.
24577  * <pre><code>
24578 var RecordDef = Roo.data.Record.create([
24579     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24580     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24581 ]);
24582 var myReader = new Roo.data.ArrayReader({
24583     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24584 }, RecordDef);
24585 </code></pre>
24586  * <p>
24587  * This would consume an Array like this:
24588  * <pre><code>
24589 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24590   </code></pre>
24591  
24592  * @constructor
24593  * Create a new JsonReader
24594  * @param {Object} meta Metadata configuration options.
24595  * @param {Object|Array} recordType Either an Array of field definition objects
24596  * 
24597  * @cfg {Array} fields Array of field definition objects
24598  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24599  * as specified to {@link Roo.data.Record#create},
24600  * or an {@link Roo.data.Record} object
24601  *
24602  * 
24603  * created using {@link Roo.data.Record#create}.
24604  */
24605 Roo.data.ArrayReader = function(meta, recordType){
24606     
24607      
24608     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24609 };
24610
24611 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24612     /**
24613      * Create a data block containing Roo.data.Records from an XML document.
24614      * @param {Object} o An Array of row objects which represents the dataset.
24615      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24616      * a cache of Roo.data.Records.
24617      */
24618     readRecords : function(o){
24619         var sid = this.meta ? this.meta.id : null;
24620         var recordType = this.recordType, fields = recordType.prototype.fields;
24621         var records = [];
24622         var root = o;
24623             for(var i = 0; i < root.length; i++){
24624                     var n = root[i];
24625                 var values = {};
24626                 var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24627                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24628                 var f = fields.items[j];
24629                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24630                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24631                 v = f.convert(v);
24632                 values[f.name] = v;
24633             }
24634                 var record = new recordType(values, id);
24635                 record.json = n;
24636                 records[records.length] = record;
24637             }
24638             return {
24639                 records : records,
24640                 totalRecords : records.length
24641             };
24642     }
24643 });/*
24644  * Based on:
24645  * Ext JS Library 1.1.1
24646  * Copyright(c) 2006-2007, Ext JS, LLC.
24647  *
24648  * Originally Released Under LGPL - original licence link has changed is not relivant.
24649  *
24650  * Fork - LGPL
24651  * <script type="text/javascript">
24652  */
24653
24654
24655 /**
24656  * @class Roo.data.Tree
24657  * @extends Roo.util.Observable
24658  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24659  * in the tree have most standard DOM functionality.
24660  * @constructor
24661  * @param {Node} root (optional) The root node
24662  */
24663 Roo.data.Tree = function(root){
24664    this.nodeHash = {};
24665    /**
24666     * The root node for this tree
24667     * @type Node
24668     */
24669    this.root = null;
24670    if(root){
24671        this.setRootNode(root);
24672    }
24673    this.addEvents({
24674        /**
24675         * @event append
24676         * Fires when a new child node is appended to a node in this tree.
24677         * @param {Tree} tree The owner tree
24678         * @param {Node} parent The parent node
24679         * @param {Node} node The newly appended node
24680         * @param {Number} index The index of the newly appended node
24681         */
24682        "append" : true,
24683        /**
24684         * @event remove
24685         * Fires when a child node is removed from a node in this tree.
24686         * @param {Tree} tree The owner tree
24687         * @param {Node} parent The parent node
24688         * @param {Node} node The child node removed
24689         */
24690        "remove" : true,
24691        /**
24692         * @event move
24693         * Fires when a node is moved to a new location in the tree
24694         * @param {Tree} tree The owner tree
24695         * @param {Node} node The node moved
24696         * @param {Node} oldParent The old parent of this node
24697         * @param {Node} newParent The new parent of this node
24698         * @param {Number} index The index it was moved to
24699         */
24700        "move" : true,
24701        /**
24702         * @event insert
24703         * Fires when a new child node is inserted in a node in this tree.
24704         * @param {Tree} tree The owner tree
24705         * @param {Node} parent The parent node
24706         * @param {Node} node The child node inserted
24707         * @param {Node} refNode The child node the node was inserted before
24708         */
24709        "insert" : true,
24710        /**
24711         * @event beforeappend
24712         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24713         * @param {Tree} tree The owner tree
24714         * @param {Node} parent The parent node
24715         * @param {Node} node The child node to be appended
24716         */
24717        "beforeappend" : true,
24718        /**
24719         * @event beforeremove
24720         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24721         * @param {Tree} tree The owner tree
24722         * @param {Node} parent The parent node
24723         * @param {Node} node The child node to be removed
24724         */
24725        "beforeremove" : true,
24726        /**
24727         * @event beforemove
24728         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24729         * @param {Tree} tree The owner tree
24730         * @param {Node} node The node being moved
24731         * @param {Node} oldParent The parent of the node
24732         * @param {Node} newParent The new parent the node is moving to
24733         * @param {Number} index The index it is being moved to
24734         */
24735        "beforemove" : true,
24736        /**
24737         * @event beforeinsert
24738         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24739         * @param {Tree} tree The owner tree
24740         * @param {Node} parent The parent node
24741         * @param {Node} node The child node to be inserted
24742         * @param {Node} refNode The child node the node is being inserted before
24743         */
24744        "beforeinsert" : true
24745    });
24746
24747     Roo.data.Tree.superclass.constructor.call(this);
24748 };
24749
24750 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24751     pathSeparator: "/",
24752
24753     proxyNodeEvent : function(){
24754         return this.fireEvent.apply(this, arguments);
24755     },
24756
24757     /**
24758      * Returns the root node for this tree.
24759      * @return {Node}
24760      */
24761     getRootNode : function(){
24762         return this.root;
24763     },
24764
24765     /**
24766      * Sets the root node for this tree.
24767      * @param {Node} node
24768      * @return {Node}
24769      */
24770     setRootNode : function(node){
24771         this.root = node;
24772         node.ownerTree = this;
24773         node.isRoot = true;
24774         this.registerNode(node);
24775         return node;
24776     },
24777
24778     /**
24779      * Gets a node in this tree by its id.
24780      * @param {String} id
24781      * @return {Node}
24782      */
24783     getNodeById : function(id){
24784         return this.nodeHash[id];
24785     },
24786
24787     registerNode : function(node){
24788         this.nodeHash[node.id] = node;
24789     },
24790
24791     unregisterNode : function(node){
24792         delete this.nodeHash[node.id];
24793     },
24794
24795     toString : function(){
24796         return "[Tree"+(this.id?" "+this.id:"")+"]";
24797     }
24798 });
24799
24800 /**
24801  * @class Roo.data.Node
24802  * @extends Roo.util.Observable
24803  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24804  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24805  * @constructor
24806  * @param {Object} attributes The attributes/config for the node
24807  */
24808 Roo.data.Node = function(attributes){
24809     /**
24810      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24811      * @type {Object}
24812      */
24813     this.attributes = attributes || {};
24814     this.leaf = this.attributes.leaf;
24815     /**
24816      * The node id. @type String
24817      */
24818     this.id = this.attributes.id;
24819     if(!this.id){
24820         this.id = Roo.id(null, "ynode-");
24821         this.attributes.id = this.id;
24822     }
24823      
24824     
24825     /**
24826      * All child nodes of this node. @type Array
24827      */
24828     this.childNodes = [];
24829     if(!this.childNodes.indexOf){ // indexOf is a must
24830         this.childNodes.indexOf = function(o){
24831             for(var i = 0, len = this.length; i < len; i++){
24832                 if(this[i] == o) {
24833                     return i;
24834                 }
24835             }
24836             return -1;
24837         };
24838     }
24839     /**
24840      * The parent node for this node. @type Node
24841      */
24842     this.parentNode = null;
24843     /**
24844      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24845      */
24846     this.firstChild = null;
24847     /**
24848      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24849      */
24850     this.lastChild = null;
24851     /**
24852      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24853      */
24854     this.previousSibling = null;
24855     /**
24856      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24857      */
24858     this.nextSibling = null;
24859
24860     this.addEvents({
24861        /**
24862         * @event append
24863         * Fires when a new child node is appended
24864         * @param {Tree} tree The owner tree
24865         * @param {Node} this This node
24866         * @param {Node} node The newly appended node
24867         * @param {Number} index The index of the newly appended node
24868         */
24869        "append" : true,
24870        /**
24871         * @event remove
24872         * Fires when a child node is removed
24873         * @param {Tree} tree The owner tree
24874         * @param {Node} this This node
24875         * @param {Node} node The removed node
24876         */
24877        "remove" : true,
24878        /**
24879         * @event move
24880         * Fires when this node is moved to a new location in the tree
24881         * @param {Tree} tree The owner tree
24882         * @param {Node} this This node
24883         * @param {Node} oldParent The old parent of this node
24884         * @param {Node} newParent The new parent of this node
24885         * @param {Number} index The index it was moved to
24886         */
24887        "move" : true,
24888        /**
24889         * @event insert
24890         * Fires when a new child node is inserted.
24891         * @param {Tree} tree The owner tree
24892         * @param {Node} this This node
24893         * @param {Node} node The child node inserted
24894         * @param {Node} refNode The child node the node was inserted before
24895         */
24896        "insert" : true,
24897        /**
24898         * @event beforeappend
24899         * Fires before a new child is appended, return false to cancel the append.
24900         * @param {Tree} tree The owner tree
24901         * @param {Node} this This node
24902         * @param {Node} node The child node to be appended
24903         */
24904        "beforeappend" : true,
24905        /**
24906         * @event beforeremove
24907         * Fires before a child is removed, return false to cancel the remove.
24908         * @param {Tree} tree The owner tree
24909         * @param {Node} this This node
24910         * @param {Node} node The child node to be removed
24911         */
24912        "beforeremove" : true,
24913        /**
24914         * @event beforemove
24915         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24916         * @param {Tree} tree The owner tree
24917         * @param {Node} this This node
24918         * @param {Node} oldParent The parent of this node
24919         * @param {Node} newParent The new parent this node is moving to
24920         * @param {Number} index The index it is being moved to
24921         */
24922        "beforemove" : true,
24923        /**
24924         * @event beforeinsert
24925         * Fires before a new child is inserted, return false to cancel the insert.
24926         * @param {Tree} tree The owner tree
24927         * @param {Node} this This node
24928         * @param {Node} node The child node to be inserted
24929         * @param {Node} refNode The child node the node is being inserted before
24930         */
24931        "beforeinsert" : true
24932    });
24933     this.listeners = this.attributes.listeners;
24934     Roo.data.Node.superclass.constructor.call(this);
24935 };
24936
24937 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24938     fireEvent : function(evtName){
24939         // first do standard event for this node
24940         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24941             return false;
24942         }
24943         // then bubble it up to the tree if the event wasn't cancelled
24944         var ot = this.getOwnerTree();
24945         if(ot){
24946             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24947                 return false;
24948             }
24949         }
24950         return true;
24951     },
24952
24953     /**
24954      * Returns true if this node is a leaf
24955      * @return {Boolean}
24956      */
24957     isLeaf : function(){
24958         return this.leaf === true;
24959     },
24960
24961     // private
24962     setFirstChild : function(node){
24963         this.firstChild = node;
24964     },
24965
24966     //private
24967     setLastChild : function(node){
24968         this.lastChild = node;
24969     },
24970
24971
24972     /**
24973      * Returns true if this node is the last child of its parent
24974      * @return {Boolean}
24975      */
24976     isLast : function(){
24977        return (!this.parentNode ? true : this.parentNode.lastChild == this);
24978     },
24979
24980     /**
24981      * Returns true if this node is the first child of its parent
24982      * @return {Boolean}
24983      */
24984     isFirst : function(){
24985        return (!this.parentNode ? true : this.parentNode.firstChild == this);
24986     },
24987
24988     hasChildNodes : function(){
24989         return !this.isLeaf() && this.childNodes.length > 0;
24990     },
24991
24992     /**
24993      * Insert node(s) as the last child node of this node.
24994      * @param {Node/Array} node The node or Array of nodes to append
24995      * @return {Node} The appended node if single append, or null if an array was passed
24996      */
24997     appendChild : function(node){
24998         var multi = false;
24999         if(node instanceof Array){
25000             multi = node;
25001         }else if(arguments.length > 1){
25002             multi = arguments;
25003         }
25004         // if passed an array or multiple args do them one by one
25005         if(multi){
25006             for(var i = 0, len = multi.length; i < len; i++) {
25007                 this.appendChild(multi[i]);
25008             }
25009         }else{
25010             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25011                 return false;
25012             }
25013             var index = this.childNodes.length;
25014             var oldParent = node.parentNode;
25015             // it's a move, make sure we move it cleanly
25016             if(oldParent){
25017                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25018                     return false;
25019                 }
25020                 oldParent.removeChild(node);
25021             }
25022             index = this.childNodes.length;
25023             if(index == 0){
25024                 this.setFirstChild(node);
25025             }
25026             this.childNodes.push(node);
25027             node.parentNode = this;
25028             var ps = this.childNodes[index-1];
25029             if(ps){
25030                 node.previousSibling = ps;
25031                 ps.nextSibling = node;
25032             }else{
25033                 node.previousSibling = null;
25034             }
25035             node.nextSibling = null;
25036             this.setLastChild(node);
25037             node.setOwnerTree(this.getOwnerTree());
25038             this.fireEvent("append", this.ownerTree, this, node, index);
25039             if(oldParent){
25040                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25041             }
25042             return node;
25043         }
25044     },
25045
25046     /**
25047      * Removes a child node from this node.
25048      * @param {Node} node The node to remove
25049      * @return {Node} The removed node
25050      */
25051     removeChild : function(node){
25052         var index = this.childNodes.indexOf(node);
25053         if(index == -1){
25054             return false;
25055         }
25056         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25057             return false;
25058         }
25059
25060         // remove it from childNodes collection
25061         this.childNodes.splice(index, 1);
25062
25063         // update siblings
25064         if(node.previousSibling){
25065             node.previousSibling.nextSibling = node.nextSibling;
25066         }
25067         if(node.nextSibling){
25068             node.nextSibling.previousSibling = node.previousSibling;
25069         }
25070
25071         // update child refs
25072         if(this.firstChild == node){
25073             this.setFirstChild(node.nextSibling);
25074         }
25075         if(this.lastChild == node){
25076             this.setLastChild(node.previousSibling);
25077         }
25078
25079         node.setOwnerTree(null);
25080         // clear any references from the node
25081         node.parentNode = null;
25082         node.previousSibling = null;
25083         node.nextSibling = null;
25084         this.fireEvent("remove", this.ownerTree, this, node);
25085         return node;
25086     },
25087
25088     /**
25089      * Inserts the first node before the second node in this nodes childNodes collection.
25090      * @param {Node} node The node to insert
25091      * @param {Node} refNode The node to insert before (if null the node is appended)
25092      * @return {Node} The inserted node
25093      */
25094     insertBefore : function(node, refNode){
25095         if(!refNode){ // like standard Dom, refNode can be null for append
25096             return this.appendChild(node);
25097         }
25098         // nothing to do
25099         if(node == refNode){
25100             return false;
25101         }
25102
25103         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25104             return false;
25105         }
25106         var index = this.childNodes.indexOf(refNode);
25107         var oldParent = node.parentNode;
25108         var refIndex = index;
25109
25110         // when moving internally, indexes will change after remove
25111         if(oldParent == this && this.childNodes.indexOf(node) < index){
25112             refIndex--;
25113         }
25114
25115         // it's a move, make sure we move it cleanly
25116         if(oldParent){
25117             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25118                 return false;
25119             }
25120             oldParent.removeChild(node);
25121         }
25122         if(refIndex == 0){
25123             this.setFirstChild(node);
25124         }
25125         this.childNodes.splice(refIndex, 0, node);
25126         node.parentNode = this;
25127         var ps = this.childNodes[refIndex-1];
25128         if(ps){
25129             node.previousSibling = ps;
25130             ps.nextSibling = node;
25131         }else{
25132             node.previousSibling = null;
25133         }
25134         node.nextSibling = refNode;
25135         refNode.previousSibling = node;
25136         node.setOwnerTree(this.getOwnerTree());
25137         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25138         if(oldParent){
25139             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25140         }
25141         return node;
25142     },
25143
25144     /**
25145      * Returns the child node at the specified index.
25146      * @param {Number} index
25147      * @return {Node}
25148      */
25149     item : function(index){
25150         return this.childNodes[index];
25151     },
25152
25153     /**
25154      * Replaces one child node in this node with another.
25155      * @param {Node} newChild The replacement node
25156      * @param {Node} oldChild The node to replace
25157      * @return {Node} The replaced node
25158      */
25159     replaceChild : function(newChild, oldChild){
25160         this.insertBefore(newChild, oldChild);
25161         this.removeChild(oldChild);
25162         return oldChild;
25163     },
25164
25165     /**
25166      * Returns the index of a child node
25167      * @param {Node} node
25168      * @return {Number} The index of the node or -1 if it was not found
25169      */
25170     indexOf : function(child){
25171         return this.childNodes.indexOf(child);
25172     },
25173
25174     /**
25175      * Returns the tree this node is in.
25176      * @return {Tree}
25177      */
25178     getOwnerTree : function(){
25179         // if it doesn't have one, look for one
25180         if(!this.ownerTree){
25181             var p = this;
25182             while(p){
25183                 if(p.ownerTree){
25184                     this.ownerTree = p.ownerTree;
25185                     break;
25186                 }
25187                 p = p.parentNode;
25188             }
25189         }
25190         return this.ownerTree;
25191     },
25192
25193     /**
25194      * Returns depth of this node (the root node has a depth of 0)
25195      * @return {Number}
25196      */
25197     getDepth : function(){
25198         var depth = 0;
25199         var p = this;
25200         while(p.parentNode){
25201             ++depth;
25202             p = p.parentNode;
25203         }
25204         return depth;
25205     },
25206
25207     // private
25208     setOwnerTree : function(tree){
25209         // if it's move, we need to update everyone
25210         if(tree != this.ownerTree){
25211             if(this.ownerTree){
25212                 this.ownerTree.unregisterNode(this);
25213             }
25214             this.ownerTree = tree;
25215             var cs = this.childNodes;
25216             for(var i = 0, len = cs.length; i < len; i++) {
25217                 cs[i].setOwnerTree(tree);
25218             }
25219             if(tree){
25220                 tree.registerNode(this);
25221             }
25222         }
25223     },
25224
25225     /**
25226      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25227      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25228      * @return {String} The path
25229      */
25230     getPath : function(attr){
25231         attr = attr || "id";
25232         var p = this.parentNode;
25233         var b = [this.attributes[attr]];
25234         while(p){
25235             b.unshift(p.attributes[attr]);
25236             p = p.parentNode;
25237         }
25238         var sep = this.getOwnerTree().pathSeparator;
25239         return sep + b.join(sep);
25240     },
25241
25242     /**
25243      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25244      * function call will be the scope provided or the current node. The arguments to the function
25245      * will be the args provided or the current node. If the function returns false at any point,
25246      * the bubble is stopped.
25247      * @param {Function} fn The function to call
25248      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25249      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25250      */
25251     bubble : function(fn, scope, args){
25252         var p = this;
25253         while(p){
25254             if(fn.call(scope || p, args || p) === false){
25255                 break;
25256             }
25257             p = p.parentNode;
25258         }
25259     },
25260
25261     /**
25262      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25263      * function call will be the scope provided or the current node. The arguments to the function
25264      * will be the args provided or the current node. If the function returns false at any point,
25265      * the cascade is stopped on that branch.
25266      * @param {Function} fn The function to call
25267      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25268      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25269      */
25270     cascade : function(fn, scope, args){
25271         if(fn.call(scope || this, args || this) !== false){
25272             var cs = this.childNodes;
25273             for(var i = 0, len = cs.length; i < len; i++) {
25274                 cs[i].cascade(fn, scope, args);
25275             }
25276         }
25277     },
25278
25279     /**
25280      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25281      * function call will be the scope provided or the current node. The arguments to the function
25282      * will be the args provided or the current node. If the function returns false at any point,
25283      * the iteration stops.
25284      * @param {Function} fn The function to call
25285      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25286      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25287      */
25288     eachChild : function(fn, scope, args){
25289         var cs = this.childNodes;
25290         for(var i = 0, len = cs.length; i < len; i++) {
25291                 if(fn.call(scope || this, args || cs[i]) === false){
25292                     break;
25293                 }
25294         }
25295     },
25296
25297     /**
25298      * Finds the first child that has the attribute with the specified value.
25299      * @param {String} attribute The attribute name
25300      * @param {Mixed} value The value to search for
25301      * @return {Node} The found child or null if none was found
25302      */
25303     findChild : function(attribute, value){
25304         var cs = this.childNodes;
25305         for(var i = 0, len = cs.length; i < len; i++) {
25306                 if(cs[i].attributes[attribute] == value){
25307                     return cs[i];
25308                 }
25309         }
25310         return null;
25311     },
25312
25313     /**
25314      * Finds the first child by a custom function. The child matches if the function passed
25315      * returns true.
25316      * @param {Function} fn
25317      * @param {Object} scope (optional)
25318      * @return {Node} The found child or null if none was found
25319      */
25320     findChildBy : function(fn, scope){
25321         var cs = this.childNodes;
25322         for(var i = 0, len = cs.length; i < len; i++) {
25323                 if(fn.call(scope||cs[i], cs[i]) === true){
25324                     return cs[i];
25325                 }
25326         }
25327         return null;
25328     },
25329
25330     /**
25331      * Sorts this nodes children using the supplied sort function
25332      * @param {Function} fn
25333      * @param {Object} scope (optional)
25334      */
25335     sort : function(fn, scope){
25336         var cs = this.childNodes;
25337         var len = cs.length;
25338         if(len > 0){
25339             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25340             cs.sort(sortFn);
25341             for(var i = 0; i < len; i++){
25342                 var n = cs[i];
25343                 n.previousSibling = cs[i-1];
25344                 n.nextSibling = cs[i+1];
25345                 if(i == 0){
25346                     this.setFirstChild(n);
25347                 }
25348                 if(i == len-1){
25349                     this.setLastChild(n);
25350                 }
25351             }
25352         }
25353     },
25354
25355     /**
25356      * Returns true if this node is an ancestor (at any point) of the passed node.
25357      * @param {Node} node
25358      * @return {Boolean}
25359      */
25360     contains : function(node){
25361         return node.isAncestor(this);
25362     },
25363
25364     /**
25365      * Returns true if the passed node is an ancestor (at any point) of this node.
25366      * @param {Node} node
25367      * @return {Boolean}
25368      */
25369     isAncestor : function(node){
25370         var p = this.parentNode;
25371         while(p){
25372             if(p == node){
25373                 return true;
25374             }
25375             p = p.parentNode;
25376         }
25377         return false;
25378     },
25379
25380     toString : function(){
25381         return "[Node"+(this.id?" "+this.id:"")+"]";
25382     }
25383 });/*
25384  * Based on:
25385  * Ext JS Library 1.1.1
25386  * Copyright(c) 2006-2007, Ext JS, LLC.
25387  *
25388  * Originally Released Under LGPL - original licence link has changed is not relivant.
25389  *
25390  * Fork - LGPL
25391  * <script type="text/javascript">
25392  */
25393  (function(){ 
25394 /**
25395  * @class Roo.Layer
25396  * @extends Roo.Element
25397  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25398  * automatic maintaining of shadow/shim positions.
25399  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25400  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25401  * you can pass a string with a CSS class name. False turns off the shadow.
25402  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25403  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25404  * @cfg {String} cls CSS class to add to the element
25405  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25406  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25407  * @constructor
25408  * @param {Object} config An object with config options.
25409  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25410  */
25411
25412 Roo.Layer = function(config, existingEl){
25413     config = config || {};
25414     var dh = Roo.DomHelper;
25415     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25416     if(existingEl){
25417         this.dom = Roo.getDom(existingEl);
25418     }
25419     if(!this.dom){
25420         var o = config.dh || {tag: "div", cls: "x-layer"};
25421         this.dom = dh.append(pel, o);
25422     }
25423     if(config.cls){
25424         this.addClass(config.cls);
25425     }
25426     this.constrain = config.constrain !== false;
25427     this.visibilityMode = Roo.Element.VISIBILITY;
25428     if(config.id){
25429         this.id = this.dom.id = config.id;
25430     }else{
25431         this.id = Roo.id(this.dom);
25432     }
25433     this.zindex = config.zindex || this.getZIndex();
25434     this.position("absolute", this.zindex);
25435     if(config.shadow){
25436         this.shadowOffset = config.shadowOffset || 4;
25437         this.shadow = new Roo.Shadow({
25438             offset : this.shadowOffset,
25439             mode : config.shadow
25440         });
25441     }else{
25442         this.shadowOffset = 0;
25443     }
25444     this.useShim = config.shim !== false && Roo.useShims;
25445     this.useDisplay = config.useDisplay;
25446     this.hide();
25447 };
25448
25449 var supr = Roo.Element.prototype;
25450
25451 // shims are shared among layer to keep from having 100 iframes
25452 var shims = [];
25453
25454 Roo.extend(Roo.Layer, Roo.Element, {
25455
25456     getZIndex : function(){
25457         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25458     },
25459
25460     getShim : function(){
25461         if(!this.useShim){
25462             return null;
25463         }
25464         if(this.shim){
25465             return this.shim;
25466         }
25467         var shim = shims.shift();
25468         if(!shim){
25469             shim = this.createShim();
25470             shim.enableDisplayMode('block');
25471             shim.dom.style.display = 'none';
25472             shim.dom.style.visibility = 'visible';
25473         }
25474         var pn = this.dom.parentNode;
25475         if(shim.dom.parentNode != pn){
25476             pn.insertBefore(shim.dom, this.dom);
25477         }
25478         shim.setStyle('z-index', this.getZIndex()-2);
25479         this.shim = shim;
25480         return shim;
25481     },
25482
25483     hideShim : function(){
25484         if(this.shim){
25485             this.shim.setDisplayed(false);
25486             shims.push(this.shim);
25487             delete this.shim;
25488         }
25489     },
25490
25491     disableShadow : function(){
25492         if(this.shadow){
25493             this.shadowDisabled = true;
25494             this.shadow.hide();
25495             this.lastShadowOffset = this.shadowOffset;
25496             this.shadowOffset = 0;
25497         }
25498     },
25499
25500     enableShadow : function(show){
25501         if(this.shadow){
25502             this.shadowDisabled = false;
25503             this.shadowOffset = this.lastShadowOffset;
25504             delete this.lastShadowOffset;
25505             if(show){
25506                 this.sync(true);
25507             }
25508         }
25509     },
25510
25511     // private
25512     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25513     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25514     sync : function(doShow){
25515         var sw = this.shadow;
25516         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25517             var sh = this.getShim();
25518
25519             var w = this.getWidth(),
25520                 h = this.getHeight();
25521
25522             var l = this.getLeft(true),
25523                 t = this.getTop(true);
25524
25525             if(sw && !this.shadowDisabled){
25526                 if(doShow && !sw.isVisible()){
25527                     sw.show(this);
25528                 }else{
25529                     sw.realign(l, t, w, h);
25530                 }
25531                 if(sh){
25532                     if(doShow){
25533                        sh.show();
25534                     }
25535                     // fit the shim behind the shadow, so it is shimmed too
25536                     var a = sw.adjusts, s = sh.dom.style;
25537                     s.left = (Math.min(l, l+a.l))+"px";
25538                     s.top = (Math.min(t, t+a.t))+"px";
25539                     s.width = (w+a.w)+"px";
25540                     s.height = (h+a.h)+"px";
25541                 }
25542             }else if(sh){
25543                 if(doShow){
25544                    sh.show();
25545                 }
25546                 sh.setSize(w, h);
25547                 sh.setLeftTop(l, t);
25548             }
25549             
25550         }
25551     },
25552
25553     // private
25554     destroy : function(){
25555         this.hideShim();
25556         if(this.shadow){
25557             this.shadow.hide();
25558         }
25559         this.removeAllListeners();
25560         var pn = this.dom.parentNode;
25561         if(pn){
25562             pn.removeChild(this.dom);
25563         }
25564         Roo.Element.uncache(this.id);
25565     },
25566
25567     remove : function(){
25568         this.destroy();
25569     },
25570
25571     // private
25572     beginUpdate : function(){
25573         this.updating = true;
25574     },
25575
25576     // private
25577     endUpdate : function(){
25578         this.updating = false;
25579         this.sync(true);
25580     },
25581
25582     // private
25583     hideUnders : function(negOffset){
25584         if(this.shadow){
25585             this.shadow.hide();
25586         }
25587         this.hideShim();
25588     },
25589
25590     // private
25591     constrainXY : function(){
25592         if(this.constrain){
25593             var vw = Roo.lib.Dom.getViewWidth(),
25594                 vh = Roo.lib.Dom.getViewHeight();
25595             var s = Roo.get(document).getScroll();
25596
25597             var xy = this.getXY();
25598             var x = xy[0], y = xy[1];   
25599             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25600             // only move it if it needs it
25601             var moved = false;
25602             // first validate right/bottom
25603             if((x + w) > vw+s.left){
25604                 x = vw - w - this.shadowOffset;
25605                 moved = true;
25606             }
25607             if((y + h) > vh+s.top){
25608                 y = vh - h - this.shadowOffset;
25609                 moved = true;
25610             }
25611             // then make sure top/left isn't negative
25612             if(x < s.left){
25613                 x = s.left;
25614                 moved = true;
25615             }
25616             if(y < s.top){
25617                 y = s.top;
25618                 moved = true;
25619             }
25620             if(moved){
25621                 if(this.avoidY){
25622                     var ay = this.avoidY;
25623                     if(y <= ay && (y+h) >= ay){
25624                         y = ay-h-5;   
25625                     }
25626                 }
25627                 xy = [x, y];
25628                 this.storeXY(xy);
25629                 supr.setXY.call(this, xy);
25630                 this.sync();
25631             }
25632         }
25633     },
25634
25635     isVisible : function(){
25636         return this.visible;    
25637     },
25638
25639     // private
25640     showAction : function(){
25641         this.visible = true; // track visibility to prevent getStyle calls
25642         if(this.useDisplay === true){
25643             this.setDisplayed("");
25644         }else if(this.lastXY){
25645             supr.setXY.call(this, this.lastXY);
25646         }else if(this.lastLT){
25647             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25648         }
25649     },
25650
25651     // private
25652     hideAction : function(){
25653         this.visible = false;
25654         if(this.useDisplay === true){
25655             this.setDisplayed(false);
25656         }else{
25657             this.setLeftTop(-10000,-10000);
25658         }
25659     },
25660
25661     // overridden Element method
25662     setVisible : function(v, a, d, c, e){
25663         if(v){
25664             this.showAction();
25665         }
25666         if(a && v){
25667             var cb = function(){
25668                 this.sync(true);
25669                 if(c){
25670                     c();
25671                 }
25672             }.createDelegate(this);
25673             supr.setVisible.call(this, true, true, d, cb, e);
25674         }else{
25675             if(!v){
25676                 this.hideUnders(true);
25677             }
25678             var cb = c;
25679             if(a){
25680                 cb = function(){
25681                     this.hideAction();
25682                     if(c){
25683                         c();
25684                     }
25685                 }.createDelegate(this);
25686             }
25687             supr.setVisible.call(this, v, a, d, cb, e);
25688             if(v){
25689                 this.sync(true);
25690             }else if(!a){
25691                 this.hideAction();
25692             }
25693         }
25694     },
25695
25696     storeXY : function(xy){
25697         delete this.lastLT;
25698         this.lastXY = xy;
25699     },
25700
25701     storeLeftTop : function(left, top){
25702         delete this.lastXY;
25703         this.lastLT = [left, top];
25704     },
25705
25706     // private
25707     beforeFx : function(){
25708         this.beforeAction();
25709         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25710     },
25711
25712     // private
25713     afterFx : function(){
25714         Roo.Layer.superclass.afterFx.apply(this, arguments);
25715         this.sync(this.isVisible());
25716     },
25717
25718     // private
25719     beforeAction : function(){
25720         if(!this.updating && this.shadow){
25721             this.shadow.hide();
25722         }
25723     },
25724
25725     // overridden Element method
25726     setLeft : function(left){
25727         this.storeLeftTop(left, this.getTop(true));
25728         supr.setLeft.apply(this, arguments);
25729         this.sync();
25730     },
25731
25732     setTop : function(top){
25733         this.storeLeftTop(this.getLeft(true), top);
25734         supr.setTop.apply(this, arguments);
25735         this.sync();
25736     },
25737
25738     setLeftTop : function(left, top){
25739         this.storeLeftTop(left, top);
25740         supr.setLeftTop.apply(this, arguments);
25741         this.sync();
25742     },
25743
25744     setXY : function(xy, a, d, c, e){
25745         this.fixDisplay();
25746         this.beforeAction();
25747         this.storeXY(xy);
25748         var cb = this.createCB(c);
25749         supr.setXY.call(this, xy, a, d, cb, e);
25750         if(!a){
25751             cb();
25752         }
25753     },
25754
25755     // private
25756     createCB : function(c){
25757         var el = this;
25758         return function(){
25759             el.constrainXY();
25760             el.sync(true);
25761             if(c){
25762                 c();
25763             }
25764         };
25765     },
25766
25767     // overridden Element method
25768     setX : function(x, a, d, c, e){
25769         this.setXY([x, this.getY()], a, d, c, e);
25770     },
25771
25772     // overridden Element method
25773     setY : function(y, a, d, c, e){
25774         this.setXY([this.getX(), y], a, d, c, e);
25775     },
25776
25777     // overridden Element method
25778     setSize : function(w, h, a, d, c, e){
25779         this.beforeAction();
25780         var cb = this.createCB(c);
25781         supr.setSize.call(this, w, h, a, d, cb, e);
25782         if(!a){
25783             cb();
25784         }
25785     },
25786
25787     // overridden Element method
25788     setWidth : function(w, a, d, c, e){
25789         this.beforeAction();
25790         var cb = this.createCB(c);
25791         supr.setWidth.call(this, w, a, d, cb, e);
25792         if(!a){
25793             cb();
25794         }
25795     },
25796
25797     // overridden Element method
25798     setHeight : function(h, a, d, c, e){
25799         this.beforeAction();
25800         var cb = this.createCB(c);
25801         supr.setHeight.call(this, h, a, d, cb, e);
25802         if(!a){
25803             cb();
25804         }
25805     },
25806
25807     // overridden Element method
25808     setBounds : function(x, y, w, h, a, d, c, e){
25809         this.beforeAction();
25810         var cb = this.createCB(c);
25811         if(!a){
25812             this.storeXY([x, y]);
25813             supr.setXY.call(this, [x, y]);
25814             supr.setSize.call(this, w, h, a, d, cb, e);
25815             cb();
25816         }else{
25817             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25818         }
25819         return this;
25820     },
25821     
25822     /**
25823      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25824      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25825      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25826      * @param {Number} zindex The new z-index to set
25827      * @return {this} The Layer
25828      */
25829     setZIndex : function(zindex){
25830         this.zindex = zindex;
25831         this.setStyle("z-index", zindex + 2);
25832         if(this.shadow){
25833             this.shadow.setZIndex(zindex + 1);
25834         }
25835         if(this.shim){
25836             this.shim.setStyle("z-index", zindex);
25837         }
25838     }
25839 });
25840 })();/*
25841  * Based on:
25842  * Ext JS Library 1.1.1
25843  * Copyright(c) 2006-2007, Ext JS, LLC.
25844  *
25845  * Originally Released Under LGPL - original licence link has changed is not relivant.
25846  *
25847  * Fork - LGPL
25848  * <script type="text/javascript">
25849  */
25850
25851
25852 /**
25853  * @class Roo.Shadow
25854  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25855  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25856  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25857  * @constructor
25858  * Create a new Shadow
25859  * @param {Object} config The config object
25860  */
25861 Roo.Shadow = function(config){
25862     Roo.apply(this, config);
25863     if(typeof this.mode != "string"){
25864         this.mode = this.defaultMode;
25865     }
25866     var o = this.offset, a = {h: 0};
25867     var rad = Math.floor(this.offset/2);
25868     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25869         case "drop":
25870             a.w = 0;
25871             a.l = a.t = o;
25872             a.t -= 1;
25873             if(Roo.isIE){
25874                 a.l -= this.offset + rad;
25875                 a.t -= this.offset + rad;
25876                 a.w -= rad;
25877                 a.h -= rad;
25878                 a.t += 1;
25879             }
25880         break;
25881         case "sides":
25882             a.w = (o*2);
25883             a.l = -o;
25884             a.t = o-1;
25885             if(Roo.isIE){
25886                 a.l -= (this.offset - rad);
25887                 a.t -= this.offset + rad;
25888                 a.l += 1;
25889                 a.w -= (this.offset - rad)*2;
25890                 a.w -= rad + 1;
25891                 a.h -= 1;
25892             }
25893         break;
25894         case "frame":
25895             a.w = a.h = (o*2);
25896             a.l = a.t = -o;
25897             a.t += 1;
25898             a.h -= 2;
25899             if(Roo.isIE){
25900                 a.l -= (this.offset - rad);
25901                 a.t -= (this.offset - rad);
25902                 a.l += 1;
25903                 a.w -= (this.offset + rad + 1);
25904                 a.h -= (this.offset + rad);
25905                 a.h += 1;
25906             }
25907         break;
25908     };
25909
25910     this.adjusts = a;
25911 };
25912
25913 Roo.Shadow.prototype = {
25914     /**
25915      * @cfg {String} mode
25916      * The shadow display mode.  Supports the following options:<br />
25917      * sides: Shadow displays on both sides and bottom only<br />
25918      * frame: Shadow displays equally on all four sides<br />
25919      * drop: Traditional bottom-right drop shadow (default)
25920      */
25921     /**
25922      * @cfg {String} offset
25923      * The number of pixels to offset the shadow from the element (defaults to 4)
25924      */
25925     offset: 4,
25926
25927     // private
25928     defaultMode: "drop",
25929
25930     /**
25931      * Displays the shadow under the target element
25932      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25933      */
25934     show : function(target){
25935         target = Roo.get(target);
25936         if(!this.el){
25937             this.el = Roo.Shadow.Pool.pull();
25938             if(this.el.dom.nextSibling != target.dom){
25939                 this.el.insertBefore(target);
25940             }
25941         }
25942         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25943         if(Roo.isIE){
25944             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25945         }
25946         this.realign(
25947             target.getLeft(true),
25948             target.getTop(true),
25949             target.getWidth(),
25950             target.getHeight()
25951         );
25952         this.el.dom.style.display = "block";
25953     },
25954
25955     /**
25956      * Returns true if the shadow is visible, else false
25957      */
25958     isVisible : function(){
25959         return this.el ? true : false;  
25960     },
25961
25962     /**
25963      * Direct alignment when values are already available. Show must be called at least once before
25964      * calling this method to ensure it is initialized.
25965      * @param {Number} left The target element left position
25966      * @param {Number} top The target element top position
25967      * @param {Number} width The target element width
25968      * @param {Number} height The target element height
25969      */
25970     realign : function(l, t, w, h){
25971         if(!this.el){
25972             return;
25973         }
25974         var a = this.adjusts, d = this.el.dom, s = d.style;
25975         var iea = 0;
25976         s.left = (l+a.l)+"px";
25977         s.top = (t+a.t)+"px";
25978         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
25979  
25980         if(s.width != sws || s.height != shs){
25981             s.width = sws;
25982             s.height = shs;
25983             if(!Roo.isIE){
25984                 var cn = d.childNodes;
25985                 var sww = Math.max(0, (sw-12))+"px";
25986                 cn[0].childNodes[1].style.width = sww;
25987                 cn[1].childNodes[1].style.width = sww;
25988                 cn[2].childNodes[1].style.width = sww;
25989                 cn[1].style.height = Math.max(0, (sh-12))+"px";
25990             }
25991         }
25992     },
25993
25994     /**
25995      * Hides this shadow
25996      */
25997     hide : function(){
25998         if(this.el){
25999             this.el.dom.style.display = "none";
26000             Roo.Shadow.Pool.push(this.el);
26001             delete this.el;
26002         }
26003     },
26004
26005     /**
26006      * Adjust the z-index of this shadow
26007      * @param {Number} zindex The new z-index
26008      */
26009     setZIndex : function(z){
26010         this.zIndex = z;
26011         if(this.el){
26012             this.el.setStyle("z-index", z);
26013         }
26014     }
26015 };
26016
26017 // Private utility class that manages the internal Shadow cache
26018 Roo.Shadow.Pool = function(){
26019     var p = [];
26020     var markup = Roo.isIE ?
26021                  '<div class="x-ie-shadow"></div>' :
26022                  '<div class="x-shadow"><div class="xst"><div class="xstl"></div><div class="xstc"></div><div class="xstr"></div></div><div class="xsc"><div class="xsml"></div><div class="xsmc"></div><div class="xsmr"></div></div><div class="xsb"><div class="xsbl"></div><div class="xsbc"></div><div class="xsbr"></div></div></div>';
26023     return {
26024         pull : function(){
26025             var sh = p.shift();
26026             if(!sh){
26027                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26028                 sh.autoBoxAdjust = false;
26029             }
26030             return sh;
26031         },
26032
26033         push : function(sh){
26034             p.push(sh);
26035         }
26036     };
26037 }();/*
26038  * Based on:
26039  * Ext JS Library 1.1.1
26040  * Copyright(c) 2006-2007, Ext JS, LLC.
26041  *
26042  * Originally Released Under LGPL - original licence link has changed is not relivant.
26043  *
26044  * Fork - LGPL
26045  * <script type="text/javascript">
26046  */
26047
26048
26049 /**
26050  * @class Roo.SplitBar
26051  * @extends Roo.util.Observable
26052  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26053  * <br><br>
26054  * Usage:
26055  * <pre><code>
26056 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26057                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26058 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26059 split.minSize = 100;
26060 split.maxSize = 600;
26061 split.animate = true;
26062 split.on('moved', splitterMoved);
26063 </code></pre>
26064  * @constructor
26065  * Create a new SplitBar
26066  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26067  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26068  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26069  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26070                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26071                         position of the SplitBar).
26072  */
26073 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26074     
26075     /** @private */
26076     this.el = Roo.get(dragElement, true);
26077     this.el.dom.unselectable = "on";
26078     /** @private */
26079     this.resizingEl = Roo.get(resizingElement, true);
26080
26081     /**
26082      * @private
26083      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26084      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26085      * @type Number
26086      */
26087     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26088     
26089     /**
26090      * The minimum size of the resizing element. (Defaults to 0)
26091      * @type Number
26092      */
26093     this.minSize = 0;
26094     
26095     /**
26096      * The maximum size of the resizing element. (Defaults to 2000)
26097      * @type Number
26098      */
26099     this.maxSize = 2000;
26100     
26101     /**
26102      * Whether to animate the transition to the new size
26103      * @type Boolean
26104      */
26105     this.animate = false;
26106     
26107     /**
26108      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26109      * @type Boolean
26110      */
26111     this.useShim = false;
26112     
26113     /** @private */
26114     this.shim = null;
26115     
26116     if(!existingProxy){
26117         /** @private */
26118         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26119     }else{
26120         this.proxy = Roo.get(existingProxy).dom;
26121     }
26122     /** @private */
26123     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26124     
26125     /** @private */
26126     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26127     
26128     /** @private */
26129     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26130     
26131     /** @private */
26132     this.dragSpecs = {};
26133     
26134     /**
26135      * @private The adapter to use to positon and resize elements
26136      */
26137     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26138     this.adapter.init(this);
26139     
26140     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26141         /** @private */
26142         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26143         this.el.addClass("x-splitbar-h");
26144     }else{
26145         /** @private */
26146         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26147         this.el.addClass("x-splitbar-v");
26148     }
26149     
26150     this.addEvents({
26151         /**
26152          * @event resize
26153          * Fires when the splitter is moved (alias for {@link #event-moved})
26154          * @param {Roo.SplitBar} this
26155          * @param {Number} newSize the new width or height
26156          */
26157         "resize" : true,
26158         /**
26159          * @event moved
26160          * Fires when the splitter is moved
26161          * @param {Roo.SplitBar} this
26162          * @param {Number} newSize the new width or height
26163          */
26164         "moved" : true,
26165         /**
26166          * @event beforeresize
26167          * Fires before the splitter is dragged
26168          * @param {Roo.SplitBar} this
26169          */
26170         "beforeresize" : true,
26171
26172         "beforeapply" : true
26173     });
26174
26175     Roo.util.Observable.call(this);
26176 };
26177
26178 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26179     onStartProxyDrag : function(x, y){
26180         this.fireEvent("beforeresize", this);
26181         if(!this.overlay){
26182             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26183             o.unselectable();
26184             o.enableDisplayMode("block");
26185             // all splitbars share the same overlay
26186             Roo.SplitBar.prototype.overlay = o;
26187         }
26188         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26189         this.overlay.show();
26190         Roo.get(this.proxy).setDisplayed("block");
26191         var size = this.adapter.getElementSize(this);
26192         this.activeMinSize = this.getMinimumSize();;
26193         this.activeMaxSize = this.getMaximumSize();;
26194         var c1 = size - this.activeMinSize;
26195         var c2 = Math.max(this.activeMaxSize - size, 0);
26196         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26197             this.dd.resetConstraints();
26198             this.dd.setXConstraint(
26199                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26200                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26201             );
26202             this.dd.setYConstraint(0, 0);
26203         }else{
26204             this.dd.resetConstraints();
26205             this.dd.setXConstraint(0, 0);
26206             this.dd.setYConstraint(
26207                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26208                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26209             );
26210          }
26211         this.dragSpecs.startSize = size;
26212         this.dragSpecs.startPoint = [x, y];
26213         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26214     },
26215     
26216     /** 
26217      * @private Called after the drag operation by the DDProxy
26218      */
26219     onEndProxyDrag : function(e){
26220         Roo.get(this.proxy).setDisplayed(false);
26221         var endPoint = Roo.lib.Event.getXY(e);
26222         if(this.overlay){
26223             this.overlay.hide();
26224         }
26225         var newSize;
26226         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26227             newSize = this.dragSpecs.startSize + 
26228                 (this.placement == Roo.SplitBar.LEFT ?
26229                     endPoint[0] - this.dragSpecs.startPoint[0] :
26230                     this.dragSpecs.startPoint[0] - endPoint[0]
26231                 );
26232         }else{
26233             newSize = this.dragSpecs.startSize + 
26234                 (this.placement == Roo.SplitBar.TOP ?
26235                     endPoint[1] - this.dragSpecs.startPoint[1] :
26236                     this.dragSpecs.startPoint[1] - endPoint[1]
26237                 );
26238         }
26239         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26240         if(newSize != this.dragSpecs.startSize){
26241             if(this.fireEvent('beforeapply', this, newSize) !== false){
26242                 this.adapter.setElementSize(this, newSize);
26243                 this.fireEvent("moved", this, newSize);
26244                 this.fireEvent("resize", this, newSize);
26245             }
26246         }
26247     },
26248     
26249     /**
26250      * Get the adapter this SplitBar uses
26251      * @return The adapter object
26252      */
26253     getAdapter : function(){
26254         return this.adapter;
26255     },
26256     
26257     /**
26258      * Set the adapter this SplitBar uses
26259      * @param {Object} adapter A SplitBar adapter object
26260      */
26261     setAdapter : function(adapter){
26262         this.adapter = adapter;
26263         this.adapter.init(this);
26264     },
26265     
26266     /**
26267      * Gets the minimum size for the resizing element
26268      * @return {Number} The minimum size
26269      */
26270     getMinimumSize : function(){
26271         return this.minSize;
26272     },
26273     
26274     /**
26275      * Sets the minimum size for the resizing element
26276      * @param {Number} minSize The minimum size
26277      */
26278     setMinimumSize : function(minSize){
26279         this.minSize = minSize;
26280     },
26281     
26282     /**
26283      * Gets the maximum size for the resizing element
26284      * @return {Number} The maximum size
26285      */
26286     getMaximumSize : function(){
26287         return this.maxSize;
26288     },
26289     
26290     /**
26291      * Sets the maximum size for the resizing element
26292      * @param {Number} maxSize The maximum size
26293      */
26294     setMaximumSize : function(maxSize){
26295         this.maxSize = maxSize;
26296     },
26297     
26298     /**
26299      * Sets the initialize size for the resizing element
26300      * @param {Number} size The initial size
26301      */
26302     setCurrentSize : function(size){
26303         var oldAnimate = this.animate;
26304         this.animate = false;
26305         this.adapter.setElementSize(this, size);
26306         this.animate = oldAnimate;
26307     },
26308     
26309     /**
26310      * Destroy this splitbar. 
26311      * @param {Boolean} removeEl True to remove the element
26312      */
26313     destroy : function(removeEl){
26314         if(this.shim){
26315             this.shim.remove();
26316         }
26317         this.dd.unreg();
26318         this.proxy.parentNode.removeChild(this.proxy);
26319         if(removeEl){
26320             this.el.remove();
26321         }
26322     }
26323 });
26324
26325 /**
26326  * @private static Create our own proxy element element. So it will be the same same size on all browsers, we won't use borders. Instead we use a background color.
26327  */
26328 Roo.SplitBar.createProxy = function(dir){
26329     var proxy = new Roo.Element(document.createElement("div"));
26330     proxy.unselectable();
26331     var cls = 'x-splitbar-proxy';
26332     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26333     document.body.appendChild(proxy.dom);
26334     return proxy.dom;
26335 };
26336
26337 /** 
26338  * @class Roo.SplitBar.BasicLayoutAdapter
26339  * Default Adapter. It assumes the splitter and resizing element are not positioned
26340  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26341  */
26342 Roo.SplitBar.BasicLayoutAdapter = function(){
26343 };
26344
26345 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26346     // do nothing for now
26347     init : function(s){
26348     
26349     },
26350     /**
26351      * Called before drag operations to get the current size of the resizing element. 
26352      * @param {Roo.SplitBar} s The SplitBar using this adapter
26353      */
26354      getElementSize : function(s){
26355         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26356             return s.resizingEl.getWidth();
26357         }else{
26358             return s.resizingEl.getHeight();
26359         }
26360     },
26361     
26362     /**
26363      * Called after drag operations to set the size of the resizing element.
26364      * @param {Roo.SplitBar} s The SplitBar using this adapter
26365      * @param {Number} newSize The new size to set
26366      * @param {Function} onComplete A function to be invoked when resizing is complete
26367      */
26368     setElementSize : function(s, newSize, onComplete){
26369         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26370             if(!s.animate){
26371                 s.resizingEl.setWidth(newSize);
26372                 if(onComplete){
26373                     onComplete(s, newSize);
26374                 }
26375             }else{
26376                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26377             }
26378         }else{
26379             
26380             if(!s.animate){
26381                 s.resizingEl.setHeight(newSize);
26382                 if(onComplete){
26383                     onComplete(s, newSize);
26384                 }
26385             }else{
26386                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26387             }
26388         }
26389     }
26390 };
26391
26392 /** 
26393  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26394  * @extends Roo.SplitBar.BasicLayoutAdapter
26395  * Adapter that  moves the splitter element to align with the resized sizing element. 
26396  * Used with an absolute positioned SplitBar.
26397  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26398  * document.body, make sure you assign an id to the body element.
26399  */
26400 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26401     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26402     this.container = Roo.get(container);
26403 };
26404
26405 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26406     init : function(s){
26407         this.basic.init(s);
26408     },
26409     
26410     getElementSize : function(s){
26411         return this.basic.getElementSize(s);
26412     },
26413     
26414     setElementSize : function(s, newSize, onComplete){
26415         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26416     },
26417     
26418     moveSplitter : function(s){
26419         var yes = Roo.SplitBar;
26420         switch(s.placement){
26421             case yes.LEFT:
26422                 s.el.setX(s.resizingEl.getRight());
26423                 break;
26424             case yes.RIGHT:
26425                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26426                 break;
26427             case yes.TOP:
26428                 s.el.setY(s.resizingEl.getBottom());
26429                 break;
26430             case yes.BOTTOM:
26431                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26432                 break;
26433         }
26434     }
26435 };
26436
26437 /**
26438  * Orientation constant - Create a vertical SplitBar
26439  * @static
26440  * @type Number
26441  */
26442 Roo.SplitBar.VERTICAL = 1;
26443
26444 /**
26445  * Orientation constant - Create a horizontal SplitBar
26446  * @static
26447  * @type Number
26448  */
26449 Roo.SplitBar.HORIZONTAL = 2;
26450
26451 /**
26452  * Placement constant - The resizing element is to the left of the splitter element
26453  * @static
26454  * @type Number
26455  */
26456 Roo.SplitBar.LEFT = 1;
26457
26458 /**
26459  * Placement constant - The resizing element is to the right of the splitter element
26460  * @static
26461  * @type Number
26462  */
26463 Roo.SplitBar.RIGHT = 2;
26464
26465 /**
26466  * Placement constant - The resizing element is positioned above the splitter element
26467  * @static
26468  * @type Number
26469  */
26470 Roo.SplitBar.TOP = 3;
26471
26472 /**
26473  * Placement constant - The resizing element is positioned under splitter element
26474  * @static
26475  * @type Number
26476  */
26477 Roo.SplitBar.BOTTOM = 4;
26478 /*
26479  * Based on:
26480  * Ext JS Library 1.1.1
26481  * Copyright(c) 2006-2007, Ext JS, LLC.
26482  *
26483  * Originally Released Under LGPL - original licence link has changed is not relivant.
26484  *
26485  * Fork - LGPL
26486  * <script type="text/javascript">
26487  */
26488
26489 /**
26490  * @class Roo.View
26491  * @extends Roo.util.Observable
26492  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26493  * This class also supports single and multi selection modes. <br>
26494  * Create a data model bound view:
26495  <pre><code>
26496  var store = new Roo.data.Store(...);
26497
26498  var view = new Roo.View({
26499     el : "my-element",
26500     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26501  
26502     singleSelect: true,
26503     selectedClass: "ydataview-selected",
26504     store: store
26505  });
26506
26507  // listen for node click?
26508  view.on("click", function(vw, index, node, e){
26509  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26510  });
26511
26512  // load XML data
26513  dataModel.load("foobar.xml");
26514  </code></pre>
26515  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26516  * <br><br>
26517  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26518  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26519  * 
26520  * Note: old style constructor is still suported (container, template, config)
26521  * 
26522  * @constructor
26523  * Create a new View
26524  * @param {Object} config The config object
26525  * 
26526  */
26527 Roo.View = function(config, depreciated_tpl, depreciated_config){
26528     
26529     this.parent = false;
26530     
26531     if (typeof(depreciated_tpl) == 'undefined') {
26532         // new way.. - universal constructor.
26533         Roo.apply(this, config);
26534         this.el  = Roo.get(this.el);
26535     } else {
26536         // old format..
26537         this.el  = Roo.get(config);
26538         this.tpl = depreciated_tpl;
26539         Roo.apply(this, depreciated_config);
26540     }
26541     this.wrapEl  = this.el.wrap().wrap();
26542     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26543     
26544     
26545     if(typeof(this.tpl) == "string"){
26546         this.tpl = new Roo.Template(this.tpl);
26547     } else {
26548         // support xtype ctors..
26549         this.tpl = new Roo.factory(this.tpl, Roo);
26550     }
26551     
26552     
26553     this.tpl.compile();
26554     
26555     /** @private */
26556     this.addEvents({
26557         /**
26558          * @event beforeclick
26559          * Fires before a click is processed. Returns false to cancel the default action.
26560          * @param {Roo.View} this
26561          * @param {Number} index The index of the target node
26562          * @param {HTMLElement} node The target node
26563          * @param {Roo.EventObject} e The raw event object
26564          */
26565             "beforeclick" : true,
26566         /**
26567          * @event click
26568          * Fires when a template node is clicked.
26569          * @param {Roo.View} this
26570          * @param {Number} index The index of the target node
26571          * @param {HTMLElement} node The target node
26572          * @param {Roo.EventObject} e The raw event object
26573          */
26574             "click" : true,
26575         /**
26576          * @event dblclick
26577          * Fires when a template node is double clicked.
26578          * @param {Roo.View} this
26579          * @param {Number} index The index of the target node
26580          * @param {HTMLElement} node The target node
26581          * @param {Roo.EventObject} e The raw event object
26582          */
26583             "dblclick" : true,
26584         /**
26585          * @event contextmenu
26586          * Fires when a template node is right clicked.
26587          * @param {Roo.View} this
26588          * @param {Number} index The index of the target node
26589          * @param {HTMLElement} node The target node
26590          * @param {Roo.EventObject} e The raw event object
26591          */
26592             "contextmenu" : true,
26593         /**
26594          * @event selectionchange
26595          * Fires when the selected nodes change.
26596          * @param {Roo.View} this
26597          * @param {Array} selections Array of the selected nodes
26598          */
26599             "selectionchange" : true,
26600     
26601         /**
26602          * @event beforeselect
26603          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26604          * @param {Roo.View} this
26605          * @param {HTMLElement} node The node to be selected
26606          * @param {Array} selections Array of currently selected nodes
26607          */
26608             "beforeselect" : true,
26609         /**
26610          * @event preparedata
26611          * Fires on every row to render, to allow you to change the data.
26612          * @param {Roo.View} this
26613          * @param {Object} data to be rendered (change this)
26614          */
26615           "preparedata" : true
26616           
26617           
26618         });
26619
26620
26621
26622     this.el.on({
26623         "click": this.onClick,
26624         "dblclick": this.onDblClick,
26625         "contextmenu": this.onContextMenu,
26626         scope:this
26627     });
26628
26629     this.selections = [];
26630     this.nodes = [];
26631     this.cmp = new Roo.CompositeElementLite([]);
26632     if(this.store){
26633         this.store = Roo.factory(this.store, Roo.data);
26634         this.setStore(this.store, true);
26635     }
26636     
26637     if ( this.footer && this.footer.xtype) {
26638            
26639          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26640         
26641         this.footer.dataSource = this.store;
26642         this.footer.container = fctr;
26643         this.footer = Roo.factory(this.footer, Roo);
26644         fctr.insertFirst(this.el);
26645         
26646         // this is a bit insane - as the paging toolbar seems to detach the el..
26647 //        dom.parentNode.parentNode.parentNode
26648          // they get detached?
26649     }
26650     
26651     
26652     Roo.View.superclass.constructor.call(this);
26653     
26654     
26655 };
26656
26657 Roo.extend(Roo.View, Roo.util.Observable, {
26658     
26659      /**
26660      * @cfg {Roo.data.Store} store Data store to load data from.
26661      */
26662     store : false,
26663     
26664     /**
26665      * @cfg {String|Roo.Element} el The container element.
26666      */
26667     el : '',
26668     
26669     /**
26670      * @cfg {String|Roo.Template} tpl The template used by this View 
26671      */
26672     tpl : false,
26673     /**
26674      * @cfg {String} dataName the named area of the template to use as the data area
26675      *                          Works with domtemplates roo-name="name"
26676      */
26677     dataName: false,
26678     /**
26679      * @cfg {String} selectedClass The css class to add to selected nodes
26680      */
26681     selectedClass : "x-view-selected",
26682      /**
26683      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26684      */
26685     emptyText : "",
26686     
26687     /**
26688      * @cfg {String} text to display on mask (default Loading)
26689      */
26690     mask : false,
26691     /**
26692      * @cfg {Boolean} multiSelect Allow multiple selection
26693      */
26694     multiSelect : false,
26695     /**
26696      * @cfg {Boolean} singleSelect Allow single selection
26697      */
26698     singleSelect:  false,
26699     
26700     /**
26701      * @cfg {Boolean} toggleSelect - selecting 
26702      */
26703     toggleSelect : false,
26704     
26705     /**
26706      * @cfg {Boolean} tickable - selecting 
26707      */
26708     tickable : false,
26709     
26710     /**
26711      * Returns the element this view is bound to.
26712      * @return {Roo.Element}
26713      */
26714     getEl : function(){
26715         return this.wrapEl;
26716     },
26717     
26718     
26719
26720     /**
26721      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26722      */
26723     refresh : function(){
26724         //Roo.log('refresh');
26725         var t = this.tpl;
26726         
26727         // if we are using something like 'domtemplate', then
26728         // the what gets used is:
26729         // t.applySubtemplate(NAME, data, wrapping data..)
26730         // the outer template then get' applied with
26731         //     the store 'extra data'
26732         // and the body get's added to the
26733         //      roo-name="data" node?
26734         //      <span class='roo-tpl-{name}'></span> ?????
26735         
26736         
26737         
26738         this.clearSelections();
26739         this.el.update("");
26740         var html = [];
26741         var records = this.store.getRange();
26742         if(records.length < 1) {
26743             
26744             // is this valid??  = should it render a template??
26745             
26746             this.el.update(this.emptyText);
26747             return;
26748         }
26749         var el = this.el;
26750         if (this.dataName) {
26751             this.el.update(t.apply(this.store.meta)); //????
26752             el = this.el.child('.roo-tpl-' + this.dataName);
26753         }
26754         
26755         for(var i = 0, len = records.length; i < len; i++){
26756             var data = this.prepareData(records[i].data, i, records[i]);
26757             this.fireEvent("preparedata", this, data, i, records[i]);
26758             
26759             var d = Roo.apply({}, data);
26760             
26761             if(this.tickable){
26762                 Roo.apply(d, {'roo-id' : Roo.id()});
26763                 
26764                 var _this = this;
26765             
26766                 Roo.each(this.parent.item, function(item){
26767                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26768                         return;
26769                     }
26770                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26771                 });
26772             }
26773             
26774             html[html.length] = Roo.util.Format.trim(
26775                 this.dataName ?
26776                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26777                     t.apply(d)
26778             );
26779         }
26780         
26781         
26782         
26783         el.update(html.join(""));
26784         this.nodes = el.dom.childNodes;
26785         this.updateIndexes(0);
26786     },
26787     
26788
26789     /**
26790      * Function to override to reformat the data that is sent to
26791      * the template for each node.
26792      * DEPRICATED - use the preparedata event handler.
26793      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26794      * a JSON object for an UpdateManager bound view).
26795      */
26796     prepareData : function(data, index, record)
26797     {
26798         this.fireEvent("preparedata", this, data, index, record);
26799         return data;
26800     },
26801
26802     onUpdate : function(ds, record){
26803         // Roo.log('on update');   
26804         this.clearSelections();
26805         var index = this.store.indexOf(record);
26806         var n = this.nodes[index];
26807         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26808         n.parentNode.removeChild(n);
26809         this.updateIndexes(index, index);
26810     },
26811
26812     
26813     
26814 // --------- FIXME     
26815     onAdd : function(ds, records, index)
26816     {
26817         //Roo.log(['on Add', ds, records, index] );        
26818         this.clearSelections();
26819         if(this.nodes.length == 0){
26820             this.refresh();
26821             return;
26822         }
26823         var n = this.nodes[index];
26824         for(var i = 0, len = records.length; i < len; i++){
26825             var d = this.prepareData(records[i].data, i, records[i]);
26826             if(n){
26827                 this.tpl.insertBefore(n, d);
26828             }else{
26829                 
26830                 this.tpl.append(this.el, d);
26831             }
26832         }
26833         this.updateIndexes(index);
26834     },
26835
26836     onRemove : function(ds, record, index){
26837        // Roo.log('onRemove');
26838         this.clearSelections();
26839         var el = this.dataName  ?
26840             this.el.child('.roo-tpl-' + this.dataName) :
26841             this.el; 
26842         
26843         el.dom.removeChild(this.nodes[index]);
26844         this.updateIndexes(index);
26845     },
26846
26847     /**
26848      * Refresh an individual node.
26849      * @param {Number} index
26850      */
26851     refreshNode : function(index){
26852         this.onUpdate(this.store, this.store.getAt(index));
26853     },
26854
26855     updateIndexes : function(startIndex, endIndex){
26856         var ns = this.nodes;
26857         startIndex = startIndex || 0;
26858         endIndex = endIndex || ns.length - 1;
26859         for(var i = startIndex; i <= endIndex; i++){
26860             ns[i].nodeIndex = i;
26861         }
26862     },
26863
26864     /**
26865      * Changes the data store this view uses and refresh the view.
26866      * @param {Store} store
26867      */
26868     setStore : function(store, initial){
26869         if(!initial && this.store){
26870             this.store.un("datachanged", this.refresh);
26871             this.store.un("add", this.onAdd);
26872             this.store.un("remove", this.onRemove);
26873             this.store.un("update", this.onUpdate);
26874             this.store.un("clear", this.refresh);
26875             this.store.un("beforeload", this.onBeforeLoad);
26876             this.store.un("load", this.onLoad);
26877             this.store.un("loadexception", this.onLoad);
26878         }
26879         if(store){
26880           
26881             store.on("datachanged", this.refresh, this);
26882             store.on("add", this.onAdd, this);
26883             store.on("remove", this.onRemove, this);
26884             store.on("update", this.onUpdate, this);
26885             store.on("clear", this.refresh, this);
26886             store.on("beforeload", this.onBeforeLoad, this);
26887             store.on("load", this.onLoad, this);
26888             store.on("loadexception", this.onLoad, this);
26889         }
26890         
26891         if(store){
26892             this.refresh();
26893         }
26894     },
26895     /**
26896      * onbeforeLoad - masks the loading area.
26897      *
26898      */
26899     onBeforeLoad : function(store,opts)
26900     {
26901          //Roo.log('onBeforeLoad');   
26902         if (!opts.add) {
26903             this.el.update("");
26904         }
26905         this.el.mask(this.mask ? this.mask : "Loading" ); 
26906     },
26907     onLoad : function ()
26908     {
26909         this.el.unmask();
26910     },
26911     
26912
26913     /**
26914      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26915      * @param {HTMLElement} node
26916      * @return {HTMLElement} The template node
26917      */
26918     findItemFromChild : function(node){
26919         var el = this.dataName  ?
26920             this.el.child('.roo-tpl-' + this.dataName,true) :
26921             this.el.dom; 
26922         
26923         if(!node || node.parentNode == el){
26924                     return node;
26925             }
26926             var p = node.parentNode;
26927             while(p && p != el){
26928             if(p.parentNode == el){
26929                 return p;
26930             }
26931             p = p.parentNode;
26932         }
26933             return null;
26934     },
26935
26936     /** @ignore */
26937     onClick : function(e){
26938         var item = this.findItemFromChild(e.getTarget());
26939         if(item){
26940             var index = this.indexOf(item);
26941             if(this.onItemClick(item, index, e) !== false){
26942                 this.fireEvent("click", this, index, item, e);
26943             }
26944         }else{
26945             this.clearSelections();
26946         }
26947     },
26948
26949     /** @ignore */
26950     onContextMenu : function(e){
26951         var item = this.findItemFromChild(e.getTarget());
26952         if(item){
26953             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26954         }
26955     },
26956
26957     /** @ignore */
26958     onDblClick : function(e){
26959         var item = this.findItemFromChild(e.getTarget());
26960         if(item){
26961             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
26962         }
26963     },
26964
26965     onItemClick : function(item, index, e)
26966     {
26967         if(this.fireEvent("beforeclick", this, index, item, e) === false){
26968             return false;
26969         }
26970         if (this.toggleSelect) {
26971             var m = this.isSelected(item) ? 'unselect' : 'select';
26972             //Roo.log(m);
26973             var _t = this;
26974             _t[m](item, true, false);
26975             return true;
26976         }
26977         if(this.multiSelect || this.singleSelect){
26978             if(this.multiSelect && e.shiftKey && this.lastSelection){
26979                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
26980             }else{
26981                 this.select(item, this.multiSelect && e.ctrlKey);
26982                 this.lastSelection = item;
26983             }
26984             
26985             if(!this.tickable){
26986                 e.preventDefault();
26987             }
26988             
26989         }
26990         return true;
26991     },
26992
26993     /**
26994      * Get the number of selected nodes.
26995      * @return {Number}
26996      */
26997     getSelectionCount : function(){
26998         return this.selections.length;
26999     },
27000
27001     /**
27002      * Get the currently selected nodes.
27003      * @return {Array} An array of HTMLElements
27004      */
27005     getSelectedNodes : function(){
27006         return this.selections;
27007     },
27008
27009     /**
27010      * Get the indexes of the selected nodes.
27011      * @return {Array}
27012      */
27013     getSelectedIndexes : function(){
27014         var indexes = [], s = this.selections;
27015         for(var i = 0, len = s.length; i < len; i++){
27016             indexes.push(s[i].nodeIndex);
27017         }
27018         return indexes;
27019     },
27020
27021     /**
27022      * Clear all selections
27023      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27024      */
27025     clearSelections : function(suppressEvent){
27026         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27027             this.cmp.elements = this.selections;
27028             this.cmp.removeClass(this.selectedClass);
27029             this.selections = [];
27030             if(!suppressEvent){
27031                 this.fireEvent("selectionchange", this, this.selections);
27032             }
27033         }
27034     },
27035
27036     /**
27037      * Returns true if the passed node is selected
27038      * @param {HTMLElement/Number} node The node or node index
27039      * @return {Boolean}
27040      */
27041     isSelected : function(node){
27042         var s = this.selections;
27043         if(s.length < 1){
27044             return false;
27045         }
27046         node = this.getNode(node);
27047         return s.indexOf(node) !== -1;
27048     },
27049
27050     /**
27051      * Selects nodes.
27052      * @param {Array/HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node, id of a template node or an array of any of those to select
27053      * @param {Boolean} keepExisting (optional) true to keep existing selections
27054      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27055      */
27056     select : function(nodeInfo, keepExisting, suppressEvent){
27057         if(nodeInfo instanceof Array){
27058             if(!keepExisting){
27059                 this.clearSelections(true);
27060             }
27061             for(var i = 0, len = nodeInfo.length; i < len; i++){
27062                 this.select(nodeInfo[i], true, true);
27063             }
27064             return;
27065         } 
27066         var node = this.getNode(nodeInfo);
27067         if(!node || this.isSelected(node)){
27068             return; // already selected.
27069         }
27070         if(!keepExisting){
27071             this.clearSelections(true);
27072         }
27073         
27074         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27075             Roo.fly(node).addClass(this.selectedClass);
27076             this.selections.push(node);
27077             if(!suppressEvent){
27078                 this.fireEvent("selectionchange", this, this.selections);
27079             }
27080         }
27081         
27082         
27083     },
27084       /**
27085      * Unselects nodes.
27086      * @param {Array/HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node, id of a template node or an array of any of those to select
27087      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27088      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27089      */
27090     unselect : function(nodeInfo, keepExisting, suppressEvent)
27091     {
27092         if(nodeInfo instanceof Array){
27093             Roo.each(this.selections, function(s) {
27094                 this.unselect(s, nodeInfo);
27095             }, this);
27096             return;
27097         }
27098         var node = this.getNode(nodeInfo);
27099         if(!node || !this.isSelected(node)){
27100             //Roo.log("not selected");
27101             return; // not selected.
27102         }
27103         // fireevent???
27104         var ns = [];
27105         Roo.each(this.selections, function(s) {
27106             if (s == node ) {
27107                 Roo.fly(node).removeClass(this.selectedClass);
27108
27109                 return;
27110             }
27111             ns.push(s);
27112         },this);
27113         
27114         this.selections= ns;
27115         this.fireEvent("selectionchange", this, this.selections);
27116     },
27117
27118     /**
27119      * Gets a template node.
27120      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27121      * @return {HTMLElement} The node or null if it wasn't found
27122      */
27123     getNode : function(nodeInfo){
27124         if(typeof nodeInfo == "string"){
27125             return document.getElementById(nodeInfo);
27126         }else if(typeof nodeInfo == "number"){
27127             return this.nodes[nodeInfo];
27128         }
27129         return nodeInfo;
27130     },
27131
27132     /**
27133      * Gets a range template nodes.
27134      * @param {Number} startIndex
27135      * @param {Number} endIndex
27136      * @return {Array} An array of nodes
27137      */
27138     getNodes : function(start, end){
27139         var ns = this.nodes;
27140         start = start || 0;
27141         end = typeof end == "undefined" ? ns.length - 1 : end;
27142         var nodes = [];
27143         if(start <= end){
27144             for(var i = start; i <= end; i++){
27145                 nodes.push(ns[i]);
27146             }
27147         } else{
27148             for(var i = start; i >= end; i--){
27149                 nodes.push(ns[i]);
27150             }
27151         }
27152         return nodes;
27153     },
27154
27155     /**
27156      * Finds the index of the passed node
27157      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27158      * @return {Number} The index of the node or -1
27159      */
27160     indexOf : function(node){
27161         node = this.getNode(node);
27162         if(typeof node.nodeIndex == "number"){
27163             return node.nodeIndex;
27164         }
27165         var ns = this.nodes;
27166         for(var i = 0, len = ns.length; i < len; i++){
27167             if(ns[i] == node){
27168                 return i;
27169             }
27170         }
27171         return -1;
27172     }
27173 });
27174 /*
27175  * Based on:
27176  * Ext JS Library 1.1.1
27177  * Copyright(c) 2006-2007, Ext JS, LLC.
27178  *
27179  * Originally Released Under LGPL - original licence link has changed is not relivant.
27180  *
27181  * Fork - LGPL
27182  * <script type="text/javascript">
27183  */
27184
27185 /**
27186  * @class Roo.JsonView
27187  * @extends Roo.View
27188  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27189 <pre><code>
27190 var view = new Roo.JsonView({
27191     container: "my-element",
27192     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27193     multiSelect: true, 
27194     jsonRoot: "data" 
27195 });
27196
27197 // listen for node click?
27198 view.on("click", function(vw, index, node, e){
27199     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27200 });
27201
27202 // direct load of JSON data
27203 view.load("foobar.php");
27204
27205 // Example from my blog list
27206 var tpl = new Roo.Template(
27207     '&lt;div class="entry"&gt;' +
27208     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27209     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27210     "&lt;/div&gt;&lt;hr /&gt;"
27211 );
27212
27213 var moreView = new Roo.JsonView({
27214     container :  "entry-list", 
27215     template : tpl,
27216     jsonRoot: "posts"
27217 });
27218 moreView.on("beforerender", this.sortEntries, this);
27219 moreView.load({
27220     url: "/blog/get-posts.php",
27221     params: "allposts=true",
27222     text: "Loading Blog Entries..."
27223 });
27224 </code></pre>
27225
27226 * Note: old code is supported with arguments : (container, template, config)
27227
27228
27229  * @constructor
27230  * Create a new JsonView
27231  * 
27232  * @param {Object} config The config object
27233  * 
27234  */
27235 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27236     
27237     
27238     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27239
27240     var um = this.el.getUpdateManager();
27241     um.setRenderer(this);
27242     um.on("update", this.onLoad, this);
27243     um.on("failure", this.onLoadException, this);
27244
27245     /**
27246      * @event beforerender
27247      * Fires before rendering of the downloaded JSON data.
27248      * @param {Roo.JsonView} this
27249      * @param {Object} data The JSON data loaded
27250      */
27251     /**
27252      * @event load
27253      * Fires when data is loaded.
27254      * @param {Roo.JsonView} this
27255      * @param {Object} data The JSON data loaded
27256      * @param {Object} response The raw Connect response object
27257      */
27258     /**
27259      * @event loadexception
27260      * Fires when loading fails.
27261      * @param {Roo.JsonView} this
27262      * @param {Object} response The raw Connect response object
27263      */
27264     this.addEvents({
27265         'beforerender' : true,
27266         'load' : true,
27267         'loadexception' : true
27268     });
27269 };
27270 Roo.extend(Roo.JsonView, Roo.View, {
27271     /**
27272      * @type {String} The root property in the loaded JSON object that contains the data
27273      */
27274     jsonRoot : "",
27275
27276     /**
27277      * Refreshes the view.
27278      */
27279     refresh : function(){
27280         this.clearSelections();
27281         this.el.update("");
27282         var html = [];
27283         var o = this.jsonData;
27284         if(o && o.length > 0){
27285             for(var i = 0, len = o.length; i < len; i++){
27286                 var data = this.prepareData(o[i], i, o);
27287                 html[html.length] = this.tpl.apply(data);
27288             }
27289         }else{
27290             html.push(this.emptyText);
27291         }
27292         this.el.update(html.join(""));
27293         this.nodes = this.el.dom.childNodes;
27294         this.updateIndexes(0);
27295     },
27296
27297     /**
27298      * Performs an async HTTP request, and loads the JSON from the response. If <i>params</i> are specified it uses POST, otherwise it uses GET.
27299      * @param {Object/String/Function} url The URL for this request, or a function to call to get the URL, or a config object containing any of the following options:
27300      <pre><code>
27301      view.load({
27302          url: "your-url.php",
27303          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27304          callback: yourFunction,
27305          scope: yourObject, //(optional scope)
27306          discardUrl: false,
27307          nocache: false,
27308          text: "Loading...",
27309          timeout: 30,
27310          scripts: false
27311      });
27312      </code></pre>
27313      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27314      * are respectively shorthand for <i>disableCaching</i>, <i>indicatorText</i>, and <i>loadScripts</i> and are used to set their associated property on this UpdateManager instance.
27315      * @param {String/Object} params (optional) The parameters to pass, as either a URL encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
27316      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27317      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used URL. If true, it will not store the URL.
27318      */
27319     load : function(){
27320         var um = this.el.getUpdateManager();
27321         um.update.apply(um, arguments);
27322     },
27323
27324     // note - render is a standard framework call...
27325     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27326     render : function(el, response){
27327         
27328         this.clearSelections();
27329         this.el.update("");
27330         var o;
27331         try{
27332             if (response != '') {
27333                 o = Roo.util.JSON.decode(response.responseText);
27334                 if(this.jsonRoot){
27335                     
27336                     o = o[this.jsonRoot];
27337                 }
27338             }
27339         } catch(e){
27340         }
27341         /**
27342          * The current JSON data or null
27343          */
27344         this.jsonData = o;
27345         this.beforeRender();
27346         this.refresh();
27347     },
27348
27349 /**
27350  * Get the number of records in the current JSON dataset
27351  * @return {Number}
27352  */
27353     getCount : function(){
27354         return this.jsonData ? this.jsonData.length : 0;
27355     },
27356
27357 /**
27358  * Returns the JSON object for the specified node(s)
27359  * @param {HTMLElement/Array} node The node or an array of nodes
27360  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27361  * you get the JSON object for the node
27362  */
27363     getNodeData : function(node){
27364         if(node instanceof Array){
27365             var data = [];
27366             for(var i = 0, len = node.length; i < len; i++){
27367                 data.push(this.getNodeData(node[i]));
27368             }
27369             return data;
27370         }
27371         return this.jsonData[this.indexOf(node)] || null;
27372     },
27373
27374     beforeRender : function(){
27375         this.snapshot = this.jsonData;
27376         if(this.sortInfo){
27377             this.sort.apply(this, this.sortInfo);
27378         }
27379         this.fireEvent("beforerender", this, this.jsonData);
27380     },
27381
27382     onLoad : function(el, o){
27383         this.fireEvent("load", this, this.jsonData, o);
27384     },
27385
27386     onLoadException : function(el, o){
27387         this.fireEvent("loadexception", this, o);
27388     },
27389
27390 /**
27391  * Filter the data by a specific property.
27392  * @param {String} property A property on your JSON objects
27393  * @param {String/RegExp} value Either string that the property values
27394  * should start with, or a RegExp to test against the property
27395  */
27396     filter : function(property, value){
27397         if(this.jsonData){
27398             var data = [];
27399             var ss = this.snapshot;
27400             if(typeof value == "string"){
27401                 var vlen = value.length;
27402                 if(vlen == 0){
27403                     this.clearFilter();
27404                     return;
27405                 }
27406                 value = value.toLowerCase();
27407                 for(var i = 0, len = ss.length; i < len; i++){
27408                     var o = ss[i];
27409                     if(o[property].substr(0, vlen).toLowerCase() == value){
27410                         data.push(o);
27411                     }
27412                 }
27413             } else if(value.exec){ // regex?
27414                 for(var i = 0, len = ss.length; i < len; i++){
27415                     var o = ss[i];
27416                     if(value.test(o[property])){
27417                         data.push(o);
27418                     }
27419                 }
27420             } else{
27421                 return;
27422             }
27423             this.jsonData = data;
27424             this.refresh();
27425         }
27426     },
27427
27428 /**
27429  * Filter by a function. The passed function will be called with each
27430  * object in the current dataset. If the function returns true the value is kept,
27431  * otherwise it is filtered.
27432  * @param {Function} fn
27433  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27434  */
27435     filterBy : function(fn, scope){
27436         if(this.jsonData){
27437             var data = [];
27438             var ss = this.snapshot;
27439             for(var i = 0, len = ss.length; i < len; i++){
27440                 var o = ss[i];
27441                 if(fn.call(scope || this, o)){
27442                     data.push(o);
27443                 }
27444             }
27445             this.jsonData = data;
27446             this.refresh();
27447         }
27448     },
27449
27450 /**
27451  * Clears the current filter.
27452  */
27453     clearFilter : function(){
27454         if(this.snapshot && this.jsonData != this.snapshot){
27455             this.jsonData = this.snapshot;
27456             this.refresh();
27457         }
27458     },
27459
27460
27461 /**
27462  * Sorts the data for this view and refreshes it.
27463  * @param {String} property A property on your JSON objects to sort on
27464  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27465  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27466  */
27467     sort : function(property, dir, sortType){
27468         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27469         if(this.jsonData){
27470             var p = property;
27471             var dsc = dir && dir.toLowerCase() == "desc";
27472             var f = function(o1, o2){
27473                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27474                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27475                 ;
27476                 if(v1 < v2){
27477                     return dsc ? +1 : -1;
27478                 } else if(v1 > v2){
27479                     return dsc ? -1 : +1;
27480                 } else{
27481                     return 0;
27482                 }
27483             };
27484             this.jsonData.sort(f);
27485             this.refresh();
27486             if(this.jsonData != this.snapshot){
27487                 this.snapshot.sort(f);
27488             }
27489         }
27490     }
27491 });/*
27492  * Based on:
27493  * Ext JS Library 1.1.1
27494  * Copyright(c) 2006-2007, Ext JS, LLC.
27495  *
27496  * Originally Released Under LGPL - original licence link has changed is not relivant.
27497  *
27498  * Fork - LGPL
27499  * <script type="text/javascript">
27500  */
27501  
27502
27503 /**
27504  * @class Roo.ColorPalette
27505  * @extends Roo.Component
27506  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27507  * Here's an example of typical usage:
27508  * <pre><code>
27509 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27510 cp.render('my-div');
27511
27512 cp.on('select', function(palette, selColor){
27513     // do something with selColor
27514 });
27515 </code></pre>
27516  * @constructor
27517  * Create a new ColorPalette
27518  * @param {Object} config The config object
27519  */
27520 Roo.ColorPalette = function(config){
27521     Roo.ColorPalette.superclass.constructor.call(this, config);
27522     this.addEvents({
27523         /**
27524              * @event select
27525              * Fires when a color is selected
27526              * @param {ColorPalette} this
27527              * @param {String} color The 6-digit color hex code (without the # symbol)
27528              */
27529         select: true
27530     });
27531
27532     if(this.handler){
27533         this.on("select", this.handler, this.scope, true);
27534     }
27535 };
27536 Roo.extend(Roo.ColorPalette, Roo.Component, {
27537     /**
27538      * @cfg {String} itemCls
27539      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27540      */
27541     itemCls : "x-color-palette",
27542     /**
27543      * @cfg {String} value
27544      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27545      * the hex codes are case-sensitive.
27546      */
27547     value : null,
27548     clickEvent:'click',
27549     // private
27550     ctype: "Roo.ColorPalette",
27551
27552     /**
27553      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27554      */
27555     allowReselect : false,
27556
27557     /**
27558      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27559      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27560      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27561      * of colors with the width setting until the box is symmetrical.</p>
27562      * <p>You can override individual colors if needed:</p>
27563      * <pre><code>
27564 var cp = new Roo.ColorPalette();
27565 cp.colors[0] = "FF0000";  // change the first box to red
27566 </code></pre>
27567
27568 Or you can provide a custom array of your own for complete control:
27569 <pre><code>
27570 var cp = new Roo.ColorPalette();
27571 cp.colors = ["000000", "993300", "333300"];
27572 </code></pre>
27573      * @type Array
27574      */
27575     colors : [
27576         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27577         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27578         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27579         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27580         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27581     ],
27582
27583     // private
27584     onRender : function(container, position){
27585         var t = new Roo.MasterTemplate(
27586             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27587         );
27588         var c = this.colors;
27589         for(var i = 0, len = c.length; i < len; i++){
27590             t.add([c[i]]);
27591         }
27592         var el = document.createElement("div");
27593         el.className = this.itemCls;
27594         t.overwrite(el);
27595         container.dom.insertBefore(el, position);
27596         this.el = Roo.get(el);
27597         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27598         if(this.clickEvent != 'click'){
27599             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27600         }
27601     },
27602
27603     // private
27604     afterRender : function(){
27605         Roo.ColorPalette.superclass.afterRender.call(this);
27606         if(this.value){
27607             var s = this.value;
27608             this.value = null;
27609             this.select(s);
27610         }
27611     },
27612
27613     // private
27614     handleClick : function(e, t){
27615         e.preventDefault();
27616         if(!this.disabled){
27617             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27618             this.select(c.toUpperCase());
27619         }
27620     },
27621
27622     /**
27623      * Selects the specified color in the palette (fires the select event)
27624      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27625      */
27626     select : function(color){
27627         color = color.replace("#", "");
27628         if(color != this.value || this.allowReselect){
27629             var el = this.el;
27630             if(this.value){
27631                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27632             }
27633             el.child("a.color-"+color).addClass("x-color-palette-sel");
27634             this.value = color;
27635             this.fireEvent("select", this, color);
27636         }
27637     }
27638 });/*
27639  * Based on:
27640  * Ext JS Library 1.1.1
27641  * Copyright(c) 2006-2007, Ext JS, LLC.
27642  *
27643  * Originally Released Under LGPL - original licence link has changed is not relivant.
27644  *
27645  * Fork - LGPL
27646  * <script type="text/javascript">
27647  */
27648  
27649 /**
27650  * @class Roo.DatePicker
27651  * @extends Roo.Component
27652  * Simple date picker class.
27653  * @constructor
27654  * Create a new DatePicker
27655  * @param {Object} config The config object
27656  */
27657 Roo.DatePicker = function(config){
27658     Roo.DatePicker.superclass.constructor.call(this, config);
27659
27660     this.value = config && config.value ?
27661                  config.value.clearTime() : new Date().clearTime();
27662
27663     this.addEvents({
27664         /**
27665              * @event select
27666              * Fires when a date is selected
27667              * @param {DatePicker} this
27668              * @param {Date} date The selected date
27669              */
27670         'select': true,
27671         /**
27672              * @event monthchange
27673              * Fires when the displayed month changes 
27674              * @param {DatePicker} this
27675              * @param {Date} date The selected month
27676              */
27677         'monthchange': true
27678     });
27679
27680     if(this.handler){
27681         this.on("select", this.handler,  this.scope || this);
27682     }
27683     // build the disabledDatesRE
27684     if(!this.disabledDatesRE && this.disabledDates){
27685         var dd = this.disabledDates;
27686         var re = "(?:";
27687         for(var i = 0; i < dd.length; i++){
27688             re += dd[i];
27689             if(i != dd.length-1) {
27690                 re += "|";
27691             }
27692         }
27693         this.disabledDatesRE = new RegExp(re + ")");
27694     }
27695 };
27696
27697 Roo.extend(Roo.DatePicker, Roo.Component, {
27698     /**
27699      * @cfg {String} todayText
27700      * The text to display on the button that selects the current date (defaults to "Today")
27701      */
27702     todayText : "Today",
27703     /**
27704      * @cfg {String} okText
27705      * The text to display on the ok button
27706      */
27707     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27708     /**
27709      * @cfg {String} cancelText
27710      * The text to display on the cancel button
27711      */
27712     cancelText : "Cancel",
27713     /**
27714      * @cfg {String} todayTip
27715      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27716      */
27717     todayTip : "{0} (Spacebar)",
27718     /**
27719      * @cfg {Date} minDate
27720      * Minimum allowable date (JavaScript date object, defaults to null)
27721      */
27722     minDate : null,
27723     /**
27724      * @cfg {Date} maxDate
27725      * Maximum allowable date (JavaScript date object, defaults to null)
27726      */
27727     maxDate : null,
27728     /**
27729      * @cfg {String} minText
27730      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27731      */
27732     minText : "This date is before the minimum date",
27733     /**
27734      * @cfg {String} maxText
27735      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27736      */
27737     maxText : "This date is after the maximum date",
27738     /**
27739      * @cfg {String} format
27740      * The default date format string which can be overriden for localization support.  The format must be
27741      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27742      */
27743     format : "m/d/y",
27744     /**
27745      * @cfg {Array} disabledDays
27746      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27747      */
27748     disabledDays : null,
27749     /**
27750      * @cfg {String} disabledDaysText
27751      * The tooltip to display when the date falls on a disabled day (defaults to "")
27752      */
27753     disabledDaysText : "",
27754     /**
27755      * @cfg {RegExp} disabledDatesRE
27756      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27757      */
27758     disabledDatesRE : null,
27759     /**
27760      * @cfg {String} disabledDatesText
27761      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27762      */
27763     disabledDatesText : "",
27764     /**
27765      * @cfg {Boolean} constrainToViewport
27766      * True to constrain the date picker to the viewport (defaults to true)
27767      */
27768     constrainToViewport : true,
27769     /**
27770      * @cfg {Array} monthNames
27771      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27772      */
27773     monthNames : Date.monthNames,
27774     /**
27775      * @cfg {Array} dayNames
27776      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27777      */
27778     dayNames : Date.dayNames,
27779     /**
27780      * @cfg {String} nextText
27781      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27782      */
27783     nextText: 'Next Month (Control+Right)',
27784     /**
27785      * @cfg {String} prevText
27786      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27787      */
27788     prevText: 'Previous Month (Control+Left)',
27789     /**
27790      * @cfg {String} monthYearText
27791      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27792      */
27793     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27794     /**
27795      * @cfg {Number} startDay
27796      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27797      */
27798     startDay : 0,
27799     /**
27800      * @cfg {Bool} showClear
27801      * Show a clear button (usefull for date form elements that can be blank.)
27802      */
27803     
27804     showClear: false,
27805     
27806     /**
27807      * Sets the value of the date field
27808      * @param {Date} value The date to set
27809      */
27810     setValue : function(value){
27811         var old = this.value;
27812         
27813         if (typeof(value) == 'string') {
27814          
27815             value = Date.parseDate(value, this.format);
27816         }
27817         if (!value) {
27818             value = new Date();
27819         }
27820         
27821         this.value = value.clearTime(true);
27822         if(this.el){
27823             this.update(this.value);
27824         }
27825     },
27826
27827     /**
27828      * Gets the current selected value of the date field
27829      * @return {Date} The selected date
27830      */
27831     getValue : function(){
27832         return this.value;
27833     },
27834
27835     // private
27836     focus : function(){
27837         if(this.el){
27838             this.update(this.activeDate);
27839         }
27840     },
27841
27842     // privateval
27843     onRender : function(container, position){
27844         
27845         var m = [
27846              '<table cellspacing="0">',
27847                 '<tr><td class="x-date-left"><a href="#" title="', this.prevText ,'">&#160;</a></td><td class="x-date-middle" align="center"></td><td class="x-date-right"><a href="#" title="', this.nextText ,'">&#160;</a></td></tr>',
27848                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27849         var dn = this.dayNames;
27850         for(var i = 0; i < 7; i++){
27851             var d = this.startDay+i;
27852             if(d > 6){
27853                 d = d-7;
27854             }
27855             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27856         }
27857         m[m.length] = "</tr></thead><tbody><tr>";
27858         for(var i = 0; i < 42; i++) {
27859             if(i % 7 == 0 && i != 0){
27860                 m[m.length] = "</tr><tr>";
27861             }
27862             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27863         }
27864         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27865             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27866
27867         var el = document.createElement("div");
27868         el.className = "x-date-picker";
27869         el.innerHTML = m.join("");
27870
27871         container.dom.insertBefore(el, position);
27872
27873         this.el = Roo.get(el);
27874         this.eventEl = Roo.get(el.firstChild);
27875
27876         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27877             handler: this.showPrevMonth,
27878             scope: this,
27879             preventDefault:true,
27880             stopDefault:true
27881         });
27882
27883         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27884             handler: this.showNextMonth,
27885             scope: this,
27886             preventDefault:true,
27887             stopDefault:true
27888         });
27889
27890         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27891
27892         this.monthPicker = this.el.down('div.x-date-mp');
27893         this.monthPicker.enableDisplayMode('block');
27894         
27895         var kn = new Roo.KeyNav(this.eventEl, {
27896             "left" : function(e){
27897                 e.ctrlKey ?
27898                     this.showPrevMonth() :
27899                     this.update(this.activeDate.add("d", -1));
27900             },
27901
27902             "right" : function(e){
27903                 e.ctrlKey ?
27904                     this.showNextMonth() :
27905                     this.update(this.activeDate.add("d", 1));
27906             },
27907
27908             "up" : function(e){
27909                 e.ctrlKey ?
27910                     this.showNextYear() :
27911                     this.update(this.activeDate.add("d", -7));
27912             },
27913
27914             "down" : function(e){
27915                 e.ctrlKey ?
27916                     this.showPrevYear() :
27917                     this.update(this.activeDate.add("d", 7));
27918             },
27919
27920             "pageUp" : function(e){
27921                 this.showNextMonth();
27922             },
27923
27924             "pageDown" : function(e){
27925                 this.showPrevMonth();
27926             },
27927
27928             "enter" : function(e){
27929                 e.stopPropagation();
27930                 return true;
27931             },
27932
27933             scope : this
27934         });
27935
27936         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27937
27938         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27939
27940         this.el.unselectable();
27941         
27942         this.cells = this.el.select("table.x-date-inner tbody td");
27943         this.textNodes = this.el.query("table.x-date-inner tbody span");
27944
27945         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27946             text: "&#160;",
27947             tooltip: this.monthYearText
27948         });
27949
27950         this.mbtn.on('click', this.showMonthPicker, this);
27951         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27952
27953
27954         var today = (new Date()).dateFormat(this.format);
27955         
27956         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
27957         if (this.showClear) {
27958             baseTb.add( new Roo.Toolbar.Fill());
27959         }
27960         baseTb.add({
27961             text: String.format(this.todayText, today),
27962             tooltip: String.format(this.todayTip, today),
27963             handler: this.selectToday,
27964             scope: this
27965         });
27966         
27967         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
27968             
27969         //});
27970         if (this.showClear) {
27971             
27972             baseTb.add( new Roo.Toolbar.Fill());
27973             baseTb.add({
27974                 text: '&#160;',
27975                 cls: 'x-btn-icon x-btn-clear',
27976                 handler: function() {
27977                     //this.value = '';
27978                     this.fireEvent("select", this, '');
27979                 },
27980                 scope: this
27981             });
27982         }
27983         
27984         
27985         if(Roo.isIE){
27986             this.el.repaint();
27987         }
27988         this.update(this.value);
27989     },
27990
27991     createMonthPicker : function(){
27992         if(!this.monthPicker.dom.firstChild){
27993             var buf = ['<table border="0" cellspacing="0">'];
27994             for(var i = 0; i < 6; i++){
27995                 buf.push(
27996                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
27997                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
27998                     i == 0 ?
27999                     '<td class="x-date-mp-ybtn" align="center"><a class="x-date-mp-prev"></a></td><td class="x-date-mp-ybtn" align="center"><a class="x-date-mp-next"></a></td></tr>' :
28000                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28001                 );
28002             }
28003             buf.push(
28004                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28005                     this.okText,
28006                     '</button><button type="button" class="x-date-mp-cancel">',
28007                     this.cancelText,
28008                     '</button></td></tr>',
28009                 '</table>'
28010             );
28011             this.monthPicker.update(buf.join(''));
28012             this.monthPicker.on('click', this.onMonthClick, this);
28013             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28014
28015             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28016             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28017
28018             this.mpMonths.each(function(m, a, i){
28019                 i += 1;
28020                 if((i%2) == 0){
28021                     m.dom.xmonth = 5 + Math.round(i * .5);
28022                 }else{
28023                     m.dom.xmonth = Math.round((i-1) * .5);
28024                 }
28025             });
28026         }
28027     },
28028
28029     showMonthPicker : function(){
28030         this.createMonthPicker();
28031         var size = this.el.getSize();
28032         this.monthPicker.setSize(size);
28033         this.monthPicker.child('table').setSize(size);
28034
28035         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28036         this.updateMPMonth(this.mpSelMonth);
28037         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28038         this.updateMPYear(this.mpSelYear);
28039
28040         this.monthPicker.slideIn('t', {duration:.2});
28041     },
28042
28043     updateMPYear : function(y){
28044         this.mpyear = y;
28045         var ys = this.mpYears.elements;
28046         for(var i = 1; i <= 10; i++){
28047             var td = ys[i-1], y2;
28048             if((i%2) == 0){
28049                 y2 = y + Math.round(i * .5);
28050                 td.firstChild.innerHTML = y2;
28051                 td.xyear = y2;
28052             }else{
28053                 y2 = y - (5-Math.round(i * .5));
28054                 td.firstChild.innerHTML = y2;
28055                 td.xyear = y2;
28056             }
28057             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28058         }
28059     },
28060
28061     updateMPMonth : function(sm){
28062         this.mpMonths.each(function(m, a, i){
28063             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28064         });
28065     },
28066
28067     selectMPMonth: function(m){
28068         
28069     },
28070
28071     onMonthClick : function(e, t){
28072         e.stopEvent();
28073         var el = new Roo.Element(t), pn;
28074         if(el.is('button.x-date-mp-cancel')){
28075             this.hideMonthPicker();
28076         }
28077         else if(el.is('button.x-date-mp-ok')){
28078             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28079             this.hideMonthPicker();
28080         }
28081         else if(pn = el.up('td.x-date-mp-month', 2)){
28082             this.mpMonths.removeClass('x-date-mp-sel');
28083             pn.addClass('x-date-mp-sel');
28084             this.mpSelMonth = pn.dom.xmonth;
28085         }
28086         else if(pn = el.up('td.x-date-mp-year', 2)){
28087             this.mpYears.removeClass('x-date-mp-sel');
28088             pn.addClass('x-date-mp-sel');
28089             this.mpSelYear = pn.dom.xyear;
28090         }
28091         else if(el.is('a.x-date-mp-prev')){
28092             this.updateMPYear(this.mpyear-10);
28093         }
28094         else if(el.is('a.x-date-mp-next')){
28095             this.updateMPYear(this.mpyear+10);
28096         }
28097     },
28098
28099     onMonthDblClick : function(e, t){
28100         e.stopEvent();
28101         var el = new Roo.Element(t), pn;
28102         if(pn = el.up('td.x-date-mp-month', 2)){
28103             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28104             this.hideMonthPicker();
28105         }
28106         else if(pn = el.up('td.x-date-mp-year', 2)){
28107             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28108             this.hideMonthPicker();
28109         }
28110     },
28111
28112     hideMonthPicker : function(disableAnim){
28113         if(this.monthPicker){
28114             if(disableAnim === true){
28115                 this.monthPicker.hide();
28116             }else{
28117                 this.monthPicker.slideOut('t', {duration:.2});
28118             }
28119         }
28120     },
28121
28122     // private
28123     showPrevMonth : function(e){
28124         this.update(this.activeDate.add("mo", -1));
28125     },
28126
28127     // private
28128     showNextMonth : function(e){
28129         this.update(this.activeDate.add("mo", 1));
28130     },
28131
28132     // private
28133     showPrevYear : function(){
28134         this.update(this.activeDate.add("y", -1));
28135     },
28136
28137     // private
28138     showNextYear : function(){
28139         this.update(this.activeDate.add("y", 1));
28140     },
28141
28142     // private
28143     handleMouseWheel : function(e){
28144         var delta = e.getWheelDelta();
28145         if(delta > 0){
28146             this.showPrevMonth();
28147             e.stopEvent();
28148         } else if(delta < 0){
28149             this.showNextMonth();
28150             e.stopEvent();
28151         }
28152     },
28153
28154     // private
28155     handleDateClick : function(e, t){
28156         e.stopEvent();
28157         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28158             this.setValue(new Date(t.dateValue));
28159             this.fireEvent("select", this, this.value);
28160         }
28161     },
28162
28163     // private
28164     selectToday : function(){
28165         this.setValue(new Date().clearTime());
28166         this.fireEvent("select", this, this.value);
28167     },
28168
28169     // private
28170     update : function(date)
28171     {
28172         var vd = this.activeDate;
28173         this.activeDate = date;
28174         if(vd && this.el){
28175             var t = date.getTime();
28176             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28177                 this.cells.removeClass("x-date-selected");
28178                 this.cells.each(function(c){
28179                    if(c.dom.firstChild.dateValue == t){
28180                        c.addClass("x-date-selected");
28181                        setTimeout(function(){
28182                             try{c.dom.firstChild.focus();}catch(e){}
28183                        }, 50);
28184                        return false;
28185                    }
28186                 });
28187                 return;
28188             }
28189         }
28190         
28191         var days = date.getDaysInMonth();
28192         var firstOfMonth = date.getFirstDateOfMonth();
28193         var startingPos = firstOfMonth.getDay()-this.startDay;
28194
28195         if(startingPos <= this.startDay){
28196             startingPos += 7;
28197         }
28198
28199         var pm = date.add("mo", -1);
28200         var prevStart = pm.getDaysInMonth()-startingPos;
28201
28202         var cells = this.cells.elements;
28203         var textEls = this.textNodes;
28204         days += startingPos;
28205
28206         // convert everything to numbers so it's fast
28207         var day = 86400000;
28208         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28209         var today = new Date().clearTime().getTime();
28210         var sel = date.clearTime().getTime();
28211         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28212         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28213         var ddMatch = this.disabledDatesRE;
28214         var ddText = this.disabledDatesText;
28215         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28216         var ddaysText = this.disabledDaysText;
28217         var format = this.format;
28218
28219         var setCellClass = function(cal, cell){
28220             cell.title = "";
28221             var t = d.getTime();
28222             cell.firstChild.dateValue = t;
28223             if(t == today){
28224                 cell.className += " x-date-today";
28225                 cell.title = cal.todayText;
28226             }
28227             if(t == sel){
28228                 cell.className += " x-date-selected";
28229                 setTimeout(function(){
28230                     try{cell.firstChild.focus();}catch(e){}
28231                 }, 50);
28232             }
28233             // disabling
28234             if(t < min) {
28235                 cell.className = " x-date-disabled";
28236                 cell.title = cal.minText;
28237                 return;
28238             }
28239             if(t > max) {
28240                 cell.className = " x-date-disabled";
28241                 cell.title = cal.maxText;
28242                 return;
28243             }
28244             if(ddays){
28245                 if(ddays.indexOf(d.getDay()) != -1){
28246                     cell.title = ddaysText;
28247                     cell.className = " x-date-disabled";
28248                 }
28249             }
28250             if(ddMatch && format){
28251                 var fvalue = d.dateFormat(format);
28252                 if(ddMatch.test(fvalue)){
28253                     cell.title = ddText.replace("%0", fvalue);
28254                     cell.className = " x-date-disabled";
28255                 }
28256             }
28257         };
28258
28259         var i = 0;
28260         for(; i < startingPos; i++) {
28261             textEls[i].innerHTML = (++prevStart);
28262             d.setDate(d.getDate()+1);
28263             cells[i].className = "x-date-prevday";
28264             setCellClass(this, cells[i]);
28265         }
28266         for(; i < days; i++){
28267             intDay = i - startingPos + 1;
28268             textEls[i].innerHTML = (intDay);
28269             d.setDate(d.getDate()+1);
28270             cells[i].className = "x-date-active";
28271             setCellClass(this, cells[i]);
28272         }
28273         var extraDays = 0;
28274         for(; i < 42; i++) {
28275              textEls[i].innerHTML = (++extraDays);
28276              d.setDate(d.getDate()+1);
28277              cells[i].className = "x-date-nextday";
28278              setCellClass(this, cells[i]);
28279         }
28280
28281         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28282         this.fireEvent('monthchange', this, date);
28283         
28284         if(!this.internalRender){
28285             var main = this.el.dom.firstChild;
28286             var w = main.offsetWidth;
28287             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28288             Roo.fly(main).setWidth(w);
28289             this.internalRender = true;
28290             // opera does not respect the auto grow header center column
28291             // then, after it gets a width opera refuses to recalculate
28292             // without a second pass
28293             if(Roo.isOpera && !this.secondPass){
28294                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28295                 this.secondPass = true;
28296                 this.update.defer(10, this, [date]);
28297             }
28298         }
28299         
28300         
28301     }
28302 });        /*
28303  * Based on:
28304  * Ext JS Library 1.1.1
28305  * Copyright(c) 2006-2007, Ext JS, LLC.
28306  *
28307  * Originally Released Under LGPL - original licence link has changed is not relivant.
28308  *
28309  * Fork - LGPL
28310  * <script type="text/javascript">
28311  */
28312 /**
28313  * @class Roo.TabPanel
28314  * @extends Roo.util.Observable
28315  * A lightweight tab container.
28316  * <br><br>
28317  * Usage:
28318  * <pre><code>
28319 // basic tabs 1, built from existing content
28320 var tabs = new Roo.TabPanel("tabs1");
28321 tabs.addTab("script", "View Script");
28322 tabs.addTab("markup", "View Markup");
28323 tabs.activate("script");
28324
28325 // more advanced tabs, built from javascript
28326 var jtabs = new Roo.TabPanel("jtabs");
28327 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28328
28329 // set up the UpdateManager
28330 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28331 var updater = tab2.getUpdateManager();
28332 updater.setDefaultUrl("ajax1.htm");
28333 tab2.on('activate', updater.refresh, updater, true);
28334
28335 // Use setUrl for Ajax loading
28336 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28337 tab3.setUrl("ajax2.htm", null, true);
28338
28339 // Disabled tab
28340 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28341 tab4.disable();
28342
28343 jtabs.activate("jtabs-1");
28344  * </code></pre>
28345  * @constructor
28346  * Create a new TabPanel.
28347  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28348  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28349  */
28350 Roo.TabPanel = function(container, config){
28351     /**
28352     * The container element for this TabPanel.
28353     * @type Roo.Element
28354     */
28355     this.el = Roo.get(container, true);
28356     if(config){
28357         if(typeof config == "boolean"){
28358             this.tabPosition = config ? "bottom" : "top";
28359         }else{
28360             Roo.apply(this, config);
28361         }
28362     }
28363     if(this.tabPosition == "bottom"){
28364         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28365         this.el.addClass("x-tabs-bottom");
28366     }
28367     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28368     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28369     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28370     if(Roo.isIE){
28371         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28372     }
28373     if(this.tabPosition != "bottom"){
28374         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28375          * @type Roo.Element
28376          */
28377         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28378         this.el.addClass("x-tabs-top");
28379     }
28380     this.items = [];
28381
28382     this.bodyEl.setStyle("position", "relative");
28383
28384     this.active = null;
28385     this.activateDelegate = this.activate.createDelegate(this);
28386
28387     this.addEvents({
28388         /**
28389          * @event tabchange
28390          * Fires when the active tab changes
28391          * @param {Roo.TabPanel} this
28392          * @param {Roo.TabPanelItem} activePanel The new active tab
28393          */
28394         "tabchange": true,
28395         /**
28396          * @event beforetabchange
28397          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28398          * @param {Roo.TabPanel} this
28399          * @param {Object} e Set cancel to true on this object to cancel the tab change
28400          * @param {Roo.TabPanelItem} tab The tab being changed to
28401          */
28402         "beforetabchange" : true
28403     });
28404
28405     Roo.EventManager.onWindowResize(this.onResize, this);
28406     this.cpad = this.el.getPadding("lr");
28407     this.hiddenCount = 0;
28408
28409
28410     // toolbar on the tabbar support...
28411     if (this.toolbar) {
28412         var tcfg = this.toolbar;
28413         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28414         this.toolbar = new Roo.Toolbar(tcfg);
28415         if (Roo.isSafari) {
28416             var tbl = tcfg.container.child('table', true);
28417             tbl.setAttribute('width', '100%');
28418         }
28419         
28420     }
28421    
28422
28423
28424     Roo.TabPanel.superclass.constructor.call(this);
28425 };
28426
28427 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28428     /*
28429      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28430      */
28431     tabPosition : "top",
28432     /*
28433      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28434      */
28435     currentTabWidth : 0,
28436     /*
28437      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28438      */
28439     minTabWidth : 40,
28440     /*
28441      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28442      */
28443     maxTabWidth : 250,
28444     /*
28445      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28446      */
28447     preferredTabWidth : 175,
28448     /*
28449      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28450      */
28451     resizeTabs : false,
28452     /*
28453      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28454      */
28455     monitorResize : true,
28456     /*
28457      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28458      */
28459     toolbar : false,
28460
28461     /**
28462      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28463      * @param {String} id The id of the div to use <b>or create</b>
28464      * @param {String} text The text for the tab
28465      * @param {String} content (optional) Content to put in the TabPanelItem body
28466      * @param {Boolean} closable (optional) True to create a close icon on the tab
28467      * @return {Roo.TabPanelItem} The created TabPanelItem
28468      */
28469     addTab : function(id, text, content, closable){
28470         var item = new Roo.TabPanelItem(this, id, text, closable);
28471         this.addTabItem(item);
28472         if(content){
28473             item.setContent(content);
28474         }
28475         return item;
28476     },
28477
28478     /**
28479      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28480      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28481      * @return {Roo.TabPanelItem}
28482      */
28483     getTab : function(id){
28484         return this.items[id];
28485     },
28486
28487     /**
28488      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28489      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28490      */
28491     hideTab : function(id){
28492         var t = this.items[id];
28493         if(!t.isHidden()){
28494            t.setHidden(true);
28495            this.hiddenCount++;
28496            this.autoSizeTabs();
28497         }
28498     },
28499
28500     /**
28501      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28502      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28503      */
28504     unhideTab : function(id){
28505         var t = this.items[id];
28506         if(t.isHidden()){
28507            t.setHidden(false);
28508            this.hiddenCount--;
28509            this.autoSizeTabs();
28510         }
28511     },
28512
28513     /**
28514      * Adds an existing {@link Roo.TabPanelItem}.
28515      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28516      */
28517     addTabItem : function(item){
28518         this.items[item.id] = item;
28519         this.items.push(item);
28520         if(this.resizeTabs){
28521            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28522            this.autoSizeTabs();
28523         }else{
28524             item.autoSize();
28525         }
28526     },
28527
28528     /**
28529      * Removes a {@link Roo.TabPanelItem}.
28530      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28531      */
28532     removeTab : function(id){
28533         var items = this.items;
28534         var tab = items[id];
28535         if(!tab) { return; }
28536         var index = items.indexOf(tab);
28537         if(this.active == tab && items.length > 1){
28538             var newTab = this.getNextAvailable(index);
28539             if(newTab) {
28540                 newTab.activate();
28541             }
28542         }
28543         this.stripEl.dom.removeChild(tab.pnode.dom);
28544         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28545             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28546         }
28547         items.splice(index, 1);
28548         delete this.items[tab.id];
28549         tab.fireEvent("close", tab);
28550         tab.purgeListeners();
28551         this.autoSizeTabs();
28552     },
28553
28554     getNextAvailable : function(start){
28555         var items = this.items;
28556         var index = start;
28557         // look for a next tab that will slide over to
28558         // replace the one being removed
28559         while(index < items.length){
28560             var item = items[++index];
28561             if(item && !item.isHidden()){
28562                 return item;
28563             }
28564         }
28565         // if one isn't found select the previous tab (on the left)
28566         index = start;
28567         while(index >= 0){
28568             var item = items[--index];
28569             if(item && !item.isHidden()){
28570                 return item;
28571             }
28572         }
28573         return null;
28574     },
28575
28576     /**
28577      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28578      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28579      */
28580     disableTab : function(id){
28581         var tab = this.items[id];
28582         if(tab && this.active != tab){
28583             tab.disable();
28584         }
28585     },
28586
28587     /**
28588      * Enables a {@link Roo.TabPanelItem} that is disabled.
28589      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28590      */
28591     enableTab : function(id){
28592         var tab = this.items[id];
28593         tab.enable();
28594     },
28595
28596     /**
28597      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28598      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28599      * @return {Roo.TabPanelItem} The TabPanelItem.
28600      */
28601     activate : function(id){
28602         var tab = this.items[id];
28603         if(!tab){
28604             return null;
28605         }
28606         if(tab == this.active || tab.disabled){
28607             return tab;
28608         }
28609         var e = {};
28610         this.fireEvent("beforetabchange", this, e, tab);
28611         if(e.cancel !== true && !tab.disabled){
28612             if(this.active){
28613                 this.active.hide();
28614             }
28615             this.active = this.items[id];
28616             this.active.show();
28617             this.fireEvent("tabchange", this, this.active);
28618         }
28619         return tab;
28620     },
28621
28622     /**
28623      * Gets the active {@link Roo.TabPanelItem}.
28624      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28625      */
28626     getActiveTab : function(){
28627         return this.active;
28628     },
28629
28630     /**
28631      * Updates the tab body element to fit the height of the container element
28632      * for overflow scrolling
28633      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28634      */
28635     syncHeight : function(targetHeight){
28636         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28637         var bm = this.bodyEl.getMargins();
28638         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28639         this.bodyEl.setHeight(newHeight);
28640         return newHeight;
28641     },
28642
28643     onResize : function(){
28644         if(this.monitorResize){
28645             this.autoSizeTabs();
28646         }
28647     },
28648
28649     /**
28650      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28651      */
28652     beginUpdate : function(){
28653         this.updating = true;
28654     },
28655
28656     /**
28657      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28658      */
28659     endUpdate : function(){
28660         this.updating = false;
28661         this.autoSizeTabs();
28662     },
28663
28664     /**
28665      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28666      */
28667     autoSizeTabs : function(){
28668         var count = this.items.length;
28669         var vcount = count - this.hiddenCount;
28670         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28671             return;
28672         }
28673         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28674         var availWidth = Math.floor(w / vcount);
28675         var b = this.stripBody;
28676         if(b.getWidth() > w){
28677             var tabs = this.items;
28678             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28679             if(availWidth < this.minTabWidth){
28680                 /*if(!this.sleft){    // incomplete scrolling code
28681                     this.createScrollButtons();
28682                 }
28683                 this.showScroll();
28684                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28685             }
28686         }else{
28687             if(this.currentTabWidth < this.preferredTabWidth){
28688                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28689             }
28690         }
28691     },
28692
28693     /**
28694      * Returns the number of tabs in this TabPanel.
28695      * @return {Number}
28696      */
28697      getCount : function(){
28698          return this.items.length;
28699      },
28700
28701     /**
28702      * Resizes all the tabs to the passed width
28703      * @param {Number} The new width
28704      */
28705     setTabWidth : function(width){
28706         this.currentTabWidth = width;
28707         for(var i = 0, len = this.items.length; i < len; i++) {
28708                 if(!this.items[i].isHidden()) {
28709                 this.items[i].setWidth(width);
28710             }
28711         }
28712     },
28713
28714     /**
28715      * Destroys this TabPanel
28716      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28717      */
28718     destroy : function(removeEl){
28719         Roo.EventManager.removeResizeListener(this.onResize, this);
28720         for(var i = 0, len = this.items.length; i < len; i++){
28721             this.items[i].purgeListeners();
28722         }
28723         if(removeEl === true){
28724             this.el.update("");
28725             this.el.remove();
28726         }
28727     }
28728 });
28729
28730 /**
28731  * @class Roo.TabPanelItem
28732  * @extends Roo.util.Observable
28733  * Represents an individual item (tab plus body) in a TabPanel.
28734  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28735  * @param {String} id The id of this TabPanelItem
28736  * @param {String} text The text for the tab of this TabPanelItem
28737  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28738  */
28739 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28740     /**
28741      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28742      * @type Roo.TabPanel
28743      */
28744     this.tabPanel = tabPanel;
28745     /**
28746      * The id for this TabPanelItem
28747      * @type String
28748      */
28749     this.id = id;
28750     /** @private */
28751     this.disabled = false;
28752     /** @private */
28753     this.text = text;
28754     /** @private */
28755     this.loaded = false;
28756     this.closable = closable;
28757
28758     /**
28759      * The body element for this TabPanelItem.
28760      * @type Roo.Element
28761      */
28762     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28763     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28764     this.bodyEl.setStyle("display", "block");
28765     this.bodyEl.setStyle("zoom", "1");
28766     this.hideAction();
28767
28768     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28769     /** @private */
28770     this.el = Roo.get(els.el, true);
28771     this.inner = Roo.get(els.inner, true);
28772     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28773     this.pnode = Roo.get(els.el.parentNode, true);
28774     this.el.on("mousedown", this.onTabMouseDown, this);
28775     this.el.on("click", this.onTabClick, this);
28776     /** @private */
28777     if(closable){
28778         var c = Roo.get(els.close, true);
28779         c.dom.title = this.closeText;
28780         c.addClassOnOver("close-over");
28781         c.on("click", this.closeClick, this);
28782      }
28783
28784     this.addEvents({
28785          /**
28786          * @event activate
28787          * Fires when this tab becomes the active tab.
28788          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28789          * @param {Roo.TabPanelItem} this
28790          */
28791         "activate": true,
28792         /**
28793          * @event beforeclose
28794          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28795          * @param {Roo.TabPanelItem} this
28796          * @param {Object} e Set cancel to true on this object to cancel the close.
28797          */
28798         "beforeclose": true,
28799         /**
28800          * @event close
28801          * Fires when this tab is closed.
28802          * @param {Roo.TabPanelItem} this
28803          */
28804          "close": true,
28805         /**
28806          * @event deactivate
28807          * Fires when this tab is no longer the active tab.
28808          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28809          * @param {Roo.TabPanelItem} this
28810          */
28811          "deactivate" : true
28812     });
28813     this.hidden = false;
28814
28815     Roo.TabPanelItem.superclass.constructor.call(this);
28816 };
28817
28818 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28819     purgeListeners : function(){
28820        Roo.util.Observable.prototype.purgeListeners.call(this);
28821        this.el.removeAllListeners();
28822     },
28823     /**
28824      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28825      */
28826     show : function(){
28827         this.pnode.addClass("on");
28828         this.showAction();
28829         if(Roo.isOpera){
28830             this.tabPanel.stripWrap.repaint();
28831         }
28832         this.fireEvent("activate", this.tabPanel, this);
28833     },
28834
28835     /**
28836      * Returns true if this tab is the active tab.
28837      * @return {Boolean}
28838      */
28839     isActive : function(){
28840         return this.tabPanel.getActiveTab() == this;
28841     },
28842
28843     /**
28844      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28845      */
28846     hide : function(){
28847         this.pnode.removeClass("on");
28848         this.hideAction();
28849         this.fireEvent("deactivate", this.tabPanel, this);
28850     },
28851
28852     hideAction : function(){
28853         this.bodyEl.hide();
28854         this.bodyEl.setStyle("position", "absolute");
28855         this.bodyEl.setLeft("-20000px");
28856         this.bodyEl.setTop("-20000px");
28857     },
28858
28859     showAction : function(){
28860         this.bodyEl.setStyle("position", "relative");
28861         this.bodyEl.setTop("");
28862         this.bodyEl.setLeft("");
28863         this.bodyEl.show();
28864     },
28865
28866     /**
28867      * Set the tooltip for the tab.
28868      * @param {String} tooltip The tab's tooltip
28869      */
28870     setTooltip : function(text){
28871         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28872             this.textEl.dom.qtip = text;
28873             this.textEl.dom.removeAttribute('title');
28874         }else{
28875             this.textEl.dom.title = text;
28876         }
28877     },
28878
28879     onTabClick : function(e){
28880         e.preventDefault();
28881         this.tabPanel.activate(this.id);
28882     },
28883
28884     onTabMouseDown : function(e){
28885         e.preventDefault();
28886         this.tabPanel.activate(this.id);
28887     },
28888
28889     getWidth : function(){
28890         return this.inner.getWidth();
28891     },
28892
28893     setWidth : function(width){
28894         var iwidth = width - this.pnode.getPadding("lr");
28895         this.inner.setWidth(iwidth);
28896         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28897         this.pnode.setWidth(width);
28898     },
28899
28900     /**
28901      * Show or hide the tab
28902      * @param {Boolean} hidden True to hide or false to show.
28903      */
28904     setHidden : function(hidden){
28905         this.hidden = hidden;
28906         this.pnode.setStyle("display", hidden ? "none" : "");
28907     },
28908
28909     /**
28910      * Returns true if this tab is "hidden"
28911      * @return {Boolean}
28912      */
28913     isHidden : function(){
28914         return this.hidden;
28915     },
28916
28917     /**
28918      * Returns the text for this tab
28919      * @return {String}
28920      */
28921     getText : function(){
28922         return this.text;
28923     },
28924
28925     autoSize : function(){
28926         //this.el.beginMeasure();
28927         this.textEl.setWidth(1);
28928         /*
28929          *  #2804 [new] Tabs in Roojs
28930          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28931          */
28932         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28933         //this.el.endMeasure();
28934     },
28935
28936     /**
28937      * Sets the text for the tab (Note: this also sets the tooltip text)
28938      * @param {String} text The tab's text and tooltip
28939      */
28940     setText : function(text){
28941         this.text = text;
28942         this.textEl.update(text);
28943         this.setTooltip(text);
28944         if(!this.tabPanel.resizeTabs){
28945             this.autoSize();
28946         }
28947     },
28948     /**
28949      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28950      */
28951     activate : function(){
28952         this.tabPanel.activate(this.id);
28953     },
28954
28955     /**
28956      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
28957      */
28958     disable : function(){
28959         if(this.tabPanel.active != this){
28960             this.disabled = true;
28961             this.pnode.addClass("disabled");
28962         }
28963     },
28964
28965     /**
28966      * Enables this TabPanelItem if it was previously disabled.
28967      */
28968     enable : function(){
28969         this.disabled = false;
28970         this.pnode.removeClass("disabled");
28971     },
28972
28973     /**
28974      * Sets the content for this TabPanelItem.
28975      * @param {String} content The content
28976      * @param {Boolean} loadScripts true to look for and load scripts
28977      */
28978     setContent : function(content, loadScripts){
28979         this.bodyEl.update(content, loadScripts);
28980     },
28981
28982     /**
28983      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
28984      * @return {Roo.UpdateManager} The UpdateManager
28985      */
28986     getUpdateManager : function(){
28987         return this.bodyEl.getUpdateManager();
28988     },
28989
28990     /**
28991      * Set a URL to be used to load the content for this TabPanelItem.
28992      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
28993      * @param {String/Object} params (optional) The string params for the update call or an object of the params. See {@link Roo.UpdateManager#update} for more details. (Defaults to null)
28994      * @param {Boolean} loadOnce (optional) Whether to only load the content once. If this is false it makes the Ajax call every time this TabPanelItem is activated. (Defaults to false)
28995      * @return {Roo.UpdateManager} The UpdateManager
28996      */
28997     setUrl : function(url, params, loadOnce){
28998         if(this.refreshDelegate){
28999             this.un('activate', this.refreshDelegate);
29000         }
29001         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29002         this.on("activate", this.refreshDelegate);
29003         return this.bodyEl.getUpdateManager();
29004     },
29005
29006     /** @private */
29007     _handleRefresh : function(url, params, loadOnce){
29008         if(!loadOnce || !this.loaded){
29009             var updater = this.bodyEl.getUpdateManager();
29010             updater.update(url, params, this._setLoaded.createDelegate(this));
29011         }
29012     },
29013
29014     /**
29015      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29016      *   Will fail silently if the setUrl method has not been called.
29017      *   This does not activate the panel, just updates its content.
29018      */
29019     refresh : function(){
29020         if(this.refreshDelegate){
29021            this.loaded = false;
29022            this.refreshDelegate();
29023         }
29024     },
29025
29026     /** @private */
29027     _setLoaded : function(){
29028         this.loaded = true;
29029     },
29030
29031     /** @private */
29032     closeClick : function(e){
29033         var o = {};
29034         e.stopEvent();
29035         this.fireEvent("beforeclose", this, o);
29036         if(o.cancel !== true){
29037             this.tabPanel.removeTab(this.id);
29038         }
29039     },
29040     /**
29041      * The text displayed in the tooltip for the close icon.
29042      * @type String
29043      */
29044     closeText : "Close this tab"
29045 });
29046
29047 /** @private */
29048 Roo.TabPanel.prototype.createStrip = function(container){
29049     var strip = document.createElement("div");
29050     strip.className = "x-tabs-wrap";
29051     container.appendChild(strip);
29052     return strip;
29053 };
29054 /** @private */
29055 Roo.TabPanel.prototype.createStripList = function(strip){
29056     // div wrapper for retard IE
29057     // returns the "tr" element.
29058     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29059         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29060         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29061     return strip.firstChild.firstChild.firstChild.firstChild;
29062 };
29063 /** @private */
29064 Roo.TabPanel.prototype.createBody = function(container){
29065     var body = document.createElement("div");
29066     Roo.id(body, "tab-body");
29067     Roo.fly(body).addClass("x-tabs-body");
29068     container.appendChild(body);
29069     return body;
29070 };
29071 /** @private */
29072 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29073     var body = Roo.getDom(id);
29074     if(!body){
29075         body = document.createElement("div");
29076         body.id = id;
29077     }
29078     Roo.fly(body).addClass("x-tabs-item-body");
29079     bodyEl.insertBefore(body, bodyEl.firstChild);
29080     return body;
29081 };
29082 /** @private */
29083 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29084     var td = document.createElement("td");
29085     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29086     //stripEl.appendChild(td);
29087     if(closable){
29088         td.className = "x-tabs-closable";
29089         if(!this.closeTpl){
29090             this.closeTpl = new Roo.Template(
29091                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29092                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29093                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29094             );
29095         }
29096         var el = this.closeTpl.overwrite(td, {"text": text});
29097         var close = el.getElementsByTagName("div")[0];
29098         var inner = el.getElementsByTagName("em")[0];
29099         return {"el": el, "close": close, "inner": inner};
29100     } else {
29101         if(!this.tabTpl){
29102             this.tabTpl = new Roo.Template(
29103                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29104                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29105             );
29106         }
29107         var el = this.tabTpl.overwrite(td, {"text": text});
29108         var inner = el.getElementsByTagName("em")[0];
29109         return {"el": el, "inner": inner};
29110     }
29111 };/*
29112  * Based on:
29113  * Ext JS Library 1.1.1
29114  * Copyright(c) 2006-2007, Ext JS, LLC.
29115  *
29116  * Originally Released Under LGPL - original licence link has changed is not relivant.
29117  *
29118  * Fork - LGPL
29119  * <script type="text/javascript">
29120  */
29121
29122 /**
29123  * @class Roo.Button
29124  * @extends Roo.util.Observable
29125  * Simple Button class
29126  * @cfg {String} text The button text
29127  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29128  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29129  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29130  * @cfg {Object} scope The scope of the handler
29131  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29132  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29133  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29134  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29135  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29136  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29137    applies if enableToggle = true)
29138  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29139  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29140   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29141  * @constructor
29142  * Create a new button
29143  * @param {Object} config The config object
29144  */
29145 Roo.Button = function(renderTo, config)
29146 {
29147     if (!config) {
29148         config = renderTo;
29149         renderTo = config.renderTo || false;
29150     }
29151     
29152     Roo.apply(this, config);
29153     this.addEvents({
29154         /**
29155              * @event click
29156              * Fires when this button is clicked
29157              * @param {Button} this
29158              * @param {EventObject} e The click event
29159              */
29160             "click" : true,
29161         /**
29162              * @event toggle
29163              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29164              * @param {Button} this
29165              * @param {Boolean} pressed
29166              */
29167             "toggle" : true,
29168         /**
29169              * @event mouseover
29170              * Fires when the mouse hovers over the button
29171              * @param {Button} this
29172              * @param {Event} e The event object
29173              */
29174         'mouseover' : true,
29175         /**
29176              * @event mouseout
29177              * Fires when the mouse exits the button
29178              * @param {Button} this
29179              * @param {Event} e The event object
29180              */
29181         'mouseout': true,
29182          /**
29183              * @event render
29184              * Fires when the button is rendered
29185              * @param {Button} this
29186              */
29187         'render': true
29188     });
29189     if(this.menu){
29190         this.menu = Roo.menu.MenuMgr.get(this.menu);
29191     }
29192     // register listeners first!!  - so render can be captured..
29193     Roo.util.Observable.call(this);
29194     if(renderTo){
29195         this.render(renderTo);
29196     }
29197     
29198   
29199 };
29200
29201 Roo.extend(Roo.Button, Roo.util.Observable, {
29202     /**
29203      * 
29204      */
29205     
29206     /**
29207      * Read-only. True if this button is hidden
29208      * @type Boolean
29209      */
29210     hidden : false,
29211     /**
29212      * Read-only. True if this button is disabled
29213      * @type Boolean
29214      */
29215     disabled : false,
29216     /**
29217      * Read-only. True if this button is pressed (only if enableToggle = true)
29218      * @type Boolean
29219      */
29220     pressed : false,
29221
29222     /**
29223      * @cfg {Number} tabIndex 
29224      * The DOM tabIndex for this button (defaults to undefined)
29225      */
29226     tabIndex : undefined,
29227
29228     /**
29229      * @cfg {Boolean} enableToggle
29230      * True to enable pressed/not pressed toggling (defaults to false)
29231      */
29232     enableToggle: false,
29233     /**
29234      * @cfg {Mixed} menu
29235      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29236      */
29237     menu : undefined,
29238     /**
29239      * @cfg {String} menuAlign
29240      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29241      */
29242     menuAlign : "tl-bl?",
29243
29244     /**
29245      * @cfg {String} iconCls
29246      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29247      */
29248     iconCls : undefined,
29249     /**
29250      * @cfg {String} type
29251      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29252      */
29253     type : 'button',
29254
29255     // private
29256     menuClassTarget: 'tr',
29257
29258     /**
29259      * @cfg {String} clickEvent
29260      * The type of event to map to the button's event handler (defaults to 'click')
29261      */
29262     clickEvent : 'click',
29263
29264     /**
29265      * @cfg {Boolean} handleMouseEvents
29266      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29267      */
29268     handleMouseEvents : true,
29269
29270     /**
29271      * @cfg {String} tooltipType
29272      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29273      */
29274     tooltipType : 'qtip',
29275
29276     /**
29277      * @cfg {String} cls
29278      * A CSS class to apply to the button's main element.
29279      */
29280     
29281     /**
29282      * @cfg {Roo.Template} template (Optional)
29283      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29284      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29285      * require code modifications if required elements (e.g. a button) aren't present.
29286      */
29287
29288     // private
29289     render : function(renderTo){
29290         var btn;
29291         if(this.hideParent){
29292             this.parentEl = Roo.get(renderTo);
29293         }
29294         if(!this.dhconfig){
29295             if(!this.template){
29296                 if(!Roo.Button.buttonTemplate){
29297                     // hideous table template
29298                     Roo.Button.buttonTemplate = new Roo.Template(
29299                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29300                         '<td class="x-btn-left"><i>&#160;</i></td><td class="x-btn-center"><em unselectable="on"><button class="x-btn-text" type="{1}">{0}</button></em></td><td class="x-btn-right"><i>&#160;</i></td>',
29301                         "</tr></tbody></table>");
29302                 }
29303                 this.template = Roo.Button.buttonTemplate;
29304             }
29305             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29306             var btnEl = btn.child("button:first");
29307             btnEl.on('focus', this.onFocus, this);
29308             btnEl.on('blur', this.onBlur, this);
29309             if(this.cls){
29310                 btn.addClass(this.cls);
29311             }
29312             if(this.icon){
29313                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29314             }
29315             if(this.iconCls){
29316                 btnEl.addClass(this.iconCls);
29317                 if(!this.cls){
29318                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29319                 }
29320             }
29321             if(this.tabIndex !== undefined){
29322                 btnEl.dom.tabIndex = this.tabIndex;
29323             }
29324             if(this.tooltip){
29325                 if(typeof this.tooltip == 'object'){
29326                     Roo.QuickTips.tips(Roo.apply({
29327                           target: btnEl.id
29328                     }, this.tooltip));
29329                 } else {
29330                     btnEl.dom[this.tooltipType] = this.tooltip;
29331                 }
29332             }
29333         }else{
29334             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29335         }
29336         this.el = btn;
29337         if(this.id){
29338             this.el.dom.id = this.el.id = this.id;
29339         }
29340         if(this.menu){
29341             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29342             this.menu.on("show", this.onMenuShow, this);
29343             this.menu.on("hide", this.onMenuHide, this);
29344         }
29345         btn.addClass("x-btn");
29346         if(Roo.isIE && !Roo.isIE7){
29347             this.autoWidth.defer(1, this);
29348         }else{
29349             this.autoWidth();
29350         }
29351         if(this.handleMouseEvents){
29352             btn.on("mouseover", this.onMouseOver, this);
29353             btn.on("mouseout", this.onMouseOut, this);
29354             btn.on("mousedown", this.onMouseDown, this);
29355         }
29356         btn.on(this.clickEvent, this.onClick, this);
29357         //btn.on("mouseup", this.onMouseUp, this);
29358         if(this.hidden){
29359             this.hide();
29360         }
29361         if(this.disabled){
29362             this.disable();
29363         }
29364         Roo.ButtonToggleMgr.register(this);
29365         if(this.pressed){
29366             this.el.addClass("x-btn-pressed");
29367         }
29368         if(this.repeat){
29369             var repeater = new Roo.util.ClickRepeater(btn,
29370                 typeof this.repeat == "object" ? this.repeat : {}
29371             );
29372             repeater.on("click", this.onClick,  this);
29373         }
29374         
29375         this.fireEvent('render', this);
29376         
29377     },
29378     /**
29379      * Returns the button's underlying element
29380      * @return {Roo.Element} The element
29381      */
29382     getEl : function(){
29383         return this.el;  
29384     },
29385     
29386     /**
29387      * Destroys this Button and removes any listeners.
29388      */
29389     destroy : function(){
29390         Roo.ButtonToggleMgr.unregister(this);
29391         this.el.removeAllListeners();
29392         this.purgeListeners();
29393         this.el.remove();
29394     },
29395
29396     // private
29397     autoWidth : function(){
29398         if(this.el){
29399             this.el.setWidth("auto");
29400             if(Roo.isIE7 && Roo.isStrict){
29401                 var ib = this.el.child('button');
29402                 if(ib && ib.getWidth() > 20){
29403                     ib.clip();
29404                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29405                 }
29406             }
29407             if(this.minWidth){
29408                 if(this.hidden){
29409                     this.el.beginMeasure();
29410                 }
29411                 if(this.el.getWidth() < this.minWidth){
29412                     this.el.setWidth(this.minWidth);
29413                 }
29414                 if(this.hidden){
29415                     this.el.endMeasure();
29416                 }
29417             }
29418         }
29419     },
29420
29421     /**
29422      * Assigns this button's click handler
29423      * @param {Function} handler The function to call when the button is clicked
29424      * @param {Object} scope (optional) Scope for the function passed in
29425      */
29426     setHandler : function(handler, scope){
29427         this.handler = handler;
29428         this.scope = scope;  
29429     },
29430     
29431     /**
29432      * Sets this button's text
29433      * @param {String} text The button text
29434      */
29435     setText : function(text){
29436         this.text = text;
29437         if(this.el){
29438             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29439         }
29440         this.autoWidth();
29441     },
29442     
29443     /**
29444      * Gets the text for this button
29445      * @return {String} The button text
29446      */
29447     getText : function(){
29448         return this.text;  
29449     },
29450     
29451     /**
29452      * Show this button
29453      */
29454     show: function(){
29455         this.hidden = false;
29456         if(this.el){
29457             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29458         }
29459     },
29460     
29461     /**
29462      * Hide this button
29463      */
29464     hide: function(){
29465         this.hidden = true;
29466         if(this.el){
29467             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29468         }
29469     },
29470     
29471     /**
29472      * Convenience function for boolean show/hide
29473      * @param {Boolean} visible True to show, false to hide
29474      */
29475     setVisible: function(visible){
29476         if(visible) {
29477             this.show();
29478         }else{
29479             this.hide();
29480         }
29481     },
29482     
29483     /**
29484      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29485      * @param {Boolean} state (optional) Force a particular state
29486      */
29487     toggle : function(state){
29488         state = state === undefined ? !this.pressed : state;
29489         if(state != this.pressed){
29490             if(state){
29491                 this.el.addClass("x-btn-pressed");
29492                 this.pressed = true;
29493                 this.fireEvent("toggle", this, true);
29494             }else{
29495                 this.el.removeClass("x-btn-pressed");
29496                 this.pressed = false;
29497                 this.fireEvent("toggle", this, false);
29498             }
29499             if(this.toggleHandler){
29500                 this.toggleHandler.call(this.scope || this, this, state);
29501             }
29502         }
29503     },
29504     
29505     /**
29506      * Focus the button
29507      */
29508     focus : function(){
29509         this.el.child('button:first').focus();
29510     },
29511     
29512     /**
29513      * Disable this button
29514      */
29515     disable : function(){
29516         if(this.el){
29517             this.el.addClass("x-btn-disabled");
29518         }
29519         this.disabled = true;
29520     },
29521     
29522     /**
29523      * Enable this button
29524      */
29525     enable : function(){
29526         if(this.el){
29527             this.el.removeClass("x-btn-disabled");
29528         }
29529         this.disabled = false;
29530     },
29531
29532     /**
29533      * Convenience function for boolean enable/disable
29534      * @param {Boolean} enabled True to enable, false to disable
29535      */
29536     setDisabled : function(v){
29537         this[v !== true ? "enable" : "disable"]();
29538     },
29539
29540     // private
29541     onClick : function(e)
29542     {
29543         if(e){
29544             e.preventDefault();
29545         }
29546         if(e.button != 0){
29547             return;
29548         }
29549         if(!this.disabled){
29550             if(this.enableToggle){
29551                 this.toggle();
29552             }
29553             if(this.menu && !this.menu.isVisible()){
29554                 this.menu.show(this.el, this.menuAlign);
29555             }
29556             this.fireEvent("click", this, e);
29557             if(this.handler){
29558                 this.el.removeClass("x-btn-over");
29559                 this.handler.call(this.scope || this, this, e);
29560             }
29561         }
29562     },
29563     // private
29564     onMouseOver : function(e){
29565         if(!this.disabled){
29566             this.el.addClass("x-btn-over");
29567             this.fireEvent('mouseover', this, e);
29568         }
29569     },
29570     // private
29571     onMouseOut : function(e){
29572         if(!e.within(this.el,  true)){
29573             this.el.removeClass("x-btn-over");
29574             this.fireEvent('mouseout', this, e);
29575         }
29576     },
29577     // private
29578     onFocus : function(e){
29579         if(!this.disabled){
29580             this.el.addClass("x-btn-focus");
29581         }
29582     },
29583     // private
29584     onBlur : function(e){
29585         this.el.removeClass("x-btn-focus");
29586     },
29587     // private
29588     onMouseDown : function(e){
29589         if(!this.disabled && e.button == 0){
29590             this.el.addClass("x-btn-click");
29591             Roo.get(document).on('mouseup', this.onMouseUp, this);
29592         }
29593     },
29594     // private
29595     onMouseUp : function(e){
29596         if(e.button == 0){
29597             this.el.removeClass("x-btn-click");
29598             Roo.get(document).un('mouseup', this.onMouseUp, this);
29599         }
29600     },
29601     // private
29602     onMenuShow : function(e){
29603         this.el.addClass("x-btn-menu-active");
29604     },
29605     // private
29606     onMenuHide : function(e){
29607         this.el.removeClass("x-btn-menu-active");
29608     }   
29609 });
29610
29611 // Private utility class used by Button
29612 Roo.ButtonToggleMgr = function(){
29613    var groups = {};
29614    
29615    function toggleGroup(btn, state){
29616        if(state){
29617            var g = groups[btn.toggleGroup];
29618            for(var i = 0, l = g.length; i < l; i++){
29619                if(g[i] != btn){
29620                    g[i].toggle(false);
29621                }
29622            }
29623        }
29624    }
29625    
29626    return {
29627        register : function(btn){
29628            if(!btn.toggleGroup){
29629                return;
29630            }
29631            var g = groups[btn.toggleGroup];
29632            if(!g){
29633                g = groups[btn.toggleGroup] = [];
29634            }
29635            g.push(btn);
29636            btn.on("toggle", toggleGroup);
29637        },
29638        
29639        unregister : function(btn){
29640            if(!btn.toggleGroup){
29641                return;
29642            }
29643            var g = groups[btn.toggleGroup];
29644            if(g){
29645                g.remove(btn);
29646                btn.un("toggle", toggleGroup);
29647            }
29648        }
29649    };
29650 }();/*
29651  * Based on:
29652  * Ext JS Library 1.1.1
29653  * Copyright(c) 2006-2007, Ext JS, LLC.
29654  *
29655  * Originally Released Under LGPL - original licence link has changed is not relivant.
29656  *
29657  * Fork - LGPL
29658  * <script type="text/javascript">
29659  */
29660  
29661 /**
29662  * @class Roo.SplitButton
29663  * @extends Roo.Button
29664  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29665  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29666  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29667  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29668  * @cfg {String} arrowTooltip The title attribute of the arrow
29669  * @constructor
29670  * Create a new menu button
29671  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29672  * @param {Object} config The config object
29673  */
29674 Roo.SplitButton = function(renderTo, config){
29675     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29676     /**
29677      * @event arrowclick
29678      * Fires when this button's arrow is clicked
29679      * @param {SplitButton} this
29680      * @param {EventObject} e The click event
29681      */
29682     this.addEvents({"arrowclick":true});
29683 };
29684
29685 Roo.extend(Roo.SplitButton, Roo.Button, {
29686     render : function(renderTo){
29687         // this is one sweet looking template!
29688         var tpl = new Roo.Template(
29689             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29690             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29691             '<tr><td class="x-btn-left"><i>&#160;</i></td><td class="x-btn-center"><button class="x-btn-text" type="{1}">{0}</button></td></tr>',
29692             "</tbody></table></td><td>",
29693             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29694             '<tr><td class="x-btn-center"><button class="x-btn-menu-arrow-el" type="button">&#160;</button></td><td class="x-btn-right"><i>&#160;</i></td></tr>',
29695             "</tbody></table></td></tr></table>"
29696         );
29697         var btn = tpl.append(renderTo, [this.text, this.type], true);
29698         var btnEl = btn.child("button");
29699         if(this.cls){
29700             btn.addClass(this.cls);
29701         }
29702         if(this.icon){
29703             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29704         }
29705         if(this.iconCls){
29706             btnEl.addClass(this.iconCls);
29707             if(!this.cls){
29708                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29709             }
29710         }
29711         this.el = btn;
29712         if(this.handleMouseEvents){
29713             btn.on("mouseover", this.onMouseOver, this);
29714             btn.on("mouseout", this.onMouseOut, this);
29715             btn.on("mousedown", this.onMouseDown, this);
29716             btn.on("mouseup", this.onMouseUp, this);
29717         }
29718         btn.on(this.clickEvent, this.onClick, this);
29719         if(this.tooltip){
29720             if(typeof this.tooltip == 'object'){
29721                 Roo.QuickTips.tips(Roo.apply({
29722                       target: btnEl.id
29723                 }, this.tooltip));
29724             } else {
29725                 btnEl.dom[this.tooltipType] = this.tooltip;
29726             }
29727         }
29728         if(this.arrowTooltip){
29729             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29730         }
29731         if(this.hidden){
29732             this.hide();
29733         }
29734         if(this.disabled){
29735             this.disable();
29736         }
29737         if(this.pressed){
29738             this.el.addClass("x-btn-pressed");
29739         }
29740         if(Roo.isIE && !Roo.isIE7){
29741             this.autoWidth.defer(1, this);
29742         }else{
29743             this.autoWidth();
29744         }
29745         if(this.menu){
29746             this.menu.on("show", this.onMenuShow, this);
29747             this.menu.on("hide", this.onMenuHide, this);
29748         }
29749         this.fireEvent('render', this);
29750     },
29751
29752     // private
29753     autoWidth : function(){
29754         if(this.el){
29755             var tbl = this.el.child("table:first");
29756             var tbl2 = this.el.child("table:last");
29757             this.el.setWidth("auto");
29758             tbl.setWidth("auto");
29759             if(Roo.isIE7 && Roo.isStrict){
29760                 var ib = this.el.child('button:first');
29761                 if(ib && ib.getWidth() > 20){
29762                     ib.clip();
29763                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29764                 }
29765             }
29766             if(this.minWidth){
29767                 if(this.hidden){
29768                     this.el.beginMeasure();
29769                 }
29770                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29771                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29772                 }
29773                 if(this.hidden){
29774                     this.el.endMeasure();
29775                 }
29776             }
29777             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29778         } 
29779     },
29780     /**
29781      * Sets this button's click handler
29782      * @param {Function} handler The function to call when the button is clicked
29783      * @param {Object} scope (optional) Scope for the function passed above
29784      */
29785     setHandler : function(handler, scope){
29786         this.handler = handler;
29787         this.scope = scope;  
29788     },
29789     
29790     /**
29791      * Sets this button's arrow click handler
29792      * @param {Function} handler The function to call when the arrow is clicked
29793      * @param {Object} scope (optional) Scope for the function passed above
29794      */
29795     setArrowHandler : function(handler, scope){
29796         this.arrowHandler = handler;
29797         this.scope = scope;  
29798     },
29799     
29800     /**
29801      * Focus the button
29802      */
29803     focus : function(){
29804         if(this.el){
29805             this.el.child("button:first").focus();
29806         }
29807     },
29808
29809     // private
29810     onClick : function(e){
29811         e.preventDefault();
29812         if(!this.disabled){
29813             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29814                 if(this.menu && !this.menu.isVisible()){
29815                     this.menu.show(this.el, this.menuAlign);
29816                 }
29817                 this.fireEvent("arrowclick", this, e);
29818                 if(this.arrowHandler){
29819                     this.arrowHandler.call(this.scope || this, this, e);
29820                 }
29821             }else{
29822                 this.fireEvent("click", this, e);
29823                 if(this.handler){
29824                     this.handler.call(this.scope || this, this, e);
29825                 }
29826             }
29827         }
29828     },
29829     // private
29830     onMouseDown : function(e){
29831         if(!this.disabled){
29832             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29833         }
29834     },
29835     // private
29836     onMouseUp : function(e){
29837         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29838     }   
29839 });
29840
29841
29842 // backwards compat
29843 Roo.MenuButton = Roo.SplitButton;/*
29844  * Based on:
29845  * Ext JS Library 1.1.1
29846  * Copyright(c) 2006-2007, Ext JS, LLC.
29847  *
29848  * Originally Released Under LGPL - original licence link has changed is not relivant.
29849  *
29850  * Fork - LGPL
29851  * <script type="text/javascript">
29852  */
29853
29854 /**
29855  * @class Roo.Toolbar
29856  * Basic Toolbar class.
29857  * @constructor
29858  * Creates a new Toolbar
29859  * @param {Object} container The config object
29860  */ 
29861 Roo.Toolbar = function(container, buttons, config)
29862 {
29863     /// old consturctor format still supported..
29864     if(container instanceof Array){ // omit the container for later rendering
29865         buttons = container;
29866         config = buttons;
29867         container = null;
29868     }
29869     if (typeof(container) == 'object' && container.xtype) {
29870         config = container;
29871         container = config.container;
29872         buttons = config.buttons || []; // not really - use items!!
29873     }
29874     var xitems = [];
29875     if (config && config.items) {
29876         xitems = config.items;
29877         delete config.items;
29878     }
29879     Roo.apply(this, config);
29880     this.buttons = buttons;
29881     
29882     if(container){
29883         this.render(container);
29884     }
29885     this.xitems = xitems;
29886     Roo.each(xitems, function(b) {
29887         this.add(b);
29888     }, this);
29889     
29890 };
29891
29892 Roo.Toolbar.prototype = {
29893     /**
29894      * @cfg {Array} items
29895      * array of button configs or elements to add (will be converted to a MixedCollection)
29896      */
29897     
29898     /**
29899      * @cfg {String/HTMLElement/Element} container
29900      * The id or element that will contain the toolbar
29901      */
29902     // private
29903     render : function(ct){
29904         this.el = Roo.get(ct);
29905         if(this.cls){
29906             this.el.addClass(this.cls);
29907         }
29908         // using a table allows for vertical alignment
29909         // 100% width is needed by Safari...
29910         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29911         this.tr = this.el.child("tr", true);
29912         var autoId = 0;
29913         this.items = new Roo.util.MixedCollection(false, function(o){
29914             return o.id || ("item" + (++autoId));
29915         });
29916         if(this.buttons){
29917             this.add.apply(this, this.buttons);
29918             delete this.buttons;
29919         }
29920     },
29921
29922     /**
29923      * Adds element(s) to the toolbar -- this function takes a variable number of 
29924      * arguments of mixed type and adds them to the toolbar.
29925      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29926      * <ul>
29927      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29928      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29929      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29930      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29931      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29932      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29933      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29934      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29935      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29936      * </ul>
29937      * @param {Mixed} arg2
29938      * @param {Mixed} etc.
29939      */
29940     add : function(){
29941         var a = arguments, l = a.length;
29942         for(var i = 0; i < l; i++){
29943             this._add(a[i]);
29944         }
29945     },
29946     // private..
29947     _add : function(el) {
29948         
29949         if (el.xtype) {
29950             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29951         }
29952         
29953         if (el.applyTo){ // some kind of form field
29954             return this.addField(el);
29955         } 
29956         if (el.render){ // some kind of Toolbar.Item
29957             return this.addItem(el);
29958         }
29959         if (typeof el == "string"){ // string
29960             if(el == "separator" || el == "-"){
29961                 return this.addSeparator();
29962             }
29963             if (el == " "){
29964                 return this.addSpacer();
29965             }
29966             if(el == "->"){
29967                 return this.addFill();
29968             }
29969             return this.addText(el);
29970             
29971         }
29972         if(el.tagName){ // element
29973             return this.addElement(el);
29974         }
29975         if(typeof el == "object"){ // must be button config?
29976             return this.addButton(el);
29977         }
29978         // and now what?!?!
29979         return false;
29980         
29981     },
29982     
29983     /**
29984      * Add an Xtype element
29985      * @param {Object} xtype Xtype Object
29986      * @return {Object} created Object
29987      */
29988     addxtype : function(e){
29989         return this.add(e);  
29990     },
29991     
29992     /**
29993      * Returns the Element for this toolbar.
29994      * @return {Roo.Element}
29995      */
29996     getEl : function(){
29997         return this.el;  
29998     },
29999     
30000     /**
30001      * Adds a separator
30002      * @return {Roo.Toolbar.Item} The separator item
30003      */
30004     addSeparator : function(){
30005         return this.addItem(new Roo.Toolbar.Separator());
30006     },
30007
30008     /**
30009      * Adds a spacer element
30010      * @return {Roo.Toolbar.Spacer} The spacer item
30011      */
30012     addSpacer : function(){
30013         return this.addItem(new Roo.Toolbar.Spacer());
30014     },
30015
30016     /**
30017      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30018      * @return {Roo.Toolbar.Fill} The fill item
30019      */
30020     addFill : function(){
30021         return this.addItem(new Roo.Toolbar.Fill());
30022     },
30023
30024     /**
30025      * Adds any standard HTML element to the toolbar
30026      * @param {String/HTMLElement/Element} el The element or id of the element to add
30027      * @return {Roo.Toolbar.Item} The element's item
30028      */
30029     addElement : function(el){
30030         return this.addItem(new Roo.Toolbar.Item(el));
30031     },
30032     /**
30033      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30034      * @type Roo.util.MixedCollection  
30035      */
30036     items : false,
30037      
30038     /**
30039      * Adds any Toolbar.Item or subclass
30040      * @param {Roo.Toolbar.Item} item
30041      * @return {Roo.Toolbar.Item} The item
30042      */
30043     addItem : function(item){
30044         var td = this.nextBlock();
30045         item.render(td);
30046         this.items.add(item);
30047         return item;
30048     },
30049     
30050     /**
30051      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30052      * @param {Object/Array} config A button config or array of configs
30053      * @return {Roo.Toolbar.Button/Array}
30054      */
30055     addButton : function(config){
30056         if(config instanceof Array){
30057             var buttons = [];
30058             for(var i = 0, len = config.length; i < len; i++) {
30059                 buttons.push(this.addButton(config[i]));
30060             }
30061             return buttons;
30062         }
30063         var b = config;
30064         if(!(config instanceof Roo.Toolbar.Button)){
30065             b = config.split ?
30066                 new Roo.Toolbar.SplitButton(config) :
30067                 new Roo.Toolbar.Button(config);
30068         }
30069         var td = this.nextBlock();
30070         b.render(td);
30071         this.items.add(b);
30072         return b;
30073     },
30074     
30075     /**
30076      * Adds text to the toolbar
30077      * @param {String} text The text to add
30078      * @return {Roo.Toolbar.Item} The element's item
30079      */
30080     addText : function(text){
30081         return this.addItem(new Roo.Toolbar.TextItem(text));
30082     },
30083     
30084     /**
30085      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30086      * @param {Number} index The index where the item is to be inserted
30087      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30088      * @return {Roo.Toolbar.Button/Item}
30089      */
30090     insertButton : function(index, item){
30091         if(item instanceof Array){
30092             var buttons = [];
30093             for(var i = 0, len = item.length; i < len; i++) {
30094                buttons.push(this.insertButton(index + i, item[i]));
30095             }
30096             return buttons;
30097         }
30098         if (!(item instanceof Roo.Toolbar.Button)){
30099            item = new Roo.Toolbar.Button(item);
30100         }
30101         var td = document.createElement("td");
30102         this.tr.insertBefore(td, this.tr.childNodes[index]);
30103         item.render(td);
30104         this.items.insert(index, item);
30105         return item;
30106     },
30107     
30108     /**
30109      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30110      * @param {Object} config
30111      * @return {Roo.Toolbar.Item} The element's item
30112      */
30113     addDom : function(config, returnEl){
30114         var td = this.nextBlock();
30115         Roo.DomHelper.overwrite(td, config);
30116         var ti = new Roo.Toolbar.Item(td.firstChild);
30117         ti.render(td);
30118         this.items.add(ti);
30119         return ti;
30120     },
30121
30122     /**
30123      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30124      * @type Roo.util.MixedCollection  
30125      */
30126     fields : false,
30127     
30128     /**
30129      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30130      * Note: the field should not have been rendered yet. For a field that has already been
30131      * rendered, use {@link #addElement}.
30132      * @param {Roo.form.Field} field
30133      * @return {Roo.ToolbarItem}
30134      */
30135      
30136       
30137     addField : function(field) {
30138         if (!this.fields) {
30139             var autoId = 0;
30140             this.fields = new Roo.util.MixedCollection(false, function(o){
30141                 return o.id || ("item" + (++autoId));
30142             });
30143
30144         }
30145         
30146         var td = this.nextBlock();
30147         field.render(td);
30148         var ti = new Roo.Toolbar.Item(td.firstChild);
30149         ti.render(td);
30150         this.items.add(ti);
30151         this.fields.add(field);
30152         return ti;
30153     },
30154     /**
30155      * Hide the toolbar
30156      * @method hide
30157      */
30158      
30159       
30160     hide : function()
30161     {
30162         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30163         this.el.child('div').hide();
30164     },
30165     /**
30166      * Show the toolbar
30167      * @method show
30168      */
30169     show : function()
30170     {
30171         this.el.child('div').show();
30172     },
30173       
30174     // private
30175     nextBlock : function(){
30176         var td = document.createElement("td");
30177         this.tr.appendChild(td);
30178         return td;
30179     },
30180
30181     // private
30182     destroy : function(){
30183         if(this.items){ // rendered?
30184             Roo.destroy.apply(Roo, this.items.items);
30185         }
30186         if(this.fields){ // rendered?
30187             Roo.destroy.apply(Roo, this.fields.items);
30188         }
30189         Roo.Element.uncache(this.el, this.tr);
30190     }
30191 };
30192
30193 /**
30194  * @class Roo.Toolbar.Item
30195  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30196  * @constructor
30197  * Creates a new Item
30198  * @param {HTMLElement} el 
30199  */
30200 Roo.Toolbar.Item = function(el){
30201     var cfg = {};
30202     if (typeof (el.xtype) != 'undefined') {
30203         cfg = el;
30204         el = cfg.el;
30205     }
30206     
30207     this.el = Roo.getDom(el);
30208     this.id = Roo.id(this.el);
30209     this.hidden = false;
30210     
30211     this.addEvents({
30212          /**
30213              * @event render
30214              * Fires when the button is rendered
30215              * @param {Button} this
30216              */
30217         'render': true
30218     });
30219     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30220 };
30221 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30222 //Roo.Toolbar.Item.prototype = {
30223     
30224     /**
30225      * Get this item's HTML Element
30226      * @return {HTMLElement}
30227      */
30228     getEl : function(){
30229        return this.el;  
30230     },
30231
30232     // private
30233     render : function(td){
30234         
30235          this.td = td;
30236         td.appendChild(this.el);
30237         
30238         this.fireEvent('render', this);
30239     },
30240     
30241     /**
30242      * Removes and destroys this item.
30243      */
30244     destroy : function(){
30245         this.td.parentNode.removeChild(this.td);
30246     },
30247     
30248     /**
30249      * Shows this item.
30250      */
30251     show: function(){
30252         this.hidden = false;
30253         this.td.style.display = "";
30254     },
30255     
30256     /**
30257      * Hides this item.
30258      */
30259     hide: function(){
30260         this.hidden = true;
30261         this.td.style.display = "none";
30262     },
30263     
30264     /**
30265      * Convenience function for boolean show/hide.
30266      * @param {Boolean} visible true to show/false to hide
30267      */
30268     setVisible: function(visible){
30269         if(visible) {
30270             this.show();
30271         }else{
30272             this.hide();
30273         }
30274     },
30275     
30276     /**
30277      * Try to focus this item.
30278      */
30279     focus : function(){
30280         Roo.fly(this.el).focus();
30281     },
30282     
30283     /**
30284      * Disables this item.
30285      */
30286     disable : function(){
30287         Roo.fly(this.td).addClass("x-item-disabled");
30288         this.disabled = true;
30289         this.el.disabled = true;
30290     },
30291     
30292     /**
30293      * Enables this item.
30294      */
30295     enable : function(){
30296         Roo.fly(this.td).removeClass("x-item-disabled");
30297         this.disabled = false;
30298         this.el.disabled = false;
30299     }
30300 });
30301
30302
30303 /**
30304  * @class Roo.Toolbar.Separator
30305  * @extends Roo.Toolbar.Item
30306  * A simple toolbar separator class
30307  * @constructor
30308  * Creates a new Separator
30309  */
30310 Roo.Toolbar.Separator = function(cfg){
30311     
30312     var s = document.createElement("span");
30313     s.className = "ytb-sep";
30314     if (cfg) {
30315         cfg.el = s;
30316     }
30317     
30318     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30319 };
30320 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30321     enable:Roo.emptyFn,
30322     disable:Roo.emptyFn,
30323     focus:Roo.emptyFn
30324 });
30325
30326 /**
30327  * @class Roo.Toolbar.Spacer
30328  * @extends Roo.Toolbar.Item
30329  * A simple element that adds extra horizontal space to a toolbar.
30330  * @constructor
30331  * Creates a new Spacer
30332  */
30333 Roo.Toolbar.Spacer = function(cfg){
30334     var s = document.createElement("div");
30335     s.className = "ytb-spacer";
30336     if (cfg) {
30337         cfg.el = s;
30338     }
30339     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30340 };
30341 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30342     enable:Roo.emptyFn,
30343     disable:Roo.emptyFn,
30344     focus:Roo.emptyFn
30345 });
30346
30347 /**
30348  * @class Roo.Toolbar.Fill
30349  * @extends Roo.Toolbar.Spacer
30350  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30351  * @constructor
30352  * Creates a new Spacer
30353  */
30354 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30355     // private
30356     render : function(td){
30357         td.style.width = '100%';
30358         Roo.Toolbar.Fill.superclass.render.call(this, td);
30359     }
30360 });
30361
30362 /**
30363  * @class Roo.Toolbar.TextItem
30364  * @extends Roo.Toolbar.Item
30365  * A simple class that renders text directly into a toolbar.
30366  * @constructor
30367  * Creates a new TextItem
30368  * @param {String} text
30369  */
30370 Roo.Toolbar.TextItem = function(cfg){
30371     var  text = cfg || "";
30372     if (typeof(cfg) == 'object') {
30373         text = cfg.text || "";
30374     }  else {
30375         cfg = null;
30376     }
30377     var s = document.createElement("span");
30378     s.className = "ytb-text";
30379     s.innerHTML = text;
30380     if (cfg) {
30381         cfg.el  = s;
30382     }
30383     
30384     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30385 };
30386 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30387     
30388      
30389     enable:Roo.emptyFn,
30390     disable:Roo.emptyFn,
30391     focus:Roo.emptyFn
30392 });
30393
30394 /**
30395  * @class Roo.Toolbar.Button
30396  * @extends Roo.Button
30397  * A button that renders into a toolbar.
30398  * @constructor
30399  * Creates a new Button
30400  * @param {Object} config A standard {@link Roo.Button} config object
30401  */
30402 Roo.Toolbar.Button = function(config){
30403     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30404 };
30405 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30406     render : function(td){
30407         this.td = td;
30408         Roo.Toolbar.Button.superclass.render.call(this, td);
30409     },
30410     
30411     /**
30412      * Removes and destroys this button
30413      */
30414     destroy : function(){
30415         Roo.Toolbar.Button.superclass.destroy.call(this);
30416         this.td.parentNode.removeChild(this.td);
30417     },
30418     
30419     /**
30420      * Shows this button
30421      */
30422     show: function(){
30423         this.hidden = false;
30424         this.td.style.display = "";
30425     },
30426     
30427     /**
30428      * Hides this button
30429      */
30430     hide: function(){
30431         this.hidden = true;
30432         this.td.style.display = "none";
30433     },
30434
30435     /**
30436      * Disables this item
30437      */
30438     disable : function(){
30439         Roo.fly(this.td).addClass("x-item-disabled");
30440         this.disabled = true;
30441     },
30442
30443     /**
30444      * Enables this item
30445      */
30446     enable : function(){
30447         Roo.fly(this.td).removeClass("x-item-disabled");
30448         this.disabled = false;
30449     }
30450 });
30451 // backwards compat
30452 Roo.ToolbarButton = Roo.Toolbar.Button;
30453
30454 /**
30455  * @class Roo.Toolbar.SplitButton
30456  * @extends Roo.SplitButton
30457  * A menu button that renders into a toolbar.
30458  * @constructor
30459  * Creates a new SplitButton
30460  * @param {Object} config A standard {@link Roo.SplitButton} config object
30461  */
30462 Roo.Toolbar.SplitButton = function(config){
30463     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30464 };
30465 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30466     render : function(td){
30467         this.td = td;
30468         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30469     },
30470     
30471     /**
30472      * Removes and destroys this button
30473      */
30474     destroy : function(){
30475         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30476         this.td.parentNode.removeChild(this.td);
30477     },
30478     
30479     /**
30480      * Shows this button
30481      */
30482     show: function(){
30483         this.hidden = false;
30484         this.td.style.display = "";
30485     },
30486     
30487     /**
30488      * Hides this button
30489      */
30490     hide: function(){
30491         this.hidden = true;
30492         this.td.style.display = "none";
30493     }
30494 });
30495
30496 // backwards compat
30497 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30498  * Based on:
30499  * Ext JS Library 1.1.1
30500  * Copyright(c) 2006-2007, Ext JS, LLC.
30501  *
30502  * Originally Released Under LGPL - original licence link has changed is not relivant.
30503  *
30504  * Fork - LGPL
30505  * <script type="text/javascript">
30506  */
30507  
30508 /**
30509  * @class Roo.PagingToolbar
30510  * @extends Roo.Toolbar
30511  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30512  * @constructor
30513  * Create a new PagingToolbar
30514  * @param {Object} config The config object
30515  */
30516 Roo.PagingToolbar = function(el, ds, config)
30517 {
30518     // old args format still supported... - xtype is prefered..
30519     if (typeof(el) == 'object' && el.xtype) {
30520         // created from xtype...
30521         config = el;
30522         ds = el.dataSource;
30523         el = config.container;
30524     }
30525     var items = [];
30526     if (config.items) {
30527         items = config.items;
30528         config.items = [];
30529     }
30530     
30531     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30532     this.ds = ds;
30533     this.cursor = 0;
30534     this.renderButtons(this.el);
30535     this.bind(ds);
30536     
30537     // supprot items array.
30538    
30539     Roo.each(items, function(e) {
30540         this.add(Roo.factory(e));
30541     },this);
30542     
30543 };
30544
30545 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30546     /**
30547      * @cfg {Roo.data.Store} dataSource
30548      * The underlying data store providing the paged data
30549      */
30550     /**
30551      * @cfg {String/HTMLElement/Element} container
30552      * container The id or element that will contain the toolbar
30553      */
30554     /**
30555      * @cfg {Boolean} displayInfo
30556      * True to display the displayMsg (defaults to false)
30557      */
30558     /**
30559      * @cfg {Number} pageSize
30560      * The number of records to display per page (defaults to 20)
30561      */
30562     pageSize: 20,
30563     /**
30564      * @cfg {String} displayMsg
30565      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30566      */
30567     displayMsg : 'Displaying {0} - {1} of {2}',
30568     /**
30569      * @cfg {String} emptyMsg
30570      * The message to display when no records are found (defaults to "No data to display")
30571      */
30572     emptyMsg : 'No data to display',
30573     /**
30574      * Customizable piece of the default paging text (defaults to "Page")
30575      * @type String
30576      */
30577     beforePageText : "Page",
30578     /**
30579      * Customizable piece of the default paging text (defaults to "of %0")
30580      * @type String
30581      */
30582     afterPageText : "of {0}",
30583     /**
30584      * Customizable piece of the default paging text (defaults to "First Page")
30585      * @type String
30586      */
30587     firstText : "First Page",
30588     /**
30589      * Customizable piece of the default paging text (defaults to "Previous Page")
30590      * @type String
30591      */
30592     prevText : "Previous Page",
30593     /**
30594      * Customizable piece of the default paging text (defaults to "Next Page")
30595      * @type String
30596      */
30597     nextText : "Next Page",
30598     /**
30599      * Customizable piece of the default paging text (defaults to "Last Page")
30600      * @type String
30601      */
30602     lastText : "Last Page",
30603     /**
30604      * Customizable piece of the default paging text (defaults to "Refresh")
30605      * @type String
30606      */
30607     refreshText : "Refresh",
30608
30609     // private
30610     renderButtons : function(el){
30611         Roo.PagingToolbar.superclass.render.call(this, el);
30612         this.first = this.addButton({
30613             tooltip: this.firstText,
30614             cls: "x-btn-icon x-grid-page-first",
30615             disabled: true,
30616             handler: this.onClick.createDelegate(this, ["first"])
30617         });
30618         this.prev = this.addButton({
30619             tooltip: this.prevText,
30620             cls: "x-btn-icon x-grid-page-prev",
30621             disabled: true,
30622             handler: this.onClick.createDelegate(this, ["prev"])
30623         });
30624         //this.addSeparator();
30625         this.add(this.beforePageText);
30626         this.field = Roo.get(this.addDom({
30627            tag: "input",
30628            type: "text",
30629            size: "3",
30630            value: "1",
30631            cls: "x-grid-page-number"
30632         }).el);
30633         this.field.on("keydown", this.onPagingKeydown, this);
30634         this.field.on("focus", function(){this.dom.select();});
30635         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30636         this.field.setHeight(18);
30637         //this.addSeparator();
30638         this.next = this.addButton({
30639             tooltip: this.nextText,
30640             cls: "x-btn-icon x-grid-page-next",
30641             disabled: true,
30642             handler: this.onClick.createDelegate(this, ["next"])
30643         });
30644         this.last = this.addButton({
30645             tooltip: this.lastText,
30646             cls: "x-btn-icon x-grid-page-last",
30647             disabled: true,
30648             handler: this.onClick.createDelegate(this, ["last"])
30649         });
30650         //this.addSeparator();
30651         this.loading = this.addButton({
30652             tooltip: this.refreshText,
30653             cls: "x-btn-icon x-grid-loading",
30654             handler: this.onClick.createDelegate(this, ["refresh"])
30655         });
30656
30657         if(this.displayInfo){
30658             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30659         }
30660     },
30661
30662     // private
30663     updateInfo : function(){
30664         if(this.displayEl){
30665             var count = this.ds.getCount();
30666             var msg = count == 0 ?
30667                 this.emptyMsg :
30668                 String.format(
30669                     this.displayMsg,
30670                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30671                 );
30672             this.displayEl.update(msg);
30673         }
30674     },
30675
30676     // private
30677     onLoad : function(ds, r, o){
30678        this.cursor = o.params ? o.params.start : 0;
30679        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30680
30681        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30682        this.field.dom.value = ap;
30683        this.first.setDisabled(ap == 1);
30684        this.prev.setDisabled(ap == 1);
30685        this.next.setDisabled(ap == ps);
30686        this.last.setDisabled(ap == ps);
30687        this.loading.enable();
30688        this.updateInfo();
30689     },
30690
30691     // private
30692     getPageData : function(){
30693         var total = this.ds.getTotalCount();
30694         return {
30695             total : total,
30696             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30697             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30698         };
30699     },
30700
30701     // private
30702     onLoadError : function(){
30703         this.loading.enable();
30704     },
30705
30706     // private
30707     onPagingKeydown : function(e){
30708         var k = e.getKey();
30709         var d = this.getPageData();
30710         if(k == e.RETURN){
30711             var v = this.field.dom.value, pageNum;
30712             if(!v || isNaN(pageNum = parseInt(v, 10))){
30713                 this.field.dom.value = d.activePage;
30714                 return;
30715             }
30716             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30717             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30718             e.stopEvent();
30719         }
30720         else if(k == e.HOME || (k == e.UP && e.ctrlKey) || (k == e.PAGEUP && e.ctrlKey) || (k == e.RIGHT && e.ctrlKey) || k == e.END || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey))
30721         {
30722           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30723           this.field.dom.value = pageNum;
30724           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30725           e.stopEvent();
30726         }
30727         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30728         {
30729           var v = this.field.dom.value, pageNum; 
30730           var increment = (e.shiftKey) ? 10 : 1;
30731           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30732             increment *= -1;
30733           }
30734           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30735             this.field.dom.value = d.activePage;
30736             return;
30737           }
30738           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30739           {
30740             this.field.dom.value = parseInt(v, 10) + increment;
30741             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30742             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30743           }
30744           e.stopEvent();
30745         }
30746     },
30747
30748     // private
30749     beforeLoad : function(){
30750         if(this.loading){
30751             this.loading.disable();
30752         }
30753     },
30754
30755     // private
30756     onClick : function(which){
30757         var ds = this.ds;
30758         switch(which){
30759             case "first":
30760                 ds.load({params:{start: 0, limit: this.pageSize}});
30761             break;
30762             case "prev":
30763                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30764             break;
30765             case "next":
30766                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30767             break;
30768             case "last":
30769                 var total = ds.getTotalCount();
30770                 var extra = total % this.pageSize;
30771                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30772                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30773             break;
30774             case "refresh":
30775                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30776             break;
30777         }
30778     },
30779
30780     /**
30781      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30782      * @param {Roo.data.Store} store The data store to unbind
30783      */
30784     unbind : function(ds){
30785         ds.un("beforeload", this.beforeLoad, this);
30786         ds.un("load", this.onLoad, this);
30787         ds.un("loadexception", this.onLoadError, this);
30788         ds.un("remove", this.updateInfo, this);
30789         ds.un("add", this.updateInfo, this);
30790         this.ds = undefined;
30791     },
30792
30793     /**
30794      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30795      * @param {Roo.data.Store} store The data store to bind
30796      */
30797     bind : function(ds){
30798         ds.on("beforeload", this.beforeLoad, this);
30799         ds.on("load", this.onLoad, this);
30800         ds.on("loadexception", this.onLoadError, this);
30801         ds.on("remove", this.updateInfo, this);
30802         ds.on("add", this.updateInfo, this);
30803         this.ds = ds;
30804     }
30805 });/*
30806  * Based on:
30807  * Ext JS Library 1.1.1
30808  * Copyright(c) 2006-2007, Ext JS, LLC.
30809  *
30810  * Originally Released Under LGPL - original licence link has changed is not relivant.
30811  *
30812  * Fork - LGPL
30813  * <script type="text/javascript">
30814  */
30815
30816 /**
30817  * @class Roo.Resizable
30818  * @extends Roo.util.Observable
30819  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30820  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30821  * the textarea in a div and set "resizeChild" to true (or to the id of the element), <b>or</b> set wrap:true in your config and
30822  * the element will be wrapped for you automatically.</p>
30823  * <p>Here is the list of valid resize handles:</p>
30824  * <pre>
30825 Value   Description
30826 ------  -------------------
30827  'n'     north
30828  's'     south
30829  'e'     east
30830  'w'     west
30831  'nw'    northwest
30832  'sw'    southwest
30833  'se'    southeast
30834  'ne'    northeast
30835  'hd'    horizontal drag
30836  'all'   all
30837 </pre>
30838  * <p>Here's an example showing the creation of a typical Resizable:</p>
30839  * <pre><code>
30840 var resizer = new Roo.Resizable("element-id", {
30841     handles: 'all',
30842     minWidth: 200,
30843     minHeight: 100,
30844     maxWidth: 500,
30845     maxHeight: 400,
30846     pinned: true
30847 });
30848 resizer.on("resize", myHandler);
30849 </code></pre>
30850  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30851  * resizer.east.setDisplayed(false);</p>
30852  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30853  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30854  * resize operation's new size (defaults to [0, 0])
30855  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30856  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30857  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30858  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30859  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30860  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30861  * @cfg {Number} width The width of the element in pixels (defaults to null)
30862  * @cfg {Number} height The height of the element in pixels (defaults to null)
30863  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30864  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30865  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30866  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30867  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30868  * in favor of the handles config option (defaults to false)
30869  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30870  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30871  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30872  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30873  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30874  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30875  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30876  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30877  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30878  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30879  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30880  * @constructor
30881  * Create a new resizable component
30882  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30883  * @param {Object} config configuration options
30884   */
30885 Roo.Resizable = function(el, config)
30886 {
30887     this.el = Roo.get(el);
30888
30889     if(config && config.wrap){
30890         config.resizeChild = this.el;
30891         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30892         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30893         this.el.setStyle("overflow", "hidden");
30894         this.el.setPositioning(config.resizeChild.getPositioning());
30895         config.resizeChild.clearPositioning();
30896         if(!config.width || !config.height){
30897             var csize = config.resizeChild.getSize();
30898             this.el.setSize(csize.width, csize.height);
30899         }
30900         if(config.pinned && !config.adjustments){
30901             config.adjustments = "auto";
30902         }
30903     }
30904
30905     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30906     this.proxy.unselectable();
30907     this.proxy.enableDisplayMode('block');
30908
30909     Roo.apply(this, config);
30910
30911     if(this.pinned){
30912         this.disableTrackOver = true;
30913         this.el.addClass("x-resizable-pinned");
30914     }
30915     // if the element isn't positioned, make it relative
30916     var position = this.el.getStyle("position");
30917     if(position != "absolute" && position != "fixed"){
30918         this.el.setStyle("position", "relative");
30919     }
30920     if(!this.handles){ // no handles passed, must be legacy style
30921         this.handles = 's,e,se';
30922         if(this.multiDirectional){
30923             this.handles += ',n,w';
30924         }
30925     }
30926     if(this.handles == "all"){
30927         this.handles = "n s e w ne nw se sw";
30928     }
30929     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30930     var ps = Roo.Resizable.positions;
30931     for(var i = 0, len = hs.length; i < len; i++){
30932         if(hs[i] && ps[hs[i]]){
30933             var pos = ps[hs[i]];
30934             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30935         }
30936     }
30937     // legacy
30938     this.corner = this.southeast;
30939     
30940     // updateBox = the box can move..
30941     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30942         this.updateBox = true;
30943     }
30944
30945     this.activeHandle = null;
30946
30947     if(this.resizeChild){
30948         if(typeof this.resizeChild == "boolean"){
30949             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30950         }else{
30951             this.resizeChild = Roo.get(this.resizeChild, true);
30952         }
30953     }
30954     
30955     if(this.adjustments == "auto"){
30956         var rc = this.resizeChild;
30957         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
30958         if(rc && (hw || hn)){
30959             rc.position("relative");
30960             rc.setLeft(hw ? hw.el.getWidth() : 0);
30961             rc.setTop(hn ? hn.el.getHeight() : 0);
30962         }
30963         this.adjustments = [
30964             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
30965             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
30966         ];
30967     }
30968
30969     if(this.draggable){
30970         this.dd = this.dynamic ?
30971             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
30972         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
30973     }
30974
30975     // public events
30976     this.addEvents({
30977         /**
30978          * @event beforeresize
30979          * Fired before resize is allowed. Set enabled to false to cancel resize.
30980          * @param {Roo.Resizable} this
30981          * @param {Roo.EventObject} e The mousedown event
30982          */
30983         "beforeresize" : true,
30984         /**
30985          * @event resizing
30986          * Fired a resizing.
30987          * @param {Roo.Resizable} this
30988          * @param {Number} x The new x position
30989          * @param {Number} y The new y position
30990          * @param {Number} w The new w width
30991          * @param {Number} h The new h hight
30992          * @param {Roo.EventObject} e The mouseup event
30993          */
30994         "resizing" : true,
30995         /**
30996          * @event resize
30997          * Fired after a resize.
30998          * @param {Roo.Resizable} this
30999          * @param {Number} width The new width
31000          * @param {Number} height The new height
31001          * @param {Roo.EventObject} e The mouseup event
31002          */
31003         "resize" : true
31004     });
31005
31006     if(this.width !== null && this.height !== null){
31007         this.resizeTo(this.width, this.height);
31008     }else{
31009         this.updateChildSize();
31010     }
31011     if(Roo.isIE){
31012         this.el.dom.style.zoom = 1;
31013     }
31014     Roo.Resizable.superclass.constructor.call(this);
31015 };
31016
31017 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31018         resizeChild : false,
31019         adjustments : [0, 0],
31020         minWidth : 5,
31021         minHeight : 5,
31022         maxWidth : 10000,
31023         maxHeight : 10000,
31024         enabled : true,
31025         animate : false,
31026         duration : .35,
31027         dynamic : false,
31028         handles : false,
31029         multiDirectional : false,
31030         disableTrackOver : false,
31031         easing : 'easeOutStrong',
31032         widthIncrement : 0,
31033         heightIncrement : 0,
31034         pinned : false,
31035         width : null,
31036         height : null,
31037         preserveRatio : false,
31038         transparent: false,
31039         minX: 0,
31040         minY: 0,
31041         draggable: false,
31042
31043         /**
31044          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31045          */
31046         constrainTo: undefined,
31047         /**
31048          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31049          */
31050         resizeRegion: undefined,
31051
31052
31053     /**
31054      * Perform a manual resize
31055      * @param {Number} width
31056      * @param {Number} height
31057      */
31058     resizeTo : function(width, height){
31059         this.el.setSize(width, height);
31060         this.updateChildSize();
31061         this.fireEvent("resize", this, width, height, null);
31062     },
31063
31064     // private
31065     startSizing : function(e, handle){
31066         this.fireEvent("beforeresize", this, e);
31067         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31068
31069             if(!this.overlay){
31070                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31071                 this.overlay.unselectable();
31072                 this.overlay.enableDisplayMode("block");
31073                 this.overlay.on("mousemove", this.onMouseMove, this);
31074                 this.overlay.on("mouseup", this.onMouseUp, this);
31075             }
31076             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31077
31078             this.resizing = true;
31079             this.startBox = this.el.getBox();
31080             this.startPoint = e.getXY();
31081             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31082                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31083
31084             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31085             this.overlay.show();
31086
31087             if(this.constrainTo) {
31088                 var ct = Roo.get(this.constrainTo);
31089                 this.resizeRegion = ct.getRegion().adjust(
31090                     ct.getFrameWidth('t'),
31091                     ct.getFrameWidth('l'),
31092                     -ct.getFrameWidth('b'),
31093                     -ct.getFrameWidth('r')
31094                 );
31095             }
31096
31097             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31098             this.proxy.show();
31099             this.proxy.setBox(this.startBox);
31100             if(!this.dynamic){
31101                 this.proxy.setStyle('visibility', 'visible');
31102             }
31103         }
31104     },
31105
31106     // private
31107     onMouseDown : function(handle, e){
31108         if(this.enabled){
31109             e.stopEvent();
31110             this.activeHandle = handle;
31111             this.startSizing(e, handle);
31112         }
31113     },
31114
31115     // private
31116     onMouseUp : function(e){
31117         var size = this.resizeElement();
31118         this.resizing = false;
31119         this.handleOut();
31120         this.overlay.hide();
31121         this.proxy.hide();
31122         this.fireEvent("resize", this, size.width, size.height, e);
31123     },
31124
31125     // private
31126     updateChildSize : function(){
31127         
31128         if(this.resizeChild){
31129             var el = this.el;
31130             var child = this.resizeChild;
31131             var adj = this.adjustments;
31132             if(el.dom.offsetWidth){
31133                 var b = el.getSize(true);
31134                 child.setSize(b.width+adj[0], b.height+adj[1]);
31135             }
31136             // Second call here for IE
31137             // The first call enables instant resizing and
31138             // the second call corrects scroll bars if they
31139             // exist
31140             if(Roo.isIE){
31141                 setTimeout(function(){
31142                     if(el.dom.offsetWidth){
31143                         var b = el.getSize(true);
31144                         child.setSize(b.width+adj[0], b.height+adj[1]);
31145                     }
31146                 }, 10);
31147             }
31148         }
31149     },
31150
31151     // private
31152     snap : function(value, inc, min){
31153         if(!inc || !value) {
31154             return value;
31155         }
31156         var newValue = value;
31157         var m = value % inc;
31158         if(m > 0){
31159             if(m > (inc/2)){
31160                 newValue = value + (inc-m);
31161             }else{
31162                 newValue = value - m;
31163             }
31164         }
31165         return Math.max(min, newValue);
31166     },
31167
31168     // private
31169     resizeElement : function(){
31170         var box = this.proxy.getBox();
31171         if(this.updateBox){
31172             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31173         }else{
31174             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31175         }
31176         this.updateChildSize();
31177         if(!this.dynamic){
31178             this.proxy.hide();
31179         }
31180         return box;
31181     },
31182
31183     // private
31184     constrain : function(v, diff, m, mx){
31185         if(v - diff < m){
31186             diff = v - m;
31187         }else if(v - diff > mx){
31188             diff = mx - v;
31189         }
31190         return diff;
31191     },
31192
31193     // private
31194     onMouseMove : function(e){
31195         
31196         if(this.enabled){
31197             try{// try catch so if something goes wrong the user doesn't get hung
31198
31199             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31200                 return;
31201             }
31202
31203             //var curXY = this.startPoint;
31204             var curSize = this.curSize || this.startBox;
31205             var x = this.startBox.x, y = this.startBox.y;
31206             var ox = x, oy = y;
31207             var w = curSize.width, h = curSize.height;
31208             var ow = w, oh = h;
31209             var mw = this.minWidth, mh = this.minHeight;
31210             var mxw = this.maxWidth, mxh = this.maxHeight;
31211             var wi = this.widthIncrement;
31212             var hi = this.heightIncrement;
31213
31214             var eventXY = e.getXY();
31215             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31216             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31217
31218             var pos = this.activeHandle.position;
31219
31220             switch(pos){
31221                 case "east":
31222                     w += diffX;
31223                     w = Math.min(Math.max(mw, w), mxw);
31224                     break;
31225              
31226                 case "south":
31227                     h += diffY;
31228                     h = Math.min(Math.max(mh, h), mxh);
31229                     break;
31230                 case "southeast":
31231                     w += diffX;
31232                     h += diffY;
31233                     w = Math.min(Math.max(mw, w), mxw);
31234                     h = Math.min(Math.max(mh, h), mxh);
31235                     break;
31236                 case "north":
31237                     diffY = this.constrain(h, diffY, mh, mxh);
31238                     y += diffY;
31239                     h -= diffY;
31240                     break;
31241                 case "hdrag":
31242                     
31243                     if (wi) {
31244                         var adiffX = Math.abs(diffX);
31245                         var sub = (adiffX % wi); // how much 
31246                         if (sub > (wi/2)) { // far enough to snap
31247                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31248                         } else {
31249                             // remove difference.. 
31250                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31251                         }
31252                     }
31253                     x += diffX;
31254                     x = Math.max(this.minX, x);
31255                     break;
31256                 case "west":
31257                     diffX = this.constrain(w, diffX, mw, mxw);
31258                     x += diffX;
31259                     w -= diffX;
31260                     break;
31261                 case "northeast":
31262                     w += diffX;
31263                     w = Math.min(Math.max(mw, w), mxw);
31264                     diffY = this.constrain(h, diffY, mh, mxh);
31265                     y += diffY;
31266                     h -= diffY;
31267                     break;
31268                 case "northwest":
31269                     diffX = this.constrain(w, diffX, mw, mxw);
31270                     diffY = this.constrain(h, diffY, mh, mxh);
31271                     y += diffY;
31272                     h -= diffY;
31273                     x += diffX;
31274                     w -= diffX;
31275                     break;
31276                case "southwest":
31277                     diffX = this.constrain(w, diffX, mw, mxw);
31278                     h += diffY;
31279                     h = Math.min(Math.max(mh, h), mxh);
31280                     x += diffX;
31281                     w -= diffX;
31282                     break;
31283             }
31284
31285             var sw = this.snap(w, wi, mw);
31286             var sh = this.snap(h, hi, mh);
31287             if(sw != w || sh != h){
31288                 switch(pos){
31289                     case "northeast":
31290                         y -= sh - h;
31291                     break;
31292                     case "north":
31293                         y -= sh - h;
31294                         break;
31295                     case "southwest":
31296                         x -= sw - w;
31297                     break;
31298                     case "west":
31299                         x -= sw - w;
31300                         break;
31301                     case "northwest":
31302                         x -= sw - w;
31303                         y -= sh - h;
31304                     break;
31305                 }
31306                 w = sw;
31307                 h = sh;
31308             }
31309
31310             if(this.preserveRatio){
31311                 switch(pos){
31312                     case "southeast":
31313                     case "east":
31314                         h = oh * (w/ow);
31315                         h = Math.min(Math.max(mh, h), mxh);
31316                         w = ow * (h/oh);
31317                        break;
31318                     case "south":
31319                         w = ow * (h/oh);
31320                         w = Math.min(Math.max(mw, w), mxw);
31321                         h = oh * (w/ow);
31322                         break;
31323                     case "northeast":
31324                         w = ow * (h/oh);
31325                         w = Math.min(Math.max(mw, w), mxw);
31326                         h = oh * (w/ow);
31327                     break;
31328                     case "north":
31329                         var tw = w;
31330                         w = ow * (h/oh);
31331                         w = Math.min(Math.max(mw, w), mxw);
31332                         h = oh * (w/ow);
31333                         x += (tw - w) / 2;
31334                         break;
31335                     case "southwest":
31336                         h = oh * (w/ow);
31337                         h = Math.min(Math.max(mh, h), mxh);
31338                         var tw = w;
31339                         w = ow * (h/oh);
31340                         x += tw - w;
31341                         break;
31342                     case "west":
31343                         var th = h;
31344                         h = oh * (w/ow);
31345                         h = Math.min(Math.max(mh, h), mxh);
31346                         y += (th - h) / 2;
31347                         var tw = w;
31348                         w = ow * (h/oh);
31349                         x += tw - w;
31350                        break;
31351                     case "northwest":
31352                         var tw = w;
31353                         var th = h;
31354                         h = oh * (w/ow);
31355                         h = Math.min(Math.max(mh, h), mxh);
31356                         w = ow * (h/oh);
31357                         y += th - h;
31358                         x += tw - w;
31359                        break;
31360
31361                 }
31362             }
31363             if (pos == 'hdrag') {
31364                 w = ow;
31365             }
31366             this.proxy.setBounds(x, y, w, h);
31367             if(this.dynamic){
31368                 this.resizeElement();
31369             }
31370             }catch(e){}
31371         }
31372         this.fireEvent("resizing", this, x, y, w, h, e);
31373     },
31374
31375     // private
31376     handleOver : function(){
31377         if(this.enabled){
31378             this.el.addClass("x-resizable-over");
31379         }
31380     },
31381
31382     // private
31383     handleOut : function(){
31384         if(!this.resizing){
31385             this.el.removeClass("x-resizable-over");
31386         }
31387     },
31388
31389     /**
31390      * Returns the element this component is bound to.
31391      * @return {Roo.Element}
31392      */
31393     getEl : function(){
31394         return this.el;
31395     },
31396
31397     /**
31398      * Returns the resizeChild element (or null).
31399      * @return {Roo.Element}
31400      */
31401     getResizeChild : function(){
31402         return this.resizeChild;
31403     },
31404     groupHandler : function()
31405     {
31406         
31407     },
31408     /**
31409      * Destroys this resizable. If the element was wrapped and
31410      * removeEl is not true then the element remains.
31411      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31412      */
31413     destroy : function(removeEl){
31414         this.proxy.remove();
31415         if(this.overlay){
31416             this.overlay.removeAllListeners();
31417             this.overlay.remove();
31418         }
31419         var ps = Roo.Resizable.positions;
31420         for(var k in ps){
31421             if(typeof ps[k] != "function" && this[ps[k]]){
31422                 var h = this[ps[k]];
31423                 h.el.removeAllListeners();
31424                 h.el.remove();
31425             }
31426         }
31427         if(removeEl){
31428             this.el.update("");
31429             this.el.remove();
31430         }
31431     }
31432 });
31433
31434 // private
31435 // hash to map config positions to true positions
31436 Roo.Resizable.positions = {
31437     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31438     hd: "hdrag"
31439 };
31440
31441 // private
31442 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31443     if(!this.tpl){
31444         // only initialize the template if resizable is used
31445         var tpl = Roo.DomHelper.createTemplate(
31446             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31447         );
31448         tpl.compile();
31449         Roo.Resizable.Handle.prototype.tpl = tpl;
31450     }
31451     this.position = pos;
31452     this.rz = rz;
31453     // show north drag fro topdra
31454     var handlepos = pos == 'hdrag' ? 'north' : pos;
31455     
31456     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31457     if (pos == 'hdrag') {
31458         this.el.setStyle('cursor', 'pointer');
31459     }
31460     this.el.unselectable();
31461     if(transparent){
31462         this.el.setOpacity(0);
31463     }
31464     this.el.on("mousedown", this.onMouseDown, this);
31465     if(!disableTrackOver){
31466         this.el.on("mouseover", this.onMouseOver, this);
31467         this.el.on("mouseout", this.onMouseOut, this);
31468     }
31469 };
31470
31471 // private
31472 Roo.Resizable.Handle.prototype = {
31473     afterResize : function(rz){
31474         Roo.log('after?');
31475         // do nothing
31476     },
31477     // private
31478     onMouseDown : function(e){
31479         this.rz.onMouseDown(this, e);
31480     },
31481     // private
31482     onMouseOver : function(e){
31483         this.rz.handleOver(this, e);
31484     },
31485     // private
31486     onMouseOut : function(e){
31487         this.rz.handleOut(this, e);
31488     }
31489 };/*
31490  * Based on:
31491  * Ext JS Library 1.1.1
31492  * Copyright(c) 2006-2007, Ext JS, LLC.
31493  *
31494  * Originally Released Under LGPL - original licence link has changed is not relivant.
31495  *
31496  * Fork - LGPL
31497  * <script type="text/javascript">
31498  */
31499
31500 /**
31501  * @class Roo.Editor
31502  * @extends Roo.Component
31503  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31504  * @constructor
31505  * Create a new Editor
31506  * @param {Roo.form.Field} field The Field object (or descendant)
31507  * @param {Object} config The config object
31508  */
31509 Roo.Editor = function(field, config){
31510     Roo.Editor.superclass.constructor.call(this, config);
31511     this.field = field;
31512     this.addEvents({
31513         /**
31514              * @event beforestartedit
31515              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31516              * false from the handler of this event.
31517              * @param {Editor} this
31518              * @param {Roo.Element} boundEl The underlying element bound to this editor
31519              * @param {Mixed} value The field value being set
31520              */
31521         "beforestartedit" : true,
31522         /**
31523              * @event startedit
31524              * Fires when this editor is displayed
31525              * @param {Roo.Element} boundEl The underlying element bound to this editor
31526              * @param {Mixed} value The starting field value
31527              */
31528         "startedit" : true,
31529         /**
31530              * @event beforecomplete
31531              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31532              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31533              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31534              * event will not fire since no edit actually occurred.
31535              * @param {Editor} this
31536              * @param {Mixed} value The current field value
31537              * @param {Mixed} startValue The original field value
31538              */
31539         "beforecomplete" : true,
31540         /**
31541              * @event complete
31542              * Fires after editing is complete and any changed value has been written to the underlying field.
31543              * @param {Editor} this
31544              * @param {Mixed} value The current field value
31545              * @param {Mixed} startValue The original field value
31546              */
31547         "complete" : true,
31548         /**
31549          * @event specialkey
31550          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31551          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31552          * @param {Roo.form.Field} this
31553          * @param {Roo.EventObject} e The event object
31554          */
31555         "specialkey" : true
31556     });
31557 };
31558
31559 Roo.extend(Roo.Editor, Roo.Component, {
31560     /**
31561      * @cfg {Boolean/String} autosize
31562      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31563      * or "height" to adopt the height only (defaults to false)
31564      */
31565     /**
31566      * @cfg {Boolean} revertInvalid
31567      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31568      * validation fails (defaults to true)
31569      */
31570     /**
31571      * @cfg {Boolean} ignoreNoChange
31572      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31573      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31574      * will never be ignored.
31575      */
31576     /**
31577      * @cfg {Boolean} hideEl
31578      * False to keep the bound element visible while the editor is displayed (defaults to true)
31579      */
31580     /**
31581      * @cfg {Mixed} value
31582      * The data value of the underlying field (defaults to "")
31583      */
31584     value : "",
31585     /**
31586      * @cfg {String} alignment
31587      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31588      */
31589     alignment: "c-c?",
31590     /**
31591      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31592      * for bottom-right shadow (defaults to "frame")
31593      */
31594     shadow : "frame",
31595     /**
31596      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31597      */
31598     constrain : false,
31599     /**
31600      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31601      */
31602     completeOnEnter : false,
31603     /**
31604      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31605      */
31606     cancelOnEsc : false,
31607     /**
31608      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31609      */
31610     updateEl : false,
31611
31612     // private
31613     onRender : function(ct, position){
31614         this.el = new Roo.Layer({
31615             shadow: this.shadow,
31616             cls: "x-editor",
31617             parentEl : ct,
31618             shim : this.shim,
31619             shadowOffset:4,
31620             id: this.id,
31621             constrain: this.constrain
31622         });
31623         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31624         if(this.field.msgTarget != 'title'){
31625             this.field.msgTarget = 'qtip';
31626         }
31627         this.field.render(this.el);
31628         if(Roo.isGecko){
31629             this.field.el.dom.setAttribute('autocomplete', 'off');
31630         }
31631         this.field.on("specialkey", this.onSpecialKey, this);
31632         if(this.swallowKeys){
31633             this.field.el.swallowEvent(['keydown','keypress']);
31634         }
31635         this.field.show();
31636         this.field.on("blur", this.onBlur, this);
31637         if(this.field.grow){
31638             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31639         }
31640     },
31641
31642     onSpecialKey : function(field, e)
31643     {
31644         //Roo.log('editor onSpecialKey');
31645         if(this.completeOnEnter && e.getKey() == e.ENTER){
31646             e.stopEvent();
31647             this.completeEdit();
31648             return;
31649         }
31650         // do not fire special key otherwise it might hide close the editor...
31651         if(e.getKey() == e.ENTER){    
31652             return;
31653         }
31654         if(this.cancelOnEsc && e.getKey() == e.ESC){
31655             this.cancelEdit();
31656             return;
31657         } 
31658         this.fireEvent('specialkey', field, e);
31659     
31660     },
31661
31662     /**
31663      * Starts the editing process and shows the editor.
31664      * @param {String/HTMLElement/Element} el The element to edit
31665      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31666       * to the innerHTML of el.
31667      */
31668     startEdit : function(el, value){
31669         if(this.editing){
31670             this.completeEdit();
31671         }
31672         this.boundEl = Roo.get(el);
31673         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31674         if(!this.rendered){
31675             this.render(this.parentEl || document.body);
31676         }
31677         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31678             return;
31679         }
31680         this.startValue = v;
31681         this.field.setValue(v);
31682         if(this.autoSize){
31683             var sz = this.boundEl.getSize();
31684             switch(this.autoSize){
31685                 case "width":
31686                 this.setSize(sz.width,  "");
31687                 break;
31688                 case "height":
31689                 this.setSize("",  sz.height);
31690                 break;
31691                 default:
31692                 this.setSize(sz.width,  sz.height);
31693             }
31694         }
31695         this.el.alignTo(this.boundEl, this.alignment);
31696         this.editing = true;
31697         if(Roo.QuickTips){
31698             Roo.QuickTips.disable();
31699         }
31700         this.show();
31701     },
31702
31703     /**
31704      * Sets the height and width of this editor.
31705      * @param {Number} width The new width
31706      * @param {Number} height The new height
31707      */
31708     setSize : function(w, h){
31709         this.field.setSize(w, h);
31710         if(this.el){
31711             this.el.sync();
31712         }
31713     },
31714
31715     /**
31716      * Realigns the editor to the bound field based on the current alignment config value.
31717      */
31718     realign : function(){
31719         this.el.alignTo(this.boundEl, this.alignment);
31720     },
31721
31722     /**
31723      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31724      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31725      */
31726     completeEdit : function(remainVisible){
31727         if(!this.editing){
31728             return;
31729         }
31730         var v = this.getValue();
31731         if(this.revertInvalid !== false && !this.field.isValid()){
31732             v = this.startValue;
31733             this.cancelEdit(true);
31734         }
31735         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31736             this.editing = false;
31737             this.hide();
31738             return;
31739         }
31740         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31741             this.editing = false;
31742             if(this.updateEl && this.boundEl){
31743                 this.boundEl.update(v);
31744             }
31745             if(remainVisible !== true){
31746                 this.hide();
31747             }
31748             this.fireEvent("complete", this, v, this.startValue);
31749         }
31750     },
31751
31752     // private
31753     onShow : function(){
31754         this.el.show();
31755         if(this.hideEl !== false){
31756             this.boundEl.hide();
31757         }
31758         this.field.show();
31759         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31760             this.fixIEFocus = true;
31761             this.deferredFocus.defer(50, this);
31762         }else{
31763             this.field.focus();
31764         }
31765         this.fireEvent("startedit", this.boundEl, this.startValue);
31766     },
31767
31768     deferredFocus : function(){
31769         if(this.editing){
31770             this.field.focus();
31771         }
31772     },
31773
31774     /**
31775      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31776      * reverted to the original starting value.
31777      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31778      * cancel (defaults to false)
31779      */
31780     cancelEdit : function(remainVisible){
31781         if(this.editing){
31782             this.setValue(this.startValue);
31783             if(remainVisible !== true){
31784                 this.hide();
31785             }
31786         }
31787     },
31788
31789     // private
31790     onBlur : function(){
31791         if(this.allowBlur !== true && this.editing){
31792             this.completeEdit();
31793         }
31794     },
31795
31796     // private
31797     onHide : function(){
31798         if(this.editing){
31799             this.completeEdit();
31800             return;
31801         }
31802         this.field.blur();
31803         if(this.field.collapse){
31804             this.field.collapse();
31805         }
31806         this.el.hide();
31807         if(this.hideEl !== false){
31808             this.boundEl.show();
31809         }
31810         if(Roo.QuickTips){
31811             Roo.QuickTips.enable();
31812         }
31813     },
31814
31815     /**
31816      * Sets the data value of the editor
31817      * @param {Mixed} value Any valid value supported by the underlying field
31818      */
31819     setValue : function(v){
31820         this.field.setValue(v);
31821     },
31822
31823     /**
31824      * Gets the data value of the editor
31825      * @return {Mixed} The data value
31826      */
31827     getValue : function(){
31828         return this.field.getValue();
31829     }
31830 });/*
31831  * Based on:
31832  * Ext JS Library 1.1.1
31833  * Copyright(c) 2006-2007, Ext JS, LLC.
31834  *
31835  * Originally Released Under LGPL - original licence link has changed is not relivant.
31836  *
31837  * Fork - LGPL
31838  * <script type="text/javascript">
31839  */
31840  
31841 /**
31842  * @class Roo.BasicDialog
31843  * @extends Roo.util.Observable
31844  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31845  * <pre><code>
31846 var dlg = new Roo.BasicDialog("my-dlg", {
31847     height: 200,
31848     width: 300,
31849     minHeight: 100,
31850     minWidth: 150,
31851     modal: true,
31852     proxyDrag: true,
31853     shadow: true
31854 });
31855 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31856 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31857 dlg.addButton('Cancel', dlg.hide, dlg);
31858 dlg.show();
31859 </code></pre>
31860   <b>A Dialog should always be a direct child of the body element.</b>
31861  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31862  * @cfg {String} title Default text to display in the title bar (defaults to null)
31863  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31864  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31865  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31866  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31867  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31868  * (defaults to null with no animation)
31869  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31870  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31871  * property for valid values (defaults to 'all')
31872  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31873  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31874  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31875  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31876  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31877  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31878  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31879  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31880  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31881  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31882  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31883  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31884  * draggable = true (defaults to false)
31885  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31886  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31887  * shadow (defaults to false)
31888  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31889  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31890  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31891  * @cfg {Array} buttons Array of buttons
31892  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31893  * @constructor
31894  * Create a new BasicDialog.
31895  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31896  * @param {Object} config Configuration options
31897  */
31898 Roo.BasicDialog = function(el, config){
31899     this.el = Roo.get(el);
31900     var dh = Roo.DomHelper;
31901     if(!this.el && config && config.autoCreate){
31902         if(typeof config.autoCreate == "object"){
31903             if(!config.autoCreate.id){
31904                 config.autoCreate.id = el;
31905             }
31906             this.el = dh.append(document.body,
31907                         config.autoCreate, true);
31908         }else{
31909             this.el = dh.append(document.body,
31910                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31911         }
31912     }
31913     el = this.el;
31914     el.setDisplayed(true);
31915     el.hide = this.hideAction;
31916     this.id = el.id;
31917     el.addClass("x-dlg");
31918
31919     Roo.apply(this, config);
31920
31921     this.proxy = el.createProxy("x-dlg-proxy");
31922     this.proxy.hide = this.hideAction;
31923     this.proxy.setOpacity(.5);
31924     this.proxy.hide();
31925
31926     if(config.width){
31927         el.setWidth(config.width);
31928     }
31929     if(config.height){
31930         el.setHeight(config.height);
31931     }
31932     this.size = el.getSize();
31933     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31934         this.xy = [config.x,config.y];
31935     }else{
31936         this.xy = el.getCenterXY(true);
31937     }
31938     /** The header element @type Roo.Element */
31939     this.header = el.child("> .x-dlg-hd");
31940     /** The body element @type Roo.Element */
31941     this.body = el.child("> .x-dlg-bd");
31942     /** The footer element @type Roo.Element */
31943     this.footer = el.child("> .x-dlg-ft");
31944
31945     if(!this.header){
31946         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31947     }
31948     if(!this.body){
31949         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31950     }
31951
31952     this.header.unselectable();
31953     if(this.title){
31954         this.header.update(this.title);
31955     }
31956     // this element allows the dialog to be focused for keyboard event
31957     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
31958     this.focusEl.swallowEvent("click", true);
31959
31960     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
31961
31962     // wrap the body and footer for special rendering
31963     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
31964     if(this.footer){
31965         this.bwrap.dom.appendChild(this.footer.dom);
31966     }
31967
31968     this.bg = this.el.createChild({
31969         tag: "div", cls:"x-dlg-bg",
31970         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
31971     });
31972     this.centerBg = this.bg.child("div.x-dlg-bg-center");
31973
31974
31975     if(this.autoScroll !== false && !this.autoTabs){
31976         this.body.setStyle("overflow", "auto");
31977     }
31978
31979     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
31980
31981     if(this.closable !== false){
31982         this.el.addClass("x-dlg-closable");
31983         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
31984         this.close.on("click", this.closeClick, this);
31985         this.close.addClassOnOver("x-dlg-close-over");
31986     }
31987     if(this.collapsible !== false){
31988         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
31989         this.collapseBtn.on("click", this.collapseClick, this);
31990         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
31991         this.header.on("dblclick", this.collapseClick, this);
31992     }
31993     if(this.resizable !== false){
31994         this.el.addClass("x-dlg-resizable");
31995         this.resizer = new Roo.Resizable(el, {
31996             minWidth: this.minWidth || 80,
31997             minHeight:this.minHeight || 80,
31998             handles: this.resizeHandles || "all",
31999             pinned: true
32000         });
32001         this.resizer.on("beforeresize", this.beforeResize, this);
32002         this.resizer.on("resize", this.onResize, this);
32003     }
32004     if(this.draggable !== false){
32005         el.addClass("x-dlg-draggable");
32006         if (!this.proxyDrag) {
32007             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32008         }
32009         else {
32010             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32011         }
32012         dd.setHandleElId(this.header.id);
32013         dd.endDrag = this.endMove.createDelegate(this);
32014         dd.startDrag = this.startMove.createDelegate(this);
32015         dd.onDrag = this.onDrag.createDelegate(this);
32016         dd.scroll = false;
32017         this.dd = dd;
32018     }
32019     if(this.modal){
32020         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32021         this.mask.enableDisplayMode("block");
32022         this.mask.hide();
32023         this.el.addClass("x-dlg-modal");
32024     }
32025     if(this.shadow){
32026         this.shadow = new Roo.Shadow({
32027             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32028             offset : this.shadowOffset
32029         });
32030     }else{
32031         this.shadowOffset = 0;
32032     }
32033     if(Roo.useShims && this.shim !== false){
32034         this.shim = this.el.createShim();
32035         this.shim.hide = this.hideAction;
32036         this.shim.hide();
32037     }else{
32038         this.shim = false;
32039     }
32040     if(this.autoTabs){
32041         this.initTabs();
32042     }
32043     if (this.buttons) { 
32044         var bts= this.buttons;
32045         this.buttons = [];
32046         Roo.each(bts, function(b) {
32047             this.addButton(b);
32048         }, this);
32049     }
32050     
32051     
32052     this.addEvents({
32053         /**
32054          * @event keydown
32055          * Fires when a key is pressed
32056          * @param {Roo.BasicDialog} this
32057          * @param {Roo.EventObject} e
32058          */
32059         "keydown" : true,
32060         /**
32061          * @event move
32062          * Fires when this dialog is moved by the user.
32063          * @param {Roo.BasicDialog} this
32064          * @param {Number} x The new page X
32065          * @param {Number} y The new page Y
32066          */
32067         "move" : true,
32068         /**
32069          * @event resize
32070          * Fires when this dialog is resized by the user.
32071          * @param {Roo.BasicDialog} this
32072          * @param {Number} width The new width
32073          * @param {Number} height The new height
32074          */
32075         "resize" : true,
32076         /**
32077          * @event beforehide
32078          * Fires before this dialog is hidden.
32079          * @param {Roo.BasicDialog} this
32080          */
32081         "beforehide" : true,
32082         /**
32083          * @event hide
32084          * Fires when this dialog is hidden.
32085          * @param {Roo.BasicDialog} this
32086          */
32087         "hide" : true,
32088         /**
32089          * @event beforeshow
32090          * Fires before this dialog is shown.
32091          * @param {Roo.BasicDialog} this
32092          */
32093         "beforeshow" : true,
32094         /**
32095          * @event show
32096          * Fires when this dialog is shown.
32097          * @param {Roo.BasicDialog} this
32098          */
32099         "show" : true
32100     });
32101     el.on("keydown", this.onKeyDown, this);
32102     el.on("mousedown", this.toFront, this);
32103     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32104     this.el.hide();
32105     Roo.DialogManager.register(this);
32106     Roo.BasicDialog.superclass.constructor.call(this);
32107 };
32108
32109 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32110     shadowOffset: Roo.isIE ? 6 : 5,
32111     minHeight: 80,
32112     minWidth: 200,
32113     minButtonWidth: 75,
32114     defaultButton: null,
32115     buttonAlign: "right",
32116     tabTag: 'div',
32117     firstShow: true,
32118
32119     /**
32120      * Sets the dialog title text
32121      * @param {String} text The title text to display
32122      * @return {Roo.BasicDialog} this
32123      */
32124     setTitle : function(text){
32125         this.header.update(text);
32126         return this;
32127     },
32128
32129     // private
32130     closeClick : function(){
32131         this.hide();
32132     },
32133
32134     // private
32135     collapseClick : function(){
32136         this[this.collapsed ? "expand" : "collapse"]();
32137     },
32138
32139     /**
32140      * Collapses the dialog to its minimized state (only the title bar is visible).
32141      * Equivalent to the user clicking the collapse dialog button.
32142      */
32143     collapse : function(){
32144         if(!this.collapsed){
32145             this.collapsed = true;
32146             this.el.addClass("x-dlg-collapsed");
32147             this.restoreHeight = this.el.getHeight();
32148             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32149         }
32150     },
32151
32152     /**
32153      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32154      * clicking the expand dialog button.
32155      */
32156     expand : function(){
32157         if(this.collapsed){
32158             this.collapsed = false;
32159             this.el.removeClass("x-dlg-collapsed");
32160             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32161         }
32162     },
32163
32164     /**
32165      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32166      * @return {Roo.TabPanel} The tabs component
32167      */
32168     initTabs : function(){
32169         var tabs = this.getTabs();
32170         while(tabs.getTab(0)){
32171             tabs.removeTab(0);
32172         }
32173         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32174             var dom = el.dom;
32175             tabs.addTab(Roo.id(dom), dom.title);
32176             dom.title = "";
32177         });
32178         tabs.activate(0);
32179         return tabs;
32180     },
32181
32182     // private
32183     beforeResize : function(){
32184         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32185     },
32186
32187     // private
32188     onResize : function(){
32189         this.refreshSize();
32190         this.syncBodyHeight();
32191         this.adjustAssets();
32192         this.focus();
32193         this.fireEvent("resize", this, this.size.width, this.size.height);
32194     },
32195
32196     // private
32197     onKeyDown : function(e){
32198         if(this.isVisible()){
32199             this.fireEvent("keydown", this, e);
32200         }
32201     },
32202
32203     /**
32204      * Resizes the dialog.
32205      * @param {Number} width
32206      * @param {Number} height
32207      * @return {Roo.BasicDialog} this
32208      */
32209     resizeTo : function(width, height){
32210         this.el.setSize(width, height);
32211         this.size = {width: width, height: height};
32212         this.syncBodyHeight();
32213         if(this.fixedcenter){
32214             this.center();
32215         }
32216         if(this.isVisible()){
32217             this.constrainXY();
32218             this.adjustAssets();
32219         }
32220         this.fireEvent("resize", this, width, height);
32221         return this;
32222     },
32223
32224
32225     /**
32226      * Resizes the dialog to fit the specified content size.
32227      * @param {Number} width
32228      * @param {Number} height
32229      * @return {Roo.BasicDialog} this
32230      */
32231     setContentSize : function(w, h){
32232         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32233         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32234         //if(!this.el.isBorderBox()){
32235             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32236             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32237         //}
32238         if(this.tabs){
32239             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32240             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32241         }
32242         this.resizeTo(w, h);
32243         return this;
32244     },
32245
32246     /**
32247      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32248      * executed in response to a particular key being pressed while the dialog is active.
32249      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32250      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32251      * @param {Function} fn The function to call
32252      * @param {Object} scope (optional) The scope of the function
32253      * @return {Roo.BasicDialog} this
32254      */
32255     addKeyListener : function(key, fn, scope){
32256         var keyCode, shift, ctrl, alt;
32257         if(typeof key == "object" && !(key instanceof Array)){
32258             keyCode = key["key"];
32259             shift = key["shift"];
32260             ctrl = key["ctrl"];
32261             alt = key["alt"];
32262         }else{
32263             keyCode = key;
32264         }
32265         var handler = function(dlg, e){
32266             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32267                 var k = e.getKey();
32268                 if(keyCode instanceof Array){
32269                     for(var i = 0, len = keyCode.length; i < len; i++){
32270                         if(keyCode[i] == k){
32271                           fn.call(scope || window, dlg, k, e);
32272                           return;
32273                         }
32274                     }
32275                 }else{
32276                     if(k == keyCode){
32277                         fn.call(scope || window, dlg, k, e);
32278                     }
32279                 }
32280             }
32281         };
32282         this.on("keydown", handler);
32283         return this;
32284     },
32285
32286     /**
32287      * Returns the TabPanel component (creates it if it doesn't exist).
32288      * Note: If you wish to simply check for the existence of tabs without creating them,
32289      * check for a null 'tabs' property.
32290      * @return {Roo.TabPanel} The tabs component
32291      */
32292     getTabs : function(){
32293         if(!this.tabs){
32294             this.el.addClass("x-dlg-auto-tabs");
32295             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32296             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32297         }
32298         return this.tabs;
32299     },
32300
32301     /**
32302      * Adds a button to the footer section of the dialog.
32303      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32304      * object or a valid Roo.DomHelper element config
32305      * @param {Function} handler The function called when the button is clicked
32306      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32307      * @return {Roo.Button} The new button
32308      */
32309     addButton : function(config, handler, scope){
32310         var dh = Roo.DomHelper;
32311         if(!this.footer){
32312             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32313         }
32314         if(!this.btnContainer){
32315             var tb = this.footer.createChild({
32316
32317                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32318                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32319             }, null, true);
32320             this.btnContainer = tb.firstChild.firstChild.firstChild;
32321         }
32322         var bconfig = {
32323             handler: handler,
32324             scope: scope,
32325             minWidth: this.minButtonWidth,
32326             hideParent:true
32327         };
32328         if(typeof config == "string"){
32329             bconfig.text = config;
32330         }else{
32331             if(config.tag){
32332                 bconfig.dhconfig = config;
32333             }else{
32334                 Roo.apply(bconfig, config);
32335             }
32336         }
32337         var fc = false;
32338         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32339             bconfig.position = Math.max(0, bconfig.position);
32340             fc = this.btnContainer.childNodes[bconfig.position];
32341         }
32342          
32343         var btn = new Roo.Button(
32344             fc ? 
32345                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32346                 : this.btnContainer.appendChild(document.createElement("td")),
32347             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32348             bconfig
32349         );
32350         this.syncBodyHeight();
32351         if(!this.buttons){
32352             /**
32353              * Array of all the buttons that have been added to this dialog via addButton
32354              * @type Array
32355              */
32356             this.buttons = [];
32357         }
32358         this.buttons.push(btn);
32359         return btn;
32360     },
32361
32362     /**
32363      * Sets the default button to be focused when the dialog is displayed.
32364      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32365      * @return {Roo.BasicDialog} this
32366      */
32367     setDefaultButton : function(btn){
32368         this.defaultButton = btn;
32369         return this;
32370     },
32371
32372     // private
32373     getHeaderFooterHeight : function(safe){
32374         var height = 0;
32375         if(this.header){
32376            height += this.header.getHeight();
32377         }
32378         if(this.footer){
32379            var fm = this.footer.getMargins();
32380             height += (this.footer.getHeight()+fm.top+fm.bottom);
32381         }
32382         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32383         height += this.centerBg.getPadding("tb");
32384         return height;
32385     },
32386
32387     // private
32388     syncBodyHeight : function()
32389     {
32390         var bd = this.body, // the text
32391             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32392             bw = this.bwrap;
32393         var height = this.size.height - this.getHeaderFooterHeight(false);
32394         bd.setHeight(height-bd.getMargins("tb"));
32395         var hh = this.header.getHeight();
32396         var h = this.size.height-hh;
32397         cb.setHeight(h);
32398         
32399         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32400         bw.setHeight(h-cb.getPadding("tb"));
32401         
32402         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32403         bd.setWidth(bw.getWidth(true));
32404         if(this.tabs){
32405             this.tabs.syncHeight();
32406             if(Roo.isIE){
32407                 this.tabs.el.repaint();
32408             }
32409         }
32410     },
32411
32412     /**
32413      * Restores the previous state of the dialog if Roo.state is configured.
32414      * @return {Roo.BasicDialog} this
32415      */
32416     restoreState : function(){
32417         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32418         if(box && box.width){
32419             this.xy = [box.x, box.y];
32420             this.resizeTo(box.width, box.height);
32421         }
32422         return this;
32423     },
32424
32425     // private
32426     beforeShow : function(){
32427         this.expand();
32428         if(this.fixedcenter){
32429             this.xy = this.el.getCenterXY(true);
32430         }
32431         if(this.modal){
32432             Roo.get(document.body).addClass("x-body-masked");
32433             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32434             this.mask.show();
32435         }
32436         this.constrainXY();
32437     },
32438
32439     // private
32440     animShow : function(){
32441         var b = Roo.get(this.animateTarget).getBox();
32442         this.proxy.setSize(b.width, b.height);
32443         this.proxy.setLocation(b.x, b.y);
32444         this.proxy.show();
32445         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32446                     true, .35, this.showEl.createDelegate(this));
32447     },
32448
32449     /**
32450      * Shows the dialog.
32451      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32452      * @return {Roo.BasicDialog} this
32453      */
32454     show : function(animateTarget){
32455         if (this.fireEvent("beforeshow", this) === false){
32456             return;
32457         }
32458         if(this.syncHeightBeforeShow){
32459             this.syncBodyHeight();
32460         }else if(this.firstShow){
32461             this.firstShow = false;
32462             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32463         }
32464         this.animateTarget = animateTarget || this.animateTarget;
32465         if(!this.el.isVisible()){
32466             this.beforeShow();
32467             if(this.animateTarget && Roo.get(this.animateTarget)){
32468                 this.animShow();
32469             }else{
32470                 this.showEl();
32471             }
32472         }
32473         return this;
32474     },
32475
32476     // private
32477     showEl : function(){
32478         this.proxy.hide();
32479         this.el.setXY(this.xy);
32480         this.el.show();
32481         this.adjustAssets(true);
32482         this.toFront();
32483         this.focus();
32484         // IE peekaboo bug - fix found by Dave Fenwick
32485         if(Roo.isIE){
32486             this.el.repaint();
32487         }
32488         this.fireEvent("show", this);
32489     },
32490
32491     /**
32492      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32493      * dialog itself will receive focus.
32494      */
32495     focus : function(){
32496         if(this.defaultButton){
32497             this.defaultButton.focus();
32498         }else{
32499             this.focusEl.focus();
32500         }
32501     },
32502
32503     // private
32504     constrainXY : function(){
32505         if(this.constraintoviewport !== false){
32506             if(!this.viewSize){
32507                 if(this.container){
32508                     var s = this.container.getSize();
32509                     this.viewSize = [s.width, s.height];
32510                 }else{
32511                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32512                 }
32513             }
32514             var s = Roo.get(this.container||document).getScroll();
32515
32516             var x = this.xy[0], y = this.xy[1];
32517             var w = this.size.width, h = this.size.height;
32518             var vw = this.viewSize[0], vh = this.viewSize[1];
32519             // only move it if it needs it
32520             var moved = false;
32521             // first validate right/bottom
32522             if(x + w > vw+s.left){
32523                 x = vw - w;
32524                 moved = true;
32525             }
32526             if(y + h > vh+s.top){
32527                 y = vh - h;
32528                 moved = true;
32529             }
32530             // then make sure top/left isn't negative
32531             if(x < s.left){
32532                 x = s.left;
32533                 moved = true;
32534             }
32535             if(y < s.top){
32536                 y = s.top;
32537                 moved = true;
32538             }
32539             if(moved){
32540                 // cache xy
32541                 this.xy = [x, y];
32542                 if(this.isVisible()){
32543                     this.el.setLocation(x, y);
32544                     this.adjustAssets();
32545                 }
32546             }
32547         }
32548     },
32549
32550     // private
32551     onDrag : function(){
32552         if(!this.proxyDrag){
32553             this.xy = this.el.getXY();
32554             this.adjustAssets();
32555         }
32556     },
32557
32558     // private
32559     adjustAssets : function(doShow){
32560         var x = this.xy[0], y = this.xy[1];
32561         var w = this.size.width, h = this.size.height;
32562         if(doShow === true){
32563             if(this.shadow){
32564                 this.shadow.show(this.el);
32565             }
32566             if(this.shim){
32567                 this.shim.show();
32568             }
32569         }
32570         if(this.shadow && this.shadow.isVisible()){
32571             this.shadow.show(this.el);
32572         }
32573         if(this.shim && this.shim.isVisible()){
32574             this.shim.setBounds(x, y, w, h);
32575         }
32576     },
32577
32578     // private
32579     adjustViewport : function(w, h){
32580         if(!w || !h){
32581             w = Roo.lib.Dom.getViewWidth();
32582             h = Roo.lib.Dom.getViewHeight();
32583         }
32584         // cache the size
32585         this.viewSize = [w, h];
32586         if(this.modal && this.mask.isVisible()){
32587             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32588             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32589         }
32590         if(this.isVisible()){
32591             this.constrainXY();
32592         }
32593     },
32594
32595     /**
32596      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32597      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32598      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32599      */
32600     destroy : function(removeEl){
32601         if(this.isVisible()){
32602             this.animateTarget = null;
32603             this.hide();
32604         }
32605         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32606         if(this.tabs){
32607             this.tabs.destroy(removeEl);
32608         }
32609         Roo.destroy(
32610              this.shim,
32611              this.proxy,
32612              this.resizer,
32613              this.close,
32614              this.mask
32615         );
32616         if(this.dd){
32617             this.dd.unreg();
32618         }
32619         if(this.buttons){
32620            for(var i = 0, len = this.buttons.length; i < len; i++){
32621                this.buttons[i].destroy();
32622            }
32623         }
32624         this.el.removeAllListeners();
32625         if(removeEl === true){
32626             this.el.update("");
32627             this.el.remove();
32628         }
32629         Roo.DialogManager.unregister(this);
32630     },
32631
32632     // private
32633     startMove : function(){
32634         if(this.proxyDrag){
32635             this.proxy.show();
32636         }
32637         if(this.constraintoviewport !== false){
32638             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32639         }
32640     },
32641
32642     // private
32643     endMove : function(){
32644         if(!this.proxyDrag){
32645             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32646         }else{
32647             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32648             this.proxy.hide();
32649         }
32650         this.refreshSize();
32651         this.adjustAssets();
32652         this.focus();
32653         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32654     },
32655
32656     /**
32657      * Brings this dialog to the front of any other visible dialogs
32658      * @return {Roo.BasicDialog} this
32659      */
32660     toFront : function(){
32661         Roo.DialogManager.bringToFront(this);
32662         return this;
32663     },
32664
32665     /**
32666      * Sends this dialog to the back (under) of any other visible dialogs
32667      * @return {Roo.BasicDialog} this
32668      */
32669     toBack : function(){
32670         Roo.DialogManager.sendToBack(this);
32671         return this;
32672     },
32673
32674     /**
32675      * Centers this dialog in the viewport
32676      * @return {Roo.BasicDialog} this
32677      */
32678     center : function(){
32679         var xy = this.el.getCenterXY(true);
32680         this.moveTo(xy[0], xy[1]);
32681         return this;
32682     },
32683
32684     /**
32685      * Moves the dialog's top-left corner to the specified point
32686      * @param {Number} x
32687      * @param {Number} y
32688      * @return {Roo.BasicDialog} this
32689      */
32690     moveTo : function(x, y){
32691         this.xy = [x,y];
32692         if(this.isVisible()){
32693             this.el.setXY(this.xy);
32694             this.adjustAssets();
32695         }
32696         return this;
32697     },
32698
32699     /**
32700      * Aligns the dialog to the specified element
32701      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32702      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32703      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32704      * @return {Roo.BasicDialog} this
32705      */
32706     alignTo : function(element, position, offsets){
32707         this.xy = this.el.getAlignToXY(element, position, offsets);
32708         if(this.isVisible()){
32709             this.el.setXY(this.xy);
32710             this.adjustAssets();
32711         }
32712         return this;
32713     },
32714
32715     /**
32716      * Anchors an element to another element and realigns it when the window is resized.
32717      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32718      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32719      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32720      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32721      * is a number, it is used as the buffer delay (defaults to 50ms).
32722      * @return {Roo.BasicDialog} this
32723      */
32724     anchorTo : function(el, alignment, offsets, monitorScroll){
32725         var action = function(){
32726             this.alignTo(el, alignment, offsets);
32727         };
32728         Roo.EventManager.onWindowResize(action, this);
32729         var tm = typeof monitorScroll;
32730         if(tm != 'undefined'){
32731             Roo.EventManager.on(window, 'scroll', action, this,
32732                 {buffer: tm == 'number' ? monitorScroll : 50});
32733         }
32734         action.call(this);
32735         return this;
32736     },
32737
32738     /**
32739      * Returns true if the dialog is visible
32740      * @return {Boolean}
32741      */
32742     isVisible : function(){
32743         return this.el.isVisible();
32744     },
32745
32746     // private
32747     animHide : function(callback){
32748         var b = Roo.get(this.animateTarget).getBox();
32749         this.proxy.show();
32750         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32751         this.el.hide();
32752         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32753                     this.hideEl.createDelegate(this, [callback]));
32754     },
32755
32756     /**
32757      * Hides the dialog.
32758      * @param {Function} callback (optional) Function to call when the dialog is hidden
32759      * @return {Roo.BasicDialog} this
32760      */
32761     hide : function(callback){
32762         if (this.fireEvent("beforehide", this) === false){
32763             return;
32764         }
32765         if(this.shadow){
32766             this.shadow.hide();
32767         }
32768         if(this.shim) {
32769           this.shim.hide();
32770         }
32771         // sometimes animateTarget seems to get set.. causing problems...
32772         // this just double checks..
32773         if(this.animateTarget && Roo.get(this.animateTarget)) {
32774            this.animHide(callback);
32775         }else{
32776             this.el.hide();
32777             this.hideEl(callback);
32778         }
32779         return this;
32780     },
32781
32782     // private
32783     hideEl : function(callback){
32784         this.proxy.hide();
32785         if(this.modal){
32786             this.mask.hide();
32787             Roo.get(document.body).removeClass("x-body-masked");
32788         }
32789         this.fireEvent("hide", this);
32790         if(typeof callback == "function"){
32791             callback();
32792         }
32793     },
32794
32795     // private
32796     hideAction : function(){
32797         this.setLeft("-10000px");
32798         this.setTop("-10000px");
32799         this.setStyle("visibility", "hidden");
32800     },
32801
32802     // private
32803     refreshSize : function(){
32804         this.size = this.el.getSize();
32805         this.xy = this.el.getXY();
32806         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32807     },
32808
32809     // private
32810     // z-index is managed by the DialogManager and may be overwritten at any time
32811     setZIndex : function(index){
32812         if(this.modal){
32813             this.mask.setStyle("z-index", index);
32814         }
32815         if(this.shim){
32816             this.shim.setStyle("z-index", ++index);
32817         }
32818         if(this.shadow){
32819             this.shadow.setZIndex(++index);
32820         }
32821         this.el.setStyle("z-index", ++index);
32822         if(this.proxy){
32823             this.proxy.setStyle("z-index", ++index);
32824         }
32825         if(this.resizer){
32826             this.resizer.proxy.setStyle("z-index", ++index);
32827         }
32828
32829         this.lastZIndex = index;
32830     },
32831
32832     /**
32833      * Returns the element for this dialog
32834      * @return {Roo.Element} The underlying dialog Element
32835      */
32836     getEl : function(){
32837         return this.el;
32838     }
32839 });
32840
32841 /**
32842  * @class Roo.DialogManager
32843  * Provides global access to BasicDialogs that have been created and
32844  * support for z-indexing (layering) multiple open dialogs.
32845  */
32846 Roo.DialogManager = function(){
32847     var list = {};
32848     var accessList = [];
32849     var front = null;
32850
32851     // private
32852     var sortDialogs = function(d1, d2){
32853         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32854     };
32855
32856     // private
32857     var orderDialogs = function(){
32858         accessList.sort(sortDialogs);
32859         var seed = Roo.DialogManager.zseed;
32860         for(var i = 0, len = accessList.length; i < len; i++){
32861             var dlg = accessList[i];
32862             if(dlg){
32863                 dlg.setZIndex(seed + (i*10));
32864             }
32865         }
32866     };
32867
32868     return {
32869         /**
32870          * The starting z-index for BasicDialogs (defaults to 9000)
32871          * @type Number The z-index value
32872          */
32873         zseed : 9000,
32874
32875         // private
32876         register : function(dlg){
32877             list[dlg.id] = dlg;
32878             accessList.push(dlg);
32879         },
32880
32881         // private
32882         unregister : function(dlg){
32883             delete list[dlg.id];
32884             var i=0;
32885             var len=0;
32886             if(!accessList.indexOf){
32887                 for(  i = 0, len = accessList.length; i < len; i++){
32888                     if(accessList[i] == dlg){
32889                         accessList.splice(i, 1);
32890                         return;
32891                     }
32892                 }
32893             }else{
32894                  i = accessList.indexOf(dlg);
32895                 if(i != -1){
32896                     accessList.splice(i, 1);
32897                 }
32898             }
32899         },
32900
32901         /**
32902          * Gets a registered dialog by id
32903          * @param {String/Object} id The id of the dialog or a dialog
32904          * @return {Roo.BasicDialog} this
32905          */
32906         get : function(id){
32907             return typeof id == "object" ? id : list[id];
32908         },
32909
32910         /**
32911          * Brings the specified dialog to the front
32912          * @param {String/Object} dlg The id of the dialog or a dialog
32913          * @return {Roo.BasicDialog} this
32914          */
32915         bringToFront : function(dlg){
32916             dlg = this.get(dlg);
32917             if(dlg != front){
32918                 front = dlg;
32919                 dlg._lastAccess = new Date().getTime();
32920                 orderDialogs();
32921             }
32922             return dlg;
32923         },
32924
32925         /**
32926          * Sends the specified dialog to the back
32927          * @param {String/Object} dlg The id of the dialog or a dialog
32928          * @return {Roo.BasicDialog} this
32929          */
32930         sendToBack : function(dlg){
32931             dlg = this.get(dlg);
32932             dlg._lastAccess = -(new Date().getTime());
32933             orderDialogs();
32934             return dlg;
32935         },
32936
32937         /**
32938          * Hides all dialogs
32939          */
32940         hideAll : function(){
32941             for(var id in list){
32942                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32943                     list[id].hide();
32944                 }
32945             }
32946         }
32947     };
32948 }();
32949
32950 /**
32951  * @class Roo.LayoutDialog
32952  * @extends Roo.BasicDialog
32953  * Dialog which provides adjustments for working with a layout in a Dialog.
32954  * Add your necessary layout config options to the dialog's config.<br>
32955  * Example usage (including a nested layout):
32956  * <pre><code>
32957 if(!dialog){
32958     dialog = new Roo.LayoutDialog("download-dlg", {
32959         modal: true,
32960         width:600,
32961         height:450,
32962         shadow:true,
32963         minWidth:500,
32964         minHeight:350,
32965         autoTabs:true,
32966         proxyDrag:true,
32967         // layout config merges with the dialog config
32968         center:{
32969             tabPosition: "top",
32970             alwaysShowTabs: true
32971         }
32972     });
32973     dialog.addKeyListener(27, dialog.hide, dialog);
32974     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
32975     dialog.addButton("Build It!", this.getDownload, this);
32976
32977     // we can even add nested layouts
32978     var innerLayout = new Roo.BorderLayout("dl-inner", {
32979         east: {
32980             initialSize: 200,
32981             autoScroll:true,
32982             split:true
32983         },
32984         center: {
32985             autoScroll:true
32986         }
32987     });
32988     innerLayout.beginUpdate();
32989     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
32990     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
32991     innerLayout.endUpdate(true);
32992
32993     var layout = dialog.getLayout();
32994     layout.beginUpdate();
32995     layout.add("center", new Roo.ContentPanel("standard-panel",
32996                         {title: "Download the Source", fitToFrame:true}));
32997     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
32998                {title: "Build your own roo.js"}));
32999     layout.getRegion("center").showPanel(sp);
33000     layout.endUpdate();
33001 }
33002 </code></pre>
33003     * @constructor
33004     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33005     * @param {Object} config configuration options
33006   */
33007 Roo.LayoutDialog = function(el, cfg){
33008     
33009     var config=  cfg;
33010     if (typeof(cfg) == 'undefined') {
33011         config = Roo.apply({}, el);
33012         // not sure why we use documentElement here.. - it should always be body.
33013         // IE7 borks horribly if we use documentElement.
33014         // webkit also does not like documentElement - it creates a body element...
33015         el = Roo.get( document.body || document.documentElement ).createChild();
33016         //config.autoCreate = true;
33017     }
33018     
33019     
33020     config.autoTabs = false;
33021     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33022     this.body.setStyle({overflow:"hidden", position:"relative"});
33023     this.layout = new Roo.BorderLayout(this.body.dom, config);
33024     this.layout.monitorWindowResize = false;
33025     this.el.addClass("x-dlg-auto-layout");
33026     // fix case when center region overwrites center function
33027     this.center = Roo.BasicDialog.prototype.center;
33028     this.on("show", this.layout.layout, this.layout, true);
33029     if (config.items) {
33030         var xitems = config.items;
33031         delete config.items;
33032         Roo.each(xitems, this.addxtype, this);
33033     }
33034     
33035     
33036 };
33037 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33038     /**
33039      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33040      * @deprecated
33041      */
33042     endUpdate : function(){
33043         this.layout.endUpdate();
33044     },
33045
33046     /**
33047      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33048      *  @deprecated
33049      */
33050     beginUpdate : function(){
33051         this.layout.beginUpdate();
33052     },
33053
33054     /**
33055      * Get the BorderLayout for this dialog
33056      * @return {Roo.BorderLayout}
33057      */
33058     getLayout : function(){
33059         return this.layout;
33060     },
33061
33062     showEl : function(){
33063         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33064         if(Roo.isIE7){
33065             this.layout.layout();
33066         }
33067     },
33068
33069     // private
33070     // Use the syncHeightBeforeShow config option to control this automatically
33071     syncBodyHeight : function(){
33072         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33073         if(this.layout){this.layout.layout();}
33074     },
33075     
33076       /**
33077      * Add an xtype element (actually adds to the layout.)
33078      * @return {Object} xdata xtype object data.
33079      */
33080     
33081     addxtype : function(c) {
33082         return this.layout.addxtype(c);
33083     }
33084 });/*
33085  * Based on:
33086  * Ext JS Library 1.1.1
33087  * Copyright(c) 2006-2007, Ext JS, LLC.
33088  *
33089  * Originally Released Under LGPL - original licence link has changed is not relivant.
33090  *
33091  * Fork - LGPL
33092  * <script type="text/javascript">
33093  */
33094  
33095 /**
33096  * @class Roo.MessageBox
33097  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33098  * Example usage:
33099  *<pre><code>
33100 // Basic alert:
33101 Roo.Msg.alert('Status', 'Changes saved successfully.');
33102
33103 // Prompt for user data:
33104 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33105     if (btn == 'ok'){
33106         // process text value...
33107     }
33108 });
33109
33110 // Show a dialog using config options:
33111 Roo.Msg.show({
33112    title:'Save Changes?',
33113    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33114    buttons: Roo.Msg.YESNOCANCEL,
33115    fn: processResult,
33116    animEl: 'elId'
33117 });
33118 </code></pre>
33119  * @singleton
33120  */
33121 Roo.MessageBox = function(){
33122     var dlg, opt, mask, waitTimer;
33123     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33124     var buttons, activeTextEl, bwidth;
33125
33126     // private
33127     var handleButton = function(button){
33128         dlg.hide();
33129         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33130     };
33131
33132     // private
33133     var handleHide = function(){
33134         if(opt && opt.cls){
33135             dlg.el.removeClass(opt.cls);
33136         }
33137         if(waitTimer){
33138             Roo.TaskMgr.stop(waitTimer);
33139             waitTimer = null;
33140         }
33141     };
33142
33143     // private
33144     var updateButtons = function(b){
33145         var width = 0;
33146         if(!b){
33147             buttons["ok"].hide();
33148             buttons["cancel"].hide();
33149             buttons["yes"].hide();
33150             buttons["no"].hide();
33151             dlg.footer.dom.style.display = 'none';
33152             return width;
33153         }
33154         dlg.footer.dom.style.display = '';
33155         for(var k in buttons){
33156             if(typeof buttons[k] != "function"){
33157                 if(b[k]){
33158                     buttons[k].show();
33159                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33160                     width += buttons[k].el.getWidth()+15;
33161                 }else{
33162                     buttons[k].hide();
33163                 }
33164             }
33165         }
33166         return width;
33167     };
33168
33169     // private
33170     var handleEsc = function(d, k, e){
33171         if(opt && opt.closable !== false){
33172             dlg.hide();
33173         }
33174         if(e){
33175             e.stopEvent();
33176         }
33177     };
33178
33179     return {
33180         /**
33181          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33182          * @return {Roo.BasicDialog} The BasicDialog element
33183          */
33184         getDialog : function(){
33185            if(!dlg){
33186                 dlg = new Roo.BasicDialog("x-msg-box", {
33187                     autoCreate : true,
33188                     shadow: true,
33189                     draggable: true,
33190                     resizable:false,
33191                     constraintoviewport:false,
33192                     fixedcenter:true,
33193                     collapsible : false,
33194                     shim:true,
33195                     modal: true,
33196                     width:400, height:100,
33197                     buttonAlign:"center",
33198                     closeClick : function(){
33199                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33200                             handleButton("no");
33201                         }else{
33202                             handleButton("cancel");
33203                         }
33204                     }
33205                 });
33206                 dlg.on("hide", handleHide);
33207                 mask = dlg.mask;
33208                 dlg.addKeyListener(27, handleEsc);
33209                 buttons = {};
33210                 var bt = this.buttonText;
33211                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33212                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33213                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33214                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33215                 bodyEl = dlg.body.createChild({
33216
33217                     html:'<span class="roo-mb-text"></span><br /><input type="text" class="roo-mb-input" /><textarea class="roo-mb-textarea"></textarea><div class="roo-mb-progress-wrap"><div class="roo-mb-progress"><div class="roo-mb-progress-bar">&#160;</div></div></div>'
33218                 });
33219                 msgEl = bodyEl.dom.firstChild;
33220                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33221                 textboxEl.enableDisplayMode();
33222                 textboxEl.addKeyListener([10,13], function(){
33223                     if(dlg.isVisible() && opt && opt.buttons){
33224                         if(opt.buttons.ok){
33225                             handleButton("ok");
33226                         }else if(opt.buttons.yes){
33227                             handleButton("yes");
33228                         }
33229                     }
33230                 });
33231                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33232                 textareaEl.enableDisplayMode();
33233                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33234                 progressEl.enableDisplayMode();
33235                 var pf = progressEl.dom.firstChild;
33236                 if (pf) {
33237                     pp = Roo.get(pf.firstChild);
33238                     pp.setHeight(pf.offsetHeight);
33239                 }
33240                 
33241             }
33242             return dlg;
33243         },
33244
33245         /**
33246          * Updates the message box body text
33247          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33248          * the XHTML-compliant non-breaking space character '&amp;#160;')
33249          * @return {Roo.MessageBox} This message box
33250          */
33251         updateText : function(text){
33252             if(!dlg.isVisible() && !opt.width){
33253                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33254             }
33255             msgEl.innerHTML = text || '&#160;';
33256       
33257             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33258             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33259             var w = Math.max(
33260                     Math.min(opt.width || cw , this.maxWidth), 
33261                     Math.max(opt.minWidth || this.minWidth, bwidth)
33262             );
33263             if(opt.prompt){
33264                 activeTextEl.setWidth(w);
33265             }
33266             if(dlg.isVisible()){
33267                 dlg.fixedcenter = false;
33268             }
33269             // to big, make it scroll. = But as usual stupid IE does not support
33270             // !important..
33271             
33272             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33273                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33274                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33275             } else {
33276                 bodyEl.dom.style.height = '';
33277                 bodyEl.dom.style.overflowY = '';
33278             }
33279             if (cw > w) {
33280                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33281             } else {
33282                 bodyEl.dom.style.overflowX = '';
33283             }
33284             
33285             dlg.setContentSize(w, bodyEl.getHeight());
33286             if(dlg.isVisible()){
33287                 dlg.fixedcenter = true;
33288             }
33289             return this;
33290         },
33291
33292         /**
33293          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33294          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33295          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33296          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33297          * @return {Roo.MessageBox} This message box
33298          */
33299         updateProgress : function(value, text){
33300             if(text){
33301                 this.updateText(text);
33302             }
33303             if (pp) { // weird bug on my firefox - for some reason this is not defined
33304                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33305             }
33306             return this;
33307         },        
33308
33309         /**
33310          * Returns true if the message box is currently displayed
33311          * @return {Boolean} True if the message box is visible, else false
33312          */
33313         isVisible : function(){
33314             return dlg && dlg.isVisible();  
33315         },
33316
33317         /**
33318          * Hides the message box if it is displayed
33319          */
33320         hide : function(){
33321             if(this.isVisible()){
33322                 dlg.hide();
33323             }  
33324         },
33325
33326         /**
33327          * Displays a new message box, or reinitializes an existing message box, based on the config options
33328          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33329          * The following config object properties are supported:
33330          * <pre>
33331 Property    Type             Description
33332 ----------  ---------------  ------------------------------------------------------------------------------------
33333 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33334                                    closes (defaults to undefined)
33335 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33336                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33337 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33338                                    progress and wait dialogs will ignore this property and always hide the
33339                                    close button as they can only be closed programmatically.
33340 cls               String           A custom CSS class to apply to the message box element
33341 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33342                                    displayed (defaults to 75)
33343 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33344                                    function will be btn (the name of the button that was clicked, if applicable,
33345                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33346                                    Progress and wait dialogs will ignore this option since they do not respond to
33347                                    user actions and can only be closed programmatically, so any required function
33348                                    should be called by the same code after it closes the dialog.
33349 icon              String           A CSS class that provides a background image to be used as an icon for
33350                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33351 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33352 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33353 modal             Boolean          False to allow user interaction with the page while the message box is
33354                                    displayed (defaults to true)
33355 msg               String           A string that will replace the existing message box body text (defaults
33356                                    to the XHTML-compliant non-breaking space character '&#160;')
33357 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33358 progress          Boolean          True to display a progress bar (defaults to false)
33359 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33360 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33361 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33362 title             String           The title text
33363 value             String           The string value to set into the active textbox element if displayed
33364 wait              Boolean          True to display a progress bar (defaults to false)
33365 width             Number           The width of the dialog in pixels
33366 </pre>
33367          *
33368          * Example usage:
33369          * <pre><code>
33370 Roo.Msg.show({
33371    title: 'Address',
33372    msg: 'Please enter your address:',
33373    width: 300,
33374    buttons: Roo.MessageBox.OKCANCEL,
33375    multiline: true,
33376    fn: saveAddress,
33377    animEl: 'addAddressBtn'
33378 });
33379 </code></pre>
33380          * @param {Object} config Configuration options
33381          * @return {Roo.MessageBox} This message box
33382          */
33383         show : function(options)
33384         {
33385             
33386             // this causes nightmares if you show one dialog after another
33387             // especially on callbacks..
33388              
33389             if(this.isVisible()){
33390                 
33391                 this.hide();
33392                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33393                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33394                 Roo.log("New Dialog Message:" +  options.msg )
33395                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33396                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33397                 
33398             }
33399             var d = this.getDialog();
33400             opt = options;
33401             d.setTitle(opt.title || "&#160;");
33402             d.close.setDisplayed(opt.closable !== false);
33403             activeTextEl = textboxEl;
33404             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33405             if(opt.prompt){
33406                 if(opt.multiline){
33407                     textboxEl.hide();
33408                     textareaEl.show();
33409                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33410                         opt.multiline : this.defaultTextHeight);
33411                     activeTextEl = textareaEl;
33412                 }else{
33413                     textboxEl.show();
33414                     textareaEl.hide();
33415                 }
33416             }else{
33417                 textboxEl.hide();
33418                 textareaEl.hide();
33419             }
33420             progressEl.setDisplayed(opt.progress === true);
33421             this.updateProgress(0);
33422             activeTextEl.dom.value = opt.value || "";
33423             if(opt.prompt){
33424                 dlg.setDefaultButton(activeTextEl);
33425             }else{
33426                 var bs = opt.buttons;
33427                 var db = null;
33428                 if(bs && bs.ok){
33429                     db = buttons["ok"];
33430                 }else if(bs && bs.yes){
33431                     db = buttons["yes"];
33432                 }
33433                 dlg.setDefaultButton(db);
33434             }
33435             bwidth = updateButtons(opt.buttons);
33436             this.updateText(opt.msg);
33437             if(opt.cls){
33438                 d.el.addClass(opt.cls);
33439             }
33440             d.proxyDrag = opt.proxyDrag === true;
33441             d.modal = opt.modal !== false;
33442             d.mask = opt.modal !== false ? mask : false;
33443             if(!d.isVisible()){
33444                 // force it to the end of the z-index stack so it gets a cursor in FF
33445                 document.body.appendChild(dlg.el.dom);
33446                 d.animateTarget = null;
33447                 d.show(options.animEl);
33448             }
33449             return this;
33450         },
33451
33452         /**
33453          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33454          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33455          * and closing the message box when the process is complete.
33456          * @param {String} title The title bar text
33457          * @param {String} msg The message box body text
33458          * @return {Roo.MessageBox} This message box
33459          */
33460         progress : function(title, msg){
33461             this.show({
33462                 title : title,
33463                 msg : msg,
33464                 buttons: false,
33465                 progress:true,
33466                 closable:false,
33467                 minWidth: this.minProgressWidth,
33468                 modal : true
33469             });
33470             return this;
33471         },
33472
33473         /**
33474          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33475          * If a callback function is passed it will be called after the user clicks the button, and the
33476          * id of the button that was clicked will be passed as the only parameter to the callback
33477          * (could also be the top-right close button).
33478          * @param {String} title The title bar text
33479          * @param {String} msg The message box body text
33480          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33481          * @param {Object} scope (optional) The scope of the callback function
33482          * @return {Roo.MessageBox} This message box
33483          */
33484         alert : function(title, msg, fn, scope){
33485             this.show({
33486                 title : title,
33487                 msg : msg,
33488                 buttons: this.OK,
33489                 fn: fn,
33490                 scope : scope,
33491                 modal : true
33492             });
33493             return this;
33494         },
33495
33496         /**
33497          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33498          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33499          * You are responsible for closing the message box when the process is complete.
33500          * @param {String} msg The message box body text
33501          * @param {String} title (optional) The title bar text
33502          * @return {Roo.MessageBox} This message box
33503          */
33504         wait : function(msg, title){
33505             this.show({
33506                 title : title,
33507                 msg : msg,
33508                 buttons: false,
33509                 closable:false,
33510                 progress:true,
33511                 modal:true,
33512                 width:300,
33513                 wait:true
33514             });
33515             waitTimer = Roo.TaskMgr.start({
33516                 run: function(i){
33517                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33518                 },
33519                 interval: 1000
33520             });
33521             return this;
33522         },
33523
33524         /**
33525          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33526          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33527          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33528          * @param {String} title The title bar text
33529          * @param {String} msg The message box body text
33530          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33531          * @param {Object} scope (optional) The scope of the callback function
33532          * @return {Roo.MessageBox} This message box
33533          */
33534         confirm : function(title, msg, fn, scope){
33535             this.show({
33536                 title : title,
33537                 msg : msg,
33538                 buttons: this.YESNO,
33539                 fn: fn,
33540                 scope : scope,
33541                 modal : true
33542             });
33543             return this;
33544         },
33545
33546         /**
33547          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33548          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33549          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33550          * (could also be the top-right close button) and the text that was entered will be passed as the two
33551          * parameters to the callback.
33552          * @param {String} title The title bar text
33553          * @param {String} msg The message box body text
33554          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33555          * @param {Object} scope (optional) The scope of the callback function
33556          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33557          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33558          * @return {Roo.MessageBox} This message box
33559          */
33560         prompt : function(title, msg, fn, scope, multiline){
33561             this.show({
33562                 title : title,
33563                 msg : msg,
33564                 buttons: this.OKCANCEL,
33565                 fn: fn,
33566                 minWidth:250,
33567                 scope : scope,
33568                 prompt:true,
33569                 multiline: multiline,
33570                 modal : true
33571             });
33572             return this;
33573         },
33574
33575         /**
33576          * Button config that displays a single OK button
33577          * @type Object
33578          */
33579         OK : {ok:true},
33580         /**
33581          * Button config that displays Yes and No buttons
33582          * @type Object
33583          */
33584         YESNO : {yes:true, no:true},
33585         /**
33586          * Button config that displays OK and Cancel buttons
33587          * @type Object
33588          */
33589         OKCANCEL : {ok:true, cancel:true},
33590         /**
33591          * Button config that displays Yes, No and Cancel buttons
33592          * @type Object
33593          */
33594         YESNOCANCEL : {yes:true, no:true, cancel:true},
33595
33596         /**
33597          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33598          * @type Number
33599          */
33600         defaultTextHeight : 75,
33601         /**
33602          * The maximum width in pixels of the message box (defaults to 600)
33603          * @type Number
33604          */
33605         maxWidth : 600,
33606         /**
33607          * The minimum width in pixels of the message box (defaults to 100)
33608          * @type Number
33609          */
33610         minWidth : 100,
33611         /**
33612          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33613          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33614          * @type Number
33615          */
33616         minProgressWidth : 250,
33617         /**
33618          * An object containing the default button text strings that can be overriden for localized language support.
33619          * Supported properties are: ok, cancel, yes and no.
33620          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33621          * @type Object
33622          */
33623         buttonText : {
33624             ok : "OK",
33625             cancel : "Cancel",
33626             yes : "Yes",
33627             no : "No"
33628         }
33629     };
33630 }();
33631
33632 /**
33633  * Shorthand for {@link Roo.MessageBox}
33634  */
33635 Roo.Msg = Roo.MessageBox;/*
33636  * Based on:
33637  * Ext JS Library 1.1.1
33638  * Copyright(c) 2006-2007, Ext JS, LLC.
33639  *
33640  * Originally Released Under LGPL - original licence link has changed is not relivant.
33641  *
33642  * Fork - LGPL
33643  * <script type="text/javascript">
33644  */
33645 /**
33646  * @class Roo.QuickTips
33647  * Provides attractive and customizable tooltips for any element.
33648  * @singleton
33649  */
33650 Roo.QuickTips = function(){
33651     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33652     var ce, bd, xy, dd;
33653     var visible = false, disabled = true, inited = false;
33654     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33655     
33656     var onOver = function(e){
33657         if(disabled){
33658             return;
33659         }
33660         var t = e.getTarget();
33661         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33662             return;
33663         }
33664         if(ce && t == ce.el){
33665             clearTimeout(hideProc);
33666             return;
33667         }
33668         if(t && tagEls[t.id]){
33669             tagEls[t.id].el = t;
33670             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33671             return;
33672         }
33673         var ttp, et = Roo.fly(t);
33674         var ns = cfg.namespace;
33675         if(tm.interceptTitles && t.title){
33676             ttp = t.title;
33677             t.qtip = ttp;
33678             t.removeAttribute("title");
33679             e.preventDefault();
33680         }else{
33681             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33682         }
33683         if(ttp){
33684             showProc = show.defer(tm.showDelay, tm, [{
33685                 el: t, 
33686                 text: ttp.replace(/\\n/g,'<br/>'),
33687                 width: et.getAttributeNS(ns, cfg.width),
33688                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33689                 title: et.getAttributeNS(ns, cfg.title),
33690                     cls: et.getAttributeNS(ns, cfg.cls)
33691             }]);
33692         }
33693     };
33694     
33695     var onOut = function(e){
33696         clearTimeout(showProc);
33697         var t = e.getTarget();
33698         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33699             hideProc = setTimeout(hide, tm.hideDelay);
33700         }
33701     };
33702     
33703     var onMove = function(e){
33704         if(disabled){
33705             return;
33706         }
33707         xy = e.getXY();
33708         xy[1] += 18;
33709         if(tm.trackMouse && ce){
33710             el.setXY(xy);
33711         }
33712     };
33713     
33714     var onDown = function(e){
33715         clearTimeout(showProc);
33716         clearTimeout(hideProc);
33717         if(!e.within(el)){
33718             if(tm.hideOnClick){
33719                 hide();
33720                 tm.disable();
33721                 tm.enable.defer(100, tm);
33722             }
33723         }
33724     };
33725     
33726     var getPad = function(){
33727         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33728     };
33729
33730     var show = function(o){
33731         if(disabled){
33732             return;
33733         }
33734         clearTimeout(dismissProc);
33735         ce = o;
33736         if(removeCls){ // in case manually hidden
33737             el.removeClass(removeCls);
33738             removeCls = null;
33739         }
33740         if(ce.cls){
33741             el.addClass(ce.cls);
33742             removeCls = ce.cls;
33743         }
33744         if(ce.title){
33745             tipTitle.update(ce.title);
33746             tipTitle.show();
33747         }else{
33748             tipTitle.update('');
33749             tipTitle.hide();
33750         }
33751         el.dom.style.width  = tm.maxWidth+'px';
33752         //tipBody.dom.style.width = '';
33753         tipBodyText.update(o.text);
33754         var p = getPad(), w = ce.width;
33755         if(!w){
33756             var td = tipBodyText.dom;
33757             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33758             if(aw > tm.maxWidth){
33759                 w = tm.maxWidth;
33760             }else if(aw < tm.minWidth){
33761                 w = tm.minWidth;
33762             }else{
33763                 w = aw;
33764             }
33765         }
33766         //tipBody.setWidth(w);
33767         el.setWidth(parseInt(w, 10) + p);
33768         if(ce.autoHide === false){
33769             close.setDisplayed(true);
33770             if(dd){
33771                 dd.unlock();
33772             }
33773         }else{
33774             close.setDisplayed(false);
33775             if(dd){
33776                 dd.lock();
33777             }
33778         }
33779         if(xy){
33780             el.avoidY = xy[1]-18;
33781             el.setXY(xy);
33782         }
33783         if(tm.animate){
33784             el.setOpacity(.1);
33785             el.setStyle("visibility", "visible");
33786             el.fadeIn({callback: afterShow});
33787         }else{
33788             afterShow();
33789         }
33790     };
33791     
33792     var afterShow = function(){
33793         if(ce){
33794             el.show();
33795             esc.enable();
33796             if(tm.autoDismiss && ce.autoHide !== false){
33797                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33798             }
33799         }
33800     };
33801     
33802     var hide = function(noanim){
33803         clearTimeout(dismissProc);
33804         clearTimeout(hideProc);
33805         ce = null;
33806         if(el.isVisible()){
33807             esc.disable();
33808             if(noanim !== true && tm.animate){
33809                 el.fadeOut({callback: afterHide});
33810             }else{
33811                 afterHide();
33812             } 
33813         }
33814     };
33815     
33816     var afterHide = function(){
33817         el.hide();
33818         if(removeCls){
33819             el.removeClass(removeCls);
33820             removeCls = null;
33821         }
33822     };
33823     
33824     return {
33825         /**
33826         * @cfg {Number} minWidth
33827         * The minimum width of the quick tip (defaults to 40)
33828         */
33829        minWidth : 40,
33830         /**
33831         * @cfg {Number} maxWidth
33832         * The maximum width of the quick tip (defaults to 300)
33833         */
33834        maxWidth : 300,
33835         /**
33836         * @cfg {Boolean} interceptTitles
33837         * True to automatically use the element's DOM title value if available (defaults to false)
33838         */
33839        interceptTitles : false,
33840         /**
33841         * @cfg {Boolean} trackMouse
33842         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33843         */
33844        trackMouse : false,
33845         /**
33846         * @cfg {Boolean} hideOnClick
33847         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33848         */
33849        hideOnClick : true,
33850         /**
33851         * @cfg {Number} showDelay
33852         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33853         */
33854        showDelay : 500,
33855         /**
33856         * @cfg {Number} hideDelay
33857         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33858         */
33859        hideDelay : 200,
33860         /**
33861         * @cfg {Boolean} autoHide
33862         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33863         * Used in conjunction with hideDelay.
33864         */
33865        autoHide : true,
33866         /**
33867         * @cfg {Boolean}
33868         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33869         * (defaults to true).  Used in conjunction with autoDismissDelay.
33870         */
33871        autoDismiss : true,
33872         /**
33873         * @cfg {Number}
33874         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33875         */
33876        autoDismissDelay : 5000,
33877        /**
33878         * @cfg {Boolean} animate
33879         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33880         */
33881        animate : false,
33882
33883        /**
33884         * @cfg {String} title
33885         * Title text to display (defaults to '').  This can be any valid HTML markup.
33886         */
33887         title: '',
33888        /**
33889         * @cfg {String} text
33890         * Body text to display (defaults to '').  This can be any valid HTML markup.
33891         */
33892         text : '',
33893        /**
33894         * @cfg {String} cls
33895         * A CSS class to apply to the base quick tip element (defaults to '').
33896         */
33897         cls : '',
33898        /**
33899         * @cfg {Number} width
33900         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33901         * minWidth or maxWidth.
33902         */
33903         width : null,
33904
33905     /**
33906      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33907      * or display QuickTips in a page.
33908      */
33909        init : function(){
33910           tm = Roo.QuickTips;
33911           cfg = tm.tagConfig;
33912           if(!inited){
33913               if(!Roo.isReady){ // allow calling of init() before onReady
33914                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33915                   return;
33916               }
33917               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33918               el.fxDefaults = {stopFx: true};
33919               // maximum custom styling
33920               //el.update('<div class="x-tip-top-left"><div class="x-tip-top-right"><div class="x-tip-top"></div></div></div><div class="x-tip-bd-left"><div class="x-tip-bd-right"><div class="x-tip-bd"><div class="x-tip-close"></div><h3></h3><div class="x-tip-bd-inner"></div><div class="x-clear"></div></div></div></div><div class="x-tip-ft-left"><div class="x-tip-ft-right"><div class="x-tip-ft"></div></div></div>');
33921               el.update('<div class="x-tip-bd"><div class="x-tip-close"></div><h3></h3><div class="x-tip-bd-inner"></div><div class="x-clear"></div></div>');              
33922               tipTitle = el.child('h3');
33923               tipTitle.enableDisplayMode("block");
33924               tipBody = el.child('div.x-tip-bd');
33925               tipBodyText = el.child('div.x-tip-bd-inner');
33926               //bdLeft = el.child('div.x-tip-bd-left');
33927               //bdRight = el.child('div.x-tip-bd-right');
33928               close = el.child('div.x-tip-close');
33929               close.enableDisplayMode("block");
33930               close.on("click", hide);
33931               var d = Roo.get(document);
33932               d.on("mousedown", onDown);
33933               d.on("mouseover", onOver);
33934               d.on("mouseout", onOut);
33935               d.on("mousemove", onMove);
33936               esc = d.addKeyListener(27, hide);
33937               esc.disable();
33938               if(Roo.dd.DD){
33939                   dd = el.initDD("default", null, {
33940                       onDrag : function(){
33941                           el.sync();  
33942                       }
33943                   });
33944                   dd.setHandleElId(tipTitle.id);
33945                   dd.lock();
33946               }
33947               inited = true;
33948           }
33949           this.enable(); 
33950        },
33951
33952     /**
33953      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33954      * are supported:
33955      * <pre>
33956 Property    Type                   Description
33957 ----------  ---------------------  ------------------------------------------------------------------------
33958 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
33959      * </ul>
33960      * @param {Object} config The config object
33961      */
33962        register : function(config){
33963            var cs = config instanceof Array ? config : arguments;
33964            for(var i = 0, len = cs.length; i < len; i++) {
33965                var c = cs[i];
33966                var target = c.target;
33967                if(target){
33968                    if(target instanceof Array){
33969                        for(var j = 0, jlen = target.length; j < jlen; j++){
33970                            tagEls[target[j]] = c;
33971                        }
33972                    }else{
33973                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
33974                    }
33975                }
33976            }
33977        },
33978
33979     /**
33980      * Removes this quick tip from its element and destroys it.
33981      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
33982      */
33983        unregister : function(el){
33984            delete tagEls[Roo.id(el)];
33985        },
33986
33987     /**
33988      * Enable this quick tip.
33989      */
33990        enable : function(){
33991            if(inited && disabled){
33992                locks.pop();
33993                if(locks.length < 1){
33994                    disabled = false;
33995                }
33996            }
33997        },
33998
33999     /**
34000      * Disable this quick tip.
34001      */
34002        disable : function(){
34003           disabled = true;
34004           clearTimeout(showProc);
34005           clearTimeout(hideProc);
34006           clearTimeout(dismissProc);
34007           if(ce){
34008               hide(true);
34009           }
34010           locks.push(1);
34011        },
34012
34013     /**
34014      * Returns true if the quick tip is enabled, else false.
34015      */
34016        isEnabled : function(){
34017             return !disabled;
34018        },
34019
34020         // private
34021        tagConfig : {
34022            namespace : "roo", // was ext?? this may break..
34023            alt_namespace : "ext",
34024            attribute : "qtip",
34025            width : "width",
34026            target : "target",
34027            title : "qtitle",
34028            hide : "hide",
34029            cls : "qclass"
34030        }
34031    };
34032 }();
34033
34034 // backwards compat
34035 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34036  * Based on:
34037  * Ext JS Library 1.1.1
34038  * Copyright(c) 2006-2007, Ext JS, LLC.
34039  *
34040  * Originally Released Under LGPL - original licence link has changed is not relivant.
34041  *
34042  * Fork - LGPL
34043  * <script type="text/javascript">
34044  */
34045  
34046
34047 /**
34048  * @class Roo.tree.TreePanel
34049  * @extends Roo.data.Tree
34050
34051  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34052  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34053  * @cfg {Boolean} enableDD true to enable drag and drop
34054  * @cfg {Boolean} enableDrag true to enable just drag
34055  * @cfg {Boolean} enableDrop true to enable just drop
34056  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34057  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34058  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34059  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34060  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34061  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34062  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34063  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34064  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34065  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34066  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34067  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34068  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34069  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34070  * @cfg {Function} renderer DEPRECATED - use TreeLoader:create event / Sets the rendering (formatting) function for the nodes. to return HTML markup for the tree view. The render function is called with  the following parameters:<ul><li>The {Object} The data for the node.</li></ul>
34071  * @cfg {Function} rendererTip DEPRECATED - use TreeLoader:create event / Sets the rendering (formatting) function for the nodes hovertip to return HTML markup for the tree view. The render function is called with  the following parameters:<ul><li>The {Object} The data for the node.</li></ul>
34072  * 
34073  * @constructor
34074  * @param {String/HTMLElement/Element} el The container element
34075  * @param {Object} config
34076  */
34077 Roo.tree.TreePanel = function(el, config){
34078     var root = false;
34079     var loader = false;
34080     if (config.root) {
34081         root = config.root;
34082         delete config.root;
34083     }
34084     if (config.loader) {
34085         loader = config.loader;
34086         delete config.loader;
34087     }
34088     
34089     Roo.apply(this, config);
34090     Roo.tree.TreePanel.superclass.constructor.call(this);
34091     this.el = Roo.get(el);
34092     this.el.addClass('x-tree');
34093     //console.log(root);
34094     if (root) {
34095         this.setRootNode( Roo.factory(root, Roo.tree));
34096     }
34097     if (loader) {
34098         this.loader = Roo.factory(loader, Roo.tree);
34099     }
34100    /**
34101     * Read-only. The id of the container element becomes this TreePanel's id.
34102     */
34103     this.id = this.el.id;
34104     this.addEvents({
34105         /**
34106         * @event beforeload
34107         * Fires before a node is loaded, return false to cancel
34108         * @param {Node} node The node being loaded
34109         */
34110         "beforeload" : true,
34111         /**
34112         * @event load
34113         * Fires when a node is loaded
34114         * @param {Node} node The node that was loaded
34115         */
34116         "load" : true,
34117         /**
34118         * @event textchange
34119         * Fires when the text for a node is changed
34120         * @param {Node} node The node
34121         * @param {String} text The new text
34122         * @param {String} oldText The old text
34123         */
34124         "textchange" : true,
34125         /**
34126         * @event beforeexpand
34127         * Fires before a node is expanded, return false to cancel.
34128         * @param {Node} node The node
34129         * @param {Boolean} deep
34130         * @param {Boolean} anim
34131         */
34132         "beforeexpand" : true,
34133         /**
34134         * @event beforecollapse
34135         * Fires before a node is collapsed, return false to cancel.
34136         * @param {Node} node The node
34137         * @param {Boolean} deep
34138         * @param {Boolean} anim
34139         */
34140         "beforecollapse" : true,
34141         /**
34142         * @event expand
34143         * Fires when a node is expanded
34144         * @param {Node} node The node
34145         */
34146         "expand" : true,
34147         /**
34148         * @event disabledchange
34149         * Fires when the disabled status of a node changes
34150         * @param {Node} node The node
34151         * @param {Boolean} disabled
34152         */
34153         "disabledchange" : true,
34154         /**
34155         * @event collapse
34156         * Fires when a node is collapsed
34157         * @param {Node} node The node
34158         */
34159         "collapse" : true,
34160         /**
34161         * @event beforeclick
34162         * Fires before click processing on a node. Return false to cancel the default action.
34163         * @param {Node} node The node
34164         * @param {Roo.EventObject} e The event object
34165         */
34166         "beforeclick":true,
34167         /**
34168         * @event checkchange
34169         * Fires when a node with a checkbox's checked property changes
34170         * @param {Node} this This node
34171         * @param {Boolean} checked
34172         */
34173         "checkchange":true,
34174         /**
34175         * @event click
34176         * Fires when a node is clicked
34177         * @param {Node} node The node
34178         * @param {Roo.EventObject} e The event object
34179         */
34180         "click":true,
34181         /**
34182         * @event dblclick
34183         * Fires when a node is double clicked
34184         * @param {Node} node The node
34185         * @param {Roo.EventObject} e The event object
34186         */
34187         "dblclick":true,
34188         /**
34189         * @event contextmenu
34190         * Fires when a node is right clicked
34191         * @param {Node} node The node
34192         * @param {Roo.EventObject} e The event object
34193         */
34194         "contextmenu":true,
34195         /**
34196         * @event beforechildrenrendered
34197         * Fires right before the child nodes for a node are rendered
34198         * @param {Node} node The node
34199         */
34200         "beforechildrenrendered":true,
34201         /**
34202         * @event startdrag
34203         * Fires when a node starts being dragged
34204         * @param {Roo.tree.TreePanel} this
34205         * @param {Roo.tree.TreeNode} node
34206         * @param {event} e The raw browser event
34207         */ 
34208        "startdrag" : true,
34209        /**
34210         * @event enddrag
34211         * Fires when a drag operation is complete
34212         * @param {Roo.tree.TreePanel} this
34213         * @param {Roo.tree.TreeNode} node
34214         * @param {event} e The raw browser event
34215         */
34216        "enddrag" : true,
34217        /**
34218         * @event dragdrop
34219         * Fires when a dragged node is dropped on a valid DD target
34220         * @param {Roo.tree.TreePanel} this
34221         * @param {Roo.tree.TreeNode} node
34222         * @param {DD} dd The dd it was dropped on
34223         * @param {event} e The raw browser event
34224         */
34225        "dragdrop" : true,
34226        /**
34227         * @event beforenodedrop
34228         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34229         * passed to handlers has the following properties:<br />
34230         * <ul style="padding:5px;padding-left:16px;">
34231         * <li>tree - The TreePanel</li>
34232         * <li>target - The node being targeted for the drop</li>
34233         * <li>data - The drag data from the drag source</li>
34234         * <li>point - The point of the drop - append, above or below</li>
34235         * <li>source - The drag source</li>
34236         * <li>rawEvent - Raw mouse event</li>
34237         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34238         * to be inserted by setting them on this object.</li>
34239         * <li>cancel - Set this to true to cancel the drop.</li>
34240         * </ul>
34241         * @param {Object} dropEvent
34242         */
34243        "beforenodedrop" : true,
34244        /**
34245         * @event nodedrop
34246         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34247         * passed to handlers has the following properties:<br />
34248         * <ul style="padding:5px;padding-left:16px;">
34249         * <li>tree - The TreePanel</li>
34250         * <li>target - The node being targeted for the drop</li>
34251         * <li>data - The drag data from the drag source</li>
34252         * <li>point - The point of the drop - append, above or below</li>
34253         * <li>source - The drag source</li>
34254         * <li>rawEvent - Raw mouse event</li>
34255         * <li>dropNode - Dropped node(s).</li>
34256         * </ul>
34257         * @param {Object} dropEvent
34258         */
34259        "nodedrop" : true,
34260         /**
34261         * @event nodedragover
34262         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34263         * passed to handlers has the following properties:<br />
34264         * <ul style="padding:5px;padding-left:16px;">
34265         * <li>tree - The TreePanel</li>
34266         * <li>target - The node being targeted for the drop</li>
34267         * <li>data - The drag data from the drag source</li>
34268         * <li>point - The point of the drop - append, above or below</li>
34269         * <li>source - The drag source</li>
34270         * <li>rawEvent - Raw mouse event</li>
34271         * <li>dropNode - Drop node(s) provided by the source.</li>
34272         * <li>cancel - Set this to true to signal drop not allowed.</li>
34273         * </ul>
34274         * @param {Object} dragOverEvent
34275         */
34276        "nodedragover" : true
34277         
34278     });
34279     if(this.singleExpand){
34280        this.on("beforeexpand", this.restrictExpand, this);
34281     }
34282     if (this.editor) {
34283         this.editor.tree = this;
34284         this.editor = Roo.factory(this.editor, Roo.tree);
34285     }
34286     
34287     if (this.selModel) {
34288         this.selModel = Roo.factory(this.selModel, Roo.tree);
34289     }
34290    
34291 };
34292 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34293     rootVisible : true,
34294     animate: Roo.enableFx,
34295     lines : true,
34296     enableDD : false,
34297     hlDrop : Roo.enableFx,
34298   
34299     renderer: false,
34300     
34301     rendererTip: false,
34302     // private
34303     restrictExpand : function(node){
34304         var p = node.parentNode;
34305         if(p){
34306             if(p.expandedChild && p.expandedChild.parentNode == p){
34307                 p.expandedChild.collapse();
34308             }
34309             p.expandedChild = node;
34310         }
34311     },
34312
34313     // private override
34314     setRootNode : function(node){
34315         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34316         if(!this.rootVisible){
34317             node.ui = new Roo.tree.RootTreeNodeUI(node);
34318         }
34319         return node;
34320     },
34321
34322     /**
34323      * Returns the container element for this TreePanel
34324      */
34325     getEl : function(){
34326         return this.el;
34327     },
34328
34329     /**
34330      * Returns the default TreeLoader for this TreePanel
34331      */
34332     getLoader : function(){
34333         return this.loader;
34334     },
34335
34336     /**
34337      * Expand all nodes
34338      */
34339     expandAll : function(){
34340         this.root.expand(true);
34341     },
34342
34343     /**
34344      * Collapse all nodes
34345      */
34346     collapseAll : function(){
34347         this.root.collapse(true);
34348     },
34349
34350     /**
34351      * Returns the selection model used by this TreePanel
34352      */
34353     getSelectionModel : function(){
34354         if(!this.selModel){
34355             this.selModel = new Roo.tree.DefaultSelectionModel();
34356         }
34357         return this.selModel;
34358     },
34359
34360     /**
34361      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34362      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34363      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34364      * @return {Array}
34365      */
34366     getChecked : function(a, startNode){
34367         startNode = startNode || this.root;
34368         var r = [];
34369         var f = function(){
34370             if(this.attributes.checked){
34371                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34372             }
34373         }
34374         startNode.cascade(f);
34375         return r;
34376     },
34377
34378     /**
34379      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34380      * @param {String} path
34381      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34382      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34383      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34384      */
34385     expandPath : function(path, attr, callback){
34386         attr = attr || "id";
34387         var keys = path.split(this.pathSeparator);
34388         var curNode = this.root;
34389         if(curNode.attributes[attr] != keys[1]){ // invalid root
34390             if(callback){
34391                 callback(false, null);
34392             }
34393             return;
34394         }
34395         var index = 1;
34396         var f = function(){
34397             if(++index == keys.length){
34398                 if(callback){
34399                     callback(true, curNode);
34400                 }
34401                 return;
34402             }
34403             var c = curNode.findChild(attr, keys[index]);
34404             if(!c){
34405                 if(callback){
34406                     callback(false, curNode);
34407                 }
34408                 return;
34409             }
34410             curNode = c;
34411             c.expand(false, false, f);
34412         };
34413         curNode.expand(false, false, f);
34414     },
34415
34416     /**
34417      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34418      * @param {String} path
34419      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34420      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34421      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34422      */
34423     selectPath : function(path, attr, callback){
34424         attr = attr || "id";
34425         var keys = path.split(this.pathSeparator);
34426         var v = keys.pop();
34427         if(keys.length > 0){
34428             var f = function(success, node){
34429                 if(success && node){
34430                     var n = node.findChild(attr, v);
34431                     if(n){
34432                         n.select();
34433                         if(callback){
34434                             callback(true, n);
34435                         }
34436                     }else if(callback){
34437                         callback(false, n);
34438                     }
34439                 }else{
34440                     if(callback){
34441                         callback(false, n);
34442                     }
34443                 }
34444             };
34445             this.expandPath(keys.join(this.pathSeparator), attr, f);
34446         }else{
34447             this.root.select();
34448             if(callback){
34449                 callback(true, this.root);
34450             }
34451         }
34452     },
34453
34454     getTreeEl : function(){
34455         return this.el;
34456     },
34457
34458     /**
34459      * Trigger rendering of this TreePanel
34460      */
34461     render : function(){
34462         if (this.innerCt) {
34463             return this; // stop it rendering more than once!!
34464         }
34465         
34466         this.innerCt = this.el.createChild({tag:"ul",
34467                cls:"x-tree-root-ct " +
34468                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34469
34470         if(this.containerScroll){
34471             Roo.dd.ScrollManager.register(this.el);
34472         }
34473         if((this.enableDD || this.enableDrop) && !this.dropZone){
34474            /**
34475             * The dropZone used by this tree if drop is enabled
34476             * @type Roo.tree.TreeDropZone
34477             */
34478              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34479                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34480            });
34481         }
34482         if((this.enableDD || this.enableDrag) && !this.dragZone){
34483            /**
34484             * The dragZone used by this tree if drag is enabled
34485             * @type Roo.tree.TreeDragZone
34486             */
34487             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34488                ddGroup: this.ddGroup || "TreeDD",
34489                scroll: this.ddScroll
34490            });
34491         }
34492         this.getSelectionModel().init(this);
34493         if (!this.root) {
34494             Roo.log("ROOT not set in tree");
34495             return this;
34496         }
34497         this.root.render();
34498         if(!this.rootVisible){
34499             this.root.renderChildren();
34500         }
34501         return this;
34502     }
34503 });/*
34504  * Based on:
34505  * Ext JS Library 1.1.1
34506  * Copyright(c) 2006-2007, Ext JS, LLC.
34507  *
34508  * Originally Released Under LGPL - original licence link has changed is not relivant.
34509  *
34510  * Fork - LGPL
34511  * <script type="text/javascript">
34512  */
34513  
34514
34515 /**
34516  * @class Roo.tree.DefaultSelectionModel
34517  * @extends Roo.util.Observable
34518  * The default single selection for a TreePanel.
34519  * @param {Object} cfg Configuration
34520  */
34521 Roo.tree.DefaultSelectionModel = function(cfg){
34522    this.selNode = null;
34523    
34524    
34525    
34526    this.addEvents({
34527        /**
34528         * @event selectionchange
34529         * Fires when the selected node changes
34530         * @param {DefaultSelectionModel} this
34531         * @param {TreeNode} node the new selection
34532         */
34533        "selectionchange" : true,
34534
34535        /**
34536         * @event beforeselect
34537         * Fires before the selected node changes, return false to cancel the change
34538         * @param {DefaultSelectionModel} this
34539         * @param {TreeNode} node the new selection
34540         * @param {TreeNode} node the old selection
34541         */
34542        "beforeselect" : true
34543    });
34544    
34545     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34546 };
34547
34548 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34549     init : function(tree){
34550         this.tree = tree;
34551         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34552         tree.on("click", this.onNodeClick, this);
34553     },
34554     
34555     onNodeClick : function(node, e){
34556         if (e.ctrlKey && this.selNode == node)  {
34557             this.unselect(node);
34558             return;
34559         }
34560         this.select(node);
34561     },
34562     
34563     /**
34564      * Select a node.
34565      * @param {TreeNode} node The node to select
34566      * @return {TreeNode} The selected node
34567      */
34568     select : function(node){
34569         var last = this.selNode;
34570         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34571             if(last){
34572                 last.ui.onSelectedChange(false);
34573             }
34574             this.selNode = node;
34575             node.ui.onSelectedChange(true);
34576             this.fireEvent("selectionchange", this, node, last);
34577         }
34578         return node;
34579     },
34580     
34581     /**
34582      * Deselect a node.
34583      * @param {TreeNode} node The node to unselect
34584      */
34585     unselect : function(node){
34586         if(this.selNode == node){
34587             this.clearSelections();
34588         }    
34589     },
34590     
34591     /**
34592      * Clear all selections
34593      */
34594     clearSelections : function(){
34595         var n = this.selNode;
34596         if(n){
34597             n.ui.onSelectedChange(false);
34598             this.selNode = null;
34599             this.fireEvent("selectionchange", this, null);
34600         }
34601         return n;
34602     },
34603     
34604     /**
34605      * Get the selected node
34606      * @return {TreeNode} The selected node
34607      */
34608     getSelectedNode : function(){
34609         return this.selNode;    
34610     },
34611     
34612     /**
34613      * Returns true if the node is selected
34614      * @param {TreeNode} node The node to check
34615      * @return {Boolean}
34616      */
34617     isSelected : function(node){
34618         return this.selNode == node;  
34619     },
34620
34621     /**
34622      * Selects the node above the selected node in the tree, intelligently walking the nodes
34623      * @return TreeNode The new selection
34624      */
34625     selectPrevious : function(){
34626         var s = this.selNode || this.lastSelNode;
34627         if(!s){
34628             return null;
34629         }
34630         var ps = s.previousSibling;
34631         if(ps){
34632             if(!ps.isExpanded() || ps.childNodes.length < 1){
34633                 return this.select(ps);
34634             } else{
34635                 var lc = ps.lastChild;
34636                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34637                     lc = lc.lastChild;
34638                 }
34639                 return this.select(lc);
34640             }
34641         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34642             return this.select(s.parentNode);
34643         }
34644         return null;
34645     },
34646
34647     /**
34648      * Selects the node above the selected node in the tree, intelligently walking the nodes
34649      * @return TreeNode The new selection
34650      */
34651     selectNext : function(){
34652         var s = this.selNode || this.lastSelNode;
34653         if(!s){
34654             return null;
34655         }
34656         if(s.firstChild && s.isExpanded()){
34657              return this.select(s.firstChild);
34658          }else if(s.nextSibling){
34659              return this.select(s.nextSibling);
34660          }else if(s.parentNode){
34661             var newS = null;
34662             s.parentNode.bubble(function(){
34663                 if(this.nextSibling){
34664                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34665                     return false;
34666                 }
34667             });
34668             return newS;
34669          }
34670         return null;
34671     },
34672
34673     onKeyDown : function(e){
34674         var s = this.selNode || this.lastSelNode;
34675         // undesirable, but required
34676         var sm = this;
34677         if(!s){
34678             return;
34679         }
34680         var k = e.getKey();
34681         switch(k){
34682              case e.DOWN:
34683                  e.stopEvent();
34684                  this.selectNext();
34685              break;
34686              case e.UP:
34687                  e.stopEvent();
34688                  this.selectPrevious();
34689              break;
34690              case e.RIGHT:
34691                  e.preventDefault();
34692                  if(s.hasChildNodes()){
34693                      if(!s.isExpanded()){
34694                          s.expand();
34695                      }else if(s.firstChild){
34696                          this.select(s.firstChild, e);
34697                      }
34698                  }
34699              break;
34700              case e.LEFT:
34701                  e.preventDefault();
34702                  if(s.hasChildNodes() && s.isExpanded()){
34703                      s.collapse();
34704                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34705                      this.select(s.parentNode, e);
34706                  }
34707              break;
34708         };
34709     }
34710 });
34711
34712 /**
34713  * @class Roo.tree.MultiSelectionModel
34714  * @extends Roo.util.Observable
34715  * Multi selection for a TreePanel.
34716  * @param {Object} cfg Configuration
34717  */
34718 Roo.tree.MultiSelectionModel = function(){
34719    this.selNodes = [];
34720    this.selMap = {};
34721    this.addEvents({
34722        /**
34723         * @event selectionchange
34724         * Fires when the selected nodes change
34725         * @param {MultiSelectionModel} this
34726         * @param {Array} nodes Array of the selected nodes
34727         */
34728        "selectionchange" : true
34729    });
34730    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34731    
34732 };
34733
34734 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34735     init : function(tree){
34736         this.tree = tree;
34737         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34738         tree.on("click", this.onNodeClick, this);
34739     },
34740     
34741     onNodeClick : function(node, e){
34742         this.select(node, e, e.ctrlKey);
34743     },
34744     
34745     /**
34746      * Select a node.
34747      * @param {TreeNode} node The node to select
34748      * @param {EventObject} e (optional) An event associated with the selection
34749      * @param {Boolean} keepExisting True to retain existing selections
34750      * @return {TreeNode} The selected node
34751      */
34752     select : function(node, e, keepExisting){
34753         if(keepExisting !== true){
34754             this.clearSelections(true);
34755         }
34756         if(this.isSelected(node)){
34757             this.lastSelNode = node;
34758             return node;
34759         }
34760         this.selNodes.push(node);
34761         this.selMap[node.id] = node;
34762         this.lastSelNode = node;
34763         node.ui.onSelectedChange(true);
34764         this.fireEvent("selectionchange", this, this.selNodes);
34765         return node;
34766     },
34767     
34768     /**
34769      * Deselect a node.
34770      * @param {TreeNode} node The node to unselect
34771      */
34772     unselect : function(node){
34773         if(this.selMap[node.id]){
34774             node.ui.onSelectedChange(false);
34775             var sn = this.selNodes;
34776             var index = -1;
34777             if(sn.indexOf){
34778                 index = sn.indexOf(node);
34779             }else{
34780                 for(var i = 0, len = sn.length; i < len; i++){
34781                     if(sn[i] == node){
34782                         index = i;
34783                         break;
34784                     }
34785                 }
34786             }
34787             if(index != -1){
34788                 this.selNodes.splice(index, 1);
34789             }
34790             delete this.selMap[node.id];
34791             this.fireEvent("selectionchange", this, this.selNodes);
34792         }
34793     },
34794     
34795     /**
34796      * Clear all selections
34797      */
34798     clearSelections : function(suppressEvent){
34799         var sn = this.selNodes;
34800         if(sn.length > 0){
34801             for(var i = 0, len = sn.length; i < len; i++){
34802                 sn[i].ui.onSelectedChange(false);
34803             }
34804             this.selNodes = [];
34805             this.selMap = {};
34806             if(suppressEvent !== true){
34807                 this.fireEvent("selectionchange", this, this.selNodes);
34808             }
34809         }
34810     },
34811     
34812     /**
34813      * Returns true if the node is selected
34814      * @param {TreeNode} node The node to check
34815      * @return {Boolean}
34816      */
34817     isSelected : function(node){
34818         return this.selMap[node.id] ? true : false;  
34819     },
34820     
34821     /**
34822      * Returns an array of the selected nodes
34823      * @return {Array}
34824      */
34825     getSelectedNodes : function(){
34826         return this.selNodes;    
34827     },
34828
34829     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34830
34831     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34832
34833     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34834 });/*
34835  * Based on:
34836  * Ext JS Library 1.1.1
34837  * Copyright(c) 2006-2007, Ext JS, LLC.
34838  *
34839  * Originally Released Under LGPL - original licence link has changed is not relivant.
34840  *
34841  * Fork - LGPL
34842  * <script type="text/javascript">
34843  */
34844  
34845 /**
34846  * @class Roo.tree.TreeNode
34847  * @extends Roo.data.Node
34848  * @cfg {String} text The text for this node
34849  * @cfg {Boolean} expanded true to start the node expanded
34850  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34851  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34852  * @cfg {Boolean} disabled true to start the node disabled
34853  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34854  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34855  * @cfg {String} cls A css class to be added to the node
34856  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34857  * @cfg {String} href URL of the link used for the node (defaults to #)
34858  * @cfg {String} hrefTarget target frame for the link
34859  * @cfg {String} qtip An Ext QuickTip for the node
34860  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34861  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34862  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34863  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34864  * (defaults to undefined with no checkbox rendered)
34865  * @constructor
34866  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34867  */
34868 Roo.tree.TreeNode = function(attributes){
34869     attributes = attributes || {};
34870     if(typeof attributes == "string"){
34871         attributes = {text: attributes};
34872     }
34873     this.childrenRendered = false;
34874     this.rendered = false;
34875     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34876     this.expanded = attributes.expanded === true;
34877     this.isTarget = attributes.isTarget !== false;
34878     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34879     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34880
34881     /**
34882      * Read-only. The text for this node. To change it use setText().
34883      * @type String
34884      */
34885     this.text = attributes.text;
34886     /**
34887      * True if this node is disabled.
34888      * @type Boolean
34889      */
34890     this.disabled = attributes.disabled === true;
34891
34892     this.addEvents({
34893         /**
34894         * @event textchange
34895         * Fires when the text for this node is changed
34896         * @param {Node} this This node
34897         * @param {String} text The new text
34898         * @param {String} oldText The old text
34899         */
34900         "textchange" : true,
34901         /**
34902         * @event beforeexpand
34903         * Fires before this node is expanded, return false to cancel.
34904         * @param {Node} this This node
34905         * @param {Boolean} deep
34906         * @param {Boolean} anim
34907         */
34908         "beforeexpand" : true,
34909         /**
34910         * @event beforecollapse
34911         * Fires before this node is collapsed, return false to cancel.
34912         * @param {Node} this This node
34913         * @param {Boolean} deep
34914         * @param {Boolean} anim
34915         */
34916         "beforecollapse" : true,
34917         /**
34918         * @event expand
34919         * Fires when this node is expanded
34920         * @param {Node} this This node
34921         */
34922         "expand" : true,
34923         /**
34924         * @event disabledchange
34925         * Fires when the disabled status of this node changes
34926         * @param {Node} this This node
34927         * @param {Boolean} disabled
34928         */
34929         "disabledchange" : true,
34930         /**
34931         * @event collapse
34932         * Fires when this node is collapsed
34933         * @param {Node} this This node
34934         */
34935         "collapse" : true,
34936         /**
34937         * @event beforeclick
34938         * Fires before click processing. Return false to cancel the default action.
34939         * @param {Node} this This node
34940         * @param {Roo.EventObject} e The event object
34941         */
34942         "beforeclick":true,
34943         /**
34944         * @event checkchange
34945         * Fires when a node with a checkbox's checked property changes
34946         * @param {Node} this This node
34947         * @param {Boolean} checked
34948         */
34949         "checkchange":true,
34950         /**
34951         * @event click
34952         * Fires when this node is clicked
34953         * @param {Node} this This node
34954         * @param {Roo.EventObject} e The event object
34955         */
34956         "click":true,
34957         /**
34958         * @event dblclick
34959         * Fires when this node is double clicked
34960         * @param {Node} this This node
34961         * @param {Roo.EventObject} e The event object
34962         */
34963         "dblclick":true,
34964         /**
34965         * @event contextmenu
34966         * Fires when this node is right clicked
34967         * @param {Node} this This node
34968         * @param {Roo.EventObject} e The event object
34969         */
34970         "contextmenu":true,
34971         /**
34972         * @event beforechildrenrendered
34973         * Fires right before the child nodes for this node are rendered
34974         * @param {Node} this This node
34975         */
34976         "beforechildrenrendered":true
34977     });
34978
34979     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
34980
34981     /**
34982      * Read-only. The UI for this node
34983      * @type TreeNodeUI
34984      */
34985     this.ui = new uiClass(this);
34986     
34987     // finally support items[]
34988     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
34989         return;
34990     }
34991     
34992     
34993     Roo.each(this.attributes.items, function(c) {
34994         this.appendChild(Roo.factory(c,Roo.Tree));
34995     }, this);
34996     delete this.attributes.items;
34997     
34998     
34999     
35000 };
35001 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35002     preventHScroll: true,
35003     /**
35004      * Returns true if this node is expanded
35005      * @return {Boolean}
35006      */
35007     isExpanded : function(){
35008         return this.expanded;
35009     },
35010
35011     /**
35012      * Returns the UI object for this node
35013      * @return {TreeNodeUI}
35014      */
35015     getUI : function(){
35016         return this.ui;
35017     },
35018
35019     // private override
35020     setFirstChild : function(node){
35021         var of = this.firstChild;
35022         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35023         if(this.childrenRendered && of && node != of){
35024             of.renderIndent(true, true);
35025         }
35026         if(this.rendered){
35027             this.renderIndent(true, true);
35028         }
35029     },
35030
35031     // private override
35032     setLastChild : function(node){
35033         var ol = this.lastChild;
35034         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35035         if(this.childrenRendered && ol && node != ol){
35036             ol.renderIndent(true, true);
35037         }
35038         if(this.rendered){
35039             this.renderIndent(true, true);
35040         }
35041     },
35042
35043     // these methods are overridden to provide lazy rendering support
35044     // private override
35045     appendChild : function()
35046     {
35047         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35048         if(node && this.childrenRendered){
35049             node.render();
35050         }
35051         this.ui.updateExpandIcon();
35052         return node;
35053     },
35054
35055     // private override
35056     removeChild : function(node){
35057         this.ownerTree.getSelectionModel().unselect(node);
35058         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35059         // if it's been rendered remove dom node
35060         if(this.childrenRendered){
35061             node.ui.remove();
35062         }
35063         if(this.childNodes.length < 1){
35064             this.collapse(false, false);
35065         }else{
35066             this.ui.updateExpandIcon();
35067         }
35068         if(!this.firstChild) {
35069             this.childrenRendered = false;
35070         }
35071         return node;
35072     },
35073
35074     // private override
35075     insertBefore : function(node, refNode){
35076         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35077         if(newNode && refNode && this.childrenRendered){
35078             node.render();
35079         }
35080         this.ui.updateExpandIcon();
35081         return newNode;
35082     },
35083
35084     /**
35085      * Sets the text for this node
35086      * @param {String} text
35087      */
35088     setText : function(text){
35089         var oldText = this.text;
35090         this.text = text;
35091         this.attributes.text = text;
35092         if(this.rendered){ // event without subscribing
35093             this.ui.onTextChange(this, text, oldText);
35094         }
35095         this.fireEvent("textchange", this, text, oldText);
35096     },
35097
35098     /**
35099      * Triggers selection of this node
35100      */
35101     select : function(){
35102         this.getOwnerTree().getSelectionModel().select(this);
35103     },
35104
35105     /**
35106      * Triggers deselection of this node
35107      */
35108     unselect : function(){
35109         this.getOwnerTree().getSelectionModel().unselect(this);
35110     },
35111
35112     /**
35113      * Returns true if this node is selected
35114      * @return {Boolean}
35115      */
35116     isSelected : function(){
35117         return this.getOwnerTree().getSelectionModel().isSelected(this);
35118     },
35119
35120     /**
35121      * Expand this node.
35122      * @param {Boolean} deep (optional) True to expand all children as well
35123      * @param {Boolean} anim (optional) false to cancel the default animation
35124      * @param {Function} callback (optional) A callback to be called when
35125      * expanding this node completes (does not wait for deep expand to complete).
35126      * Called with 1 parameter, this node.
35127      */
35128     expand : function(deep, anim, callback){
35129         if(!this.expanded){
35130             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35131                 return;
35132             }
35133             if(!this.childrenRendered){
35134                 this.renderChildren();
35135             }
35136             this.expanded = true;
35137             if(!this.isHiddenRoot() && (this.getOwnerTree().animate && anim !== false) || anim){
35138                 this.ui.animExpand(function(){
35139                     this.fireEvent("expand", this);
35140                     if(typeof callback == "function"){
35141                         callback(this);
35142                     }
35143                     if(deep === true){
35144                         this.expandChildNodes(true);
35145                     }
35146                 }.createDelegate(this));
35147                 return;
35148             }else{
35149                 this.ui.expand();
35150                 this.fireEvent("expand", this);
35151                 if(typeof callback == "function"){
35152                     callback(this);
35153                 }
35154             }
35155         }else{
35156            if(typeof callback == "function"){
35157                callback(this);
35158            }
35159         }
35160         if(deep === true){
35161             this.expandChildNodes(true);
35162         }
35163     },
35164
35165     isHiddenRoot : function(){
35166         return this.isRoot && !this.getOwnerTree().rootVisible;
35167     },
35168
35169     /**
35170      * Collapse this node.
35171      * @param {Boolean} deep (optional) True to collapse all children as well
35172      * @param {Boolean} anim (optional) false to cancel the default animation
35173      */
35174     collapse : function(deep, anim){
35175         if(this.expanded && !this.isHiddenRoot()){
35176             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35177                 return;
35178             }
35179             this.expanded = false;
35180             if((this.getOwnerTree().animate && anim !== false) || anim){
35181                 this.ui.animCollapse(function(){
35182                     this.fireEvent("collapse", this);
35183                     if(deep === true){
35184                         this.collapseChildNodes(true);
35185                     }
35186                 }.createDelegate(this));
35187                 return;
35188             }else{
35189                 this.ui.collapse();
35190                 this.fireEvent("collapse", this);
35191             }
35192         }
35193         if(deep === true){
35194             var cs = this.childNodes;
35195             for(var i = 0, len = cs.length; i < len; i++) {
35196                 cs[i].collapse(true, false);
35197             }
35198         }
35199     },
35200
35201     // private
35202     delayedExpand : function(delay){
35203         if(!this.expandProcId){
35204             this.expandProcId = this.expand.defer(delay, this);
35205         }
35206     },
35207
35208     // private
35209     cancelExpand : function(){
35210         if(this.expandProcId){
35211             clearTimeout(this.expandProcId);
35212         }
35213         this.expandProcId = false;
35214     },
35215
35216     /**
35217      * Toggles expanded/collapsed state of the node
35218      */
35219     toggle : function(){
35220         if(this.expanded){
35221             this.collapse();
35222         }else{
35223             this.expand();
35224         }
35225     },
35226
35227     /**
35228      * Ensures all parent nodes are expanded
35229      */
35230     ensureVisible : function(callback){
35231         var tree = this.getOwnerTree();
35232         tree.expandPath(this.parentNode.getPath(), false, function(){
35233             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35234             Roo.callback(callback);
35235         }.createDelegate(this));
35236     },
35237
35238     /**
35239      * Expand all child nodes
35240      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35241      */
35242     expandChildNodes : function(deep){
35243         var cs = this.childNodes;
35244         for(var i = 0, len = cs.length; i < len; i++) {
35245                 cs[i].expand(deep);
35246         }
35247     },
35248
35249     /**
35250      * Collapse all child nodes
35251      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35252      */
35253     collapseChildNodes : function(deep){
35254         var cs = this.childNodes;
35255         for(var i = 0, len = cs.length; i < len; i++) {
35256                 cs[i].collapse(deep);
35257         }
35258     },
35259
35260     /**
35261      * Disables this node
35262      */
35263     disable : function(){
35264         this.disabled = true;
35265         this.unselect();
35266         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35267             this.ui.onDisableChange(this, true);
35268         }
35269         this.fireEvent("disabledchange", this, true);
35270     },
35271
35272     /**
35273      * Enables this node
35274      */
35275     enable : function(){
35276         this.disabled = false;
35277         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35278             this.ui.onDisableChange(this, false);
35279         }
35280         this.fireEvent("disabledchange", this, false);
35281     },
35282
35283     // private
35284     renderChildren : function(suppressEvent){
35285         if(suppressEvent !== false){
35286             this.fireEvent("beforechildrenrendered", this);
35287         }
35288         var cs = this.childNodes;
35289         for(var i = 0, len = cs.length; i < len; i++){
35290             cs[i].render(true);
35291         }
35292         this.childrenRendered = true;
35293     },
35294
35295     // private
35296     sort : function(fn, scope){
35297         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35298         if(this.childrenRendered){
35299             var cs = this.childNodes;
35300             for(var i = 0, len = cs.length; i < len; i++){
35301                 cs[i].render(true);
35302             }
35303         }
35304     },
35305
35306     // private
35307     render : function(bulkRender){
35308         this.ui.render(bulkRender);
35309         if(!this.rendered){
35310             this.rendered = true;
35311             if(this.expanded){
35312                 this.expanded = false;
35313                 this.expand(false, false);
35314             }
35315         }
35316     },
35317
35318     // private
35319     renderIndent : function(deep, refresh){
35320         if(refresh){
35321             this.ui.childIndent = null;
35322         }
35323         this.ui.renderIndent();
35324         if(deep === true && this.childrenRendered){
35325             var cs = this.childNodes;
35326             for(var i = 0, len = cs.length; i < len; i++){
35327                 cs[i].renderIndent(true, refresh);
35328             }
35329         }
35330     }
35331 });/*
35332  * Based on:
35333  * Ext JS Library 1.1.1
35334  * Copyright(c) 2006-2007, Ext JS, LLC.
35335  *
35336  * Originally Released Under LGPL - original licence link has changed is not relivant.
35337  *
35338  * Fork - LGPL
35339  * <script type="text/javascript">
35340  */
35341  
35342 /**
35343  * @class Roo.tree.AsyncTreeNode
35344  * @extends Roo.tree.TreeNode
35345  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35346  * @constructor
35347  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35348  */
35349  Roo.tree.AsyncTreeNode = function(config){
35350     this.loaded = false;
35351     this.loading = false;
35352     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35353     /**
35354     * @event beforeload
35355     * Fires before this node is loaded, return false to cancel
35356     * @param {Node} this This node
35357     */
35358     this.addEvents({'beforeload':true, 'load': true});
35359     /**
35360     * @event load
35361     * Fires when this node is loaded
35362     * @param {Node} this This node
35363     */
35364     /**
35365      * The loader used by this node (defaults to using the tree's defined loader)
35366      * @type TreeLoader
35367      * @property loader
35368      */
35369 };
35370 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35371     expand : function(deep, anim, callback){
35372         if(this.loading){ // if an async load is already running, waiting til it's done
35373             var timer;
35374             var f = function(){
35375                 if(!this.loading){ // done loading
35376                     clearInterval(timer);
35377                     this.expand(deep, anim, callback);
35378                 }
35379             }.createDelegate(this);
35380             timer = setInterval(f, 200);
35381             return;
35382         }
35383         if(!this.loaded){
35384             if(this.fireEvent("beforeload", this) === false){
35385                 return;
35386             }
35387             this.loading = true;
35388             this.ui.beforeLoad(this);
35389             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35390             if(loader){
35391                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35392                 return;
35393             }
35394         }
35395         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35396     },
35397     
35398     /**
35399      * Returns true if this node is currently loading
35400      * @return {Boolean}
35401      */
35402     isLoading : function(){
35403         return this.loading;  
35404     },
35405     
35406     loadComplete : function(deep, anim, callback){
35407         this.loading = false;
35408         this.loaded = true;
35409         this.ui.afterLoad(this);
35410         this.fireEvent("load", this);
35411         this.expand(deep, anim, callback);
35412     },
35413     
35414     /**
35415      * Returns true if this node has been loaded
35416      * @return {Boolean}
35417      */
35418     isLoaded : function(){
35419         return this.loaded;
35420     },
35421     
35422     hasChildNodes : function(){
35423         if(!this.isLeaf() && !this.loaded){
35424             return true;
35425         }else{
35426             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35427         }
35428     },
35429
35430     /**
35431      * Trigger a reload for this node
35432      * @param {Function} callback
35433      */
35434     reload : function(callback){
35435         this.collapse(false, false);
35436         while(this.firstChild){
35437             this.removeChild(this.firstChild);
35438         }
35439         this.childrenRendered = false;
35440         this.loaded = false;
35441         if(this.isHiddenRoot()){
35442             this.expanded = false;
35443         }
35444         this.expand(false, false, callback);
35445     }
35446 });/*
35447  * Based on:
35448  * Ext JS Library 1.1.1
35449  * Copyright(c) 2006-2007, Ext JS, LLC.
35450  *
35451  * Originally Released Under LGPL - original licence link has changed is not relivant.
35452  *
35453  * Fork - LGPL
35454  * <script type="text/javascript">
35455  */
35456  
35457 /**
35458  * @class Roo.tree.TreeNodeUI
35459  * @constructor
35460  * @param {Object} node The node to render
35461  * The TreeNode UI implementation is separate from the
35462  * tree implementation. Unless you are customizing the tree UI,
35463  * you should never have to use this directly.
35464  */
35465 Roo.tree.TreeNodeUI = function(node){
35466     this.node = node;
35467     this.rendered = false;
35468     this.animating = false;
35469     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35470 };
35471
35472 Roo.tree.TreeNodeUI.prototype = {
35473     removeChild : function(node){
35474         if(this.rendered){
35475             this.ctNode.removeChild(node.ui.getEl());
35476         }
35477     },
35478
35479     beforeLoad : function(){
35480          this.addClass("x-tree-node-loading");
35481     },
35482
35483     afterLoad : function(){
35484          this.removeClass("x-tree-node-loading");
35485     },
35486
35487     onTextChange : function(node, text, oldText){
35488         if(this.rendered){
35489             this.textNode.innerHTML = text;
35490         }
35491     },
35492
35493     onDisableChange : function(node, state){
35494         this.disabled = state;
35495         if(state){
35496             this.addClass("x-tree-node-disabled");
35497         }else{
35498             this.removeClass("x-tree-node-disabled");
35499         }
35500     },
35501
35502     onSelectedChange : function(state){
35503         if(state){
35504             this.focus();
35505             this.addClass("x-tree-selected");
35506         }else{
35507             //this.blur();
35508             this.removeClass("x-tree-selected");
35509         }
35510     },
35511
35512     onMove : function(tree, node, oldParent, newParent, index, refNode){
35513         this.childIndent = null;
35514         if(this.rendered){
35515             var targetNode = newParent.ui.getContainer();
35516             if(!targetNode){//target not rendered
35517                 this.holder = document.createElement("div");
35518                 this.holder.appendChild(this.wrap);
35519                 return;
35520             }
35521             var insertBefore = refNode ? refNode.ui.getEl() : null;
35522             if(insertBefore){
35523                 targetNode.insertBefore(this.wrap, insertBefore);
35524             }else{
35525                 targetNode.appendChild(this.wrap);
35526             }
35527             this.node.renderIndent(true);
35528         }
35529     },
35530
35531     addClass : function(cls){
35532         if(this.elNode){
35533             Roo.fly(this.elNode).addClass(cls);
35534         }
35535     },
35536
35537     removeClass : function(cls){
35538         if(this.elNode){
35539             Roo.fly(this.elNode).removeClass(cls);
35540         }
35541     },
35542
35543     remove : function(){
35544         if(this.rendered){
35545             this.holder = document.createElement("div");
35546             this.holder.appendChild(this.wrap);
35547         }
35548     },
35549
35550     fireEvent : function(){
35551         return this.node.fireEvent.apply(this.node, arguments);
35552     },
35553
35554     initEvents : function(){
35555         this.node.on("move", this.onMove, this);
35556         var E = Roo.EventManager;
35557         var a = this.anchor;
35558
35559         var el = Roo.fly(a, '_treeui');
35560
35561         if(Roo.isOpera){ // opera render bug ignores the CSS
35562             el.setStyle("text-decoration", "none");
35563         }
35564
35565         el.on("click", this.onClick, this);
35566         el.on("dblclick", this.onDblClick, this);
35567
35568         if(this.checkbox){
35569             Roo.EventManager.on(this.checkbox,
35570                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35571         }
35572
35573         el.on("contextmenu", this.onContextMenu, this);
35574
35575         var icon = Roo.fly(this.iconNode);
35576         icon.on("click", this.onClick, this);
35577         icon.on("dblclick", this.onDblClick, this);
35578         icon.on("contextmenu", this.onContextMenu, this);
35579         E.on(this.ecNode, "click", this.ecClick, this, true);
35580
35581         if(this.node.disabled){
35582             this.addClass("x-tree-node-disabled");
35583         }
35584         if(this.node.hidden){
35585             this.addClass("x-tree-node-disabled");
35586         }
35587         var ot = this.node.getOwnerTree();
35588         var dd = ot.enableDD || ot.enableDrag || ot.enableDrop;
35589         if(dd && (!this.node.isRoot || ot.rootVisible)){
35590             Roo.dd.Registry.register(this.elNode, {
35591                 node: this.node,
35592                 handles: this.getDDHandles(),
35593                 isHandle: false
35594             });
35595         }
35596     },
35597
35598     getDDHandles : function(){
35599         return [this.iconNode, this.textNode];
35600     },
35601
35602     hide : function(){
35603         if(this.rendered){
35604             this.wrap.style.display = "none";
35605         }
35606     },
35607
35608     show : function(){
35609         if(this.rendered){
35610             this.wrap.style.display = "";
35611         }
35612     },
35613
35614     onContextMenu : function(e){
35615         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35616             e.preventDefault();
35617             this.focus();
35618             this.fireEvent("contextmenu", this.node, e);
35619         }
35620     },
35621
35622     onClick : function(e){
35623         if(this.dropping){
35624             e.stopEvent();
35625             return;
35626         }
35627         if(this.fireEvent("beforeclick", this.node, e) !== false){
35628             if(!this.disabled && this.node.attributes.href){
35629                 this.fireEvent("click", this.node, e);
35630                 return;
35631             }
35632             e.preventDefault();
35633             if(this.disabled){
35634                 return;
35635             }
35636
35637             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35638                 this.node.toggle();
35639             }
35640
35641             this.fireEvent("click", this.node, e);
35642         }else{
35643             e.stopEvent();
35644         }
35645     },
35646
35647     onDblClick : function(e){
35648         e.preventDefault();
35649         if(this.disabled){
35650             return;
35651         }
35652         if(this.checkbox){
35653             this.toggleCheck();
35654         }
35655         if(!this.animating && this.node.hasChildNodes()){
35656             this.node.toggle();
35657         }
35658         this.fireEvent("dblclick", this.node, e);
35659     },
35660
35661     onCheckChange : function(){
35662         var checked = this.checkbox.checked;
35663         this.node.attributes.checked = checked;
35664         this.fireEvent('checkchange', this.node, checked);
35665     },
35666
35667     ecClick : function(e){
35668         if(!this.animating && this.node.hasChildNodes()){
35669             this.node.toggle();
35670         }
35671     },
35672
35673     startDrop : function(){
35674         this.dropping = true;
35675     },
35676
35677     // delayed drop so the click event doesn't get fired on a drop
35678     endDrop : function(){
35679        setTimeout(function(){
35680            this.dropping = false;
35681        }.createDelegate(this), 50);
35682     },
35683
35684     expand : function(){
35685         this.updateExpandIcon();
35686         this.ctNode.style.display = "";
35687     },
35688
35689     focus : function(){
35690         if(!this.node.preventHScroll){
35691             try{this.anchor.focus();
35692             }catch(e){}
35693         }else if(!Roo.isIE){
35694             try{
35695                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35696                 var l = noscroll.scrollLeft;
35697                 this.anchor.focus();
35698                 noscroll.scrollLeft = l;
35699             }catch(e){}
35700         }
35701     },
35702
35703     toggleCheck : function(value){
35704         var cb = this.checkbox;
35705         if(cb){
35706             cb.checked = (value === undefined ? !cb.checked : value);
35707         }
35708     },
35709
35710     blur : function(){
35711         try{
35712             this.anchor.blur();
35713         }catch(e){}
35714     },
35715
35716     animExpand : function(callback){
35717         var ct = Roo.get(this.ctNode);
35718         ct.stopFx();
35719         if(!this.node.hasChildNodes()){
35720             this.updateExpandIcon();
35721             this.ctNode.style.display = "";
35722             Roo.callback(callback);
35723             return;
35724         }
35725         this.animating = true;
35726         this.updateExpandIcon();
35727
35728         ct.slideIn('t', {
35729            callback : function(){
35730                this.animating = false;
35731                Roo.callback(callback);
35732             },
35733             scope: this,
35734             duration: this.node.ownerTree.duration || .25
35735         });
35736     },
35737
35738     highlight : function(){
35739         var tree = this.node.getOwnerTree();
35740         Roo.fly(this.wrap).highlight(
35741             tree.hlColor || "C3DAF9",
35742             {endColor: tree.hlBaseColor}
35743         );
35744     },
35745
35746     collapse : function(){
35747         this.updateExpandIcon();
35748         this.ctNode.style.display = "none";
35749     },
35750
35751     animCollapse : function(callback){
35752         var ct = Roo.get(this.ctNode);
35753         ct.enableDisplayMode('block');
35754         ct.stopFx();
35755
35756         this.animating = true;
35757         this.updateExpandIcon();
35758
35759         ct.slideOut('t', {
35760             callback : function(){
35761                this.animating = false;
35762                Roo.callback(callback);
35763             },
35764             scope: this,
35765             duration: this.node.ownerTree.duration || .25
35766         });
35767     },
35768
35769     getContainer : function(){
35770         return this.ctNode;
35771     },
35772
35773     getEl : function(){
35774         return this.wrap;
35775     },
35776
35777     appendDDGhost : function(ghostNode){
35778         ghostNode.appendChild(this.elNode.cloneNode(true));
35779     },
35780
35781     getDDRepairXY : function(){
35782         return Roo.lib.Dom.getXY(this.iconNode);
35783     },
35784
35785     onRender : function(){
35786         this.render();
35787     },
35788
35789     render : function(bulkRender){
35790         var n = this.node, a = n.attributes;
35791         var targetNode = n.parentNode ?
35792               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35793
35794         if(!this.rendered){
35795             this.rendered = true;
35796
35797             this.renderElements(n, a, targetNode, bulkRender);
35798
35799             if(a.qtip){
35800                if(this.textNode.setAttributeNS){
35801                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35802                    if(a.qtipTitle){
35803                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35804                    }
35805                }else{
35806                    this.textNode.setAttribute("ext:qtip", a.qtip);
35807                    if(a.qtipTitle){
35808                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35809                    }
35810                }
35811             }else if(a.qtipCfg){
35812                 a.qtipCfg.target = Roo.id(this.textNode);
35813                 Roo.QuickTips.register(a.qtipCfg);
35814             }
35815             this.initEvents();
35816             if(!this.node.expanded){
35817                 this.updateExpandIcon();
35818             }
35819         }else{
35820             if(bulkRender === true) {
35821                 targetNode.appendChild(this.wrap);
35822             }
35823         }
35824     },
35825
35826     renderElements : function(n, a, targetNode, bulkRender)
35827     {
35828         // add some indent caching, this helps performance when rendering a large tree
35829         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35830         var t = n.getOwnerTree();
35831         var txt = t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35832         if (typeof(n.attributes.html) != 'undefined') {
35833             txt = n.attributes.html;
35834         }
35835         var tip = t.rendererTip ? t.rendererTip(n.attributes) : txt;
35836         var cb = typeof a.checked == 'boolean';
35837         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35838         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35839             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35840             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35841             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35842             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35843             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35844              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35845                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35846             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35847             "</li>"];
35848
35849         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35850             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35851                                 n.nextSibling.ui.getEl(), buf.join(""));
35852         }else{
35853             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35854         }
35855
35856         this.elNode = this.wrap.childNodes[0];
35857         this.ctNode = this.wrap.childNodes[1];
35858         var cs = this.elNode.childNodes;
35859         this.indentNode = cs[0];
35860         this.ecNode = cs[1];
35861         this.iconNode = cs[2];
35862         var index = 3;
35863         if(cb){
35864             this.checkbox = cs[3];
35865             index++;
35866         }
35867         this.anchor = cs[index];
35868         this.textNode = cs[index].firstChild;
35869     },
35870
35871     getAnchor : function(){
35872         return this.anchor;
35873     },
35874
35875     getTextEl : function(){
35876         return this.textNode;
35877     },
35878
35879     getIconEl : function(){
35880         return this.iconNode;
35881     },
35882
35883     isChecked : function(){
35884         return this.checkbox ? this.checkbox.checked : false;
35885     },
35886
35887     updateExpandIcon : function(){
35888         if(this.rendered){
35889             var n = this.node, c1, c2;
35890             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35891             var hasChild = n.hasChildNodes();
35892             if(hasChild){
35893                 if(n.expanded){
35894                     cls += "-minus";
35895                     c1 = "x-tree-node-collapsed";
35896                     c2 = "x-tree-node-expanded";
35897                 }else{
35898                     cls += "-plus";
35899                     c1 = "x-tree-node-expanded";
35900                     c2 = "x-tree-node-collapsed";
35901                 }
35902                 if(this.wasLeaf){
35903                     this.removeClass("x-tree-node-leaf");
35904                     this.wasLeaf = false;
35905                 }
35906                 if(this.c1 != c1 || this.c2 != c2){
35907                     Roo.fly(this.elNode).replaceClass(c1, c2);
35908                     this.c1 = c1; this.c2 = c2;
35909                 }
35910             }else{
35911                 // this changes non-leafs into leafs if they have no children.
35912                 // it's not very rational behaviour..
35913                 
35914                 if(!this.wasLeaf && this.node.leaf){
35915                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35916                     delete this.c1;
35917                     delete this.c2;
35918                     this.wasLeaf = true;
35919                 }
35920             }
35921             var ecc = "x-tree-ec-icon "+cls;
35922             if(this.ecc != ecc){
35923                 this.ecNode.className = ecc;
35924                 this.ecc = ecc;
35925             }
35926         }
35927     },
35928
35929     getChildIndent : function(){
35930         if(!this.childIndent){
35931             var buf = [];
35932             var p = this.node;
35933             while(p){
35934                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35935                     if(!p.isLast()) {
35936                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35937                     } else {
35938                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35939                     }
35940                 }
35941                 p = p.parentNode;
35942             }
35943             this.childIndent = buf.join("");
35944         }
35945         return this.childIndent;
35946     },
35947
35948     renderIndent : function(){
35949         if(this.rendered){
35950             var indent = "";
35951             var p = this.node.parentNode;
35952             if(p){
35953                 indent = p.ui.getChildIndent();
35954             }
35955             if(this.indentMarkup != indent){ // don't rerender if not required
35956                 this.indentNode.innerHTML = indent;
35957                 this.indentMarkup = indent;
35958             }
35959             this.updateExpandIcon();
35960         }
35961     }
35962 };
35963
35964 Roo.tree.RootTreeNodeUI = function(){
35965     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
35966 };
35967 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
35968     render : function(){
35969         if(!this.rendered){
35970             var targetNode = this.node.ownerTree.innerCt.dom;
35971             this.node.expanded = true;
35972             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
35973             this.wrap = this.ctNode = targetNode.firstChild;
35974         }
35975     },
35976     collapse : function(){
35977     },
35978     expand : function(){
35979     }
35980 });/*
35981  * Based on:
35982  * Ext JS Library 1.1.1
35983  * Copyright(c) 2006-2007, Ext JS, LLC.
35984  *
35985  * Originally Released Under LGPL - original licence link has changed is not relivant.
35986  *
35987  * Fork - LGPL
35988  * <script type="text/javascript">
35989  */
35990 /**
35991  * @class Roo.tree.TreeLoader
35992  * @extends Roo.util.Observable
35993  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
35994  * nodes from a specified URL. The response must be a javascript Array definition
35995  * who's elements are node definition objects. eg:
35996  * <pre><code>
35997 {  success : true,
35998    data :      [
35999    
36000     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36001     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36002     ]
36003 }
36004
36005
36006 </code></pre>
36007  * <br><br>
36008  * The old style respose with just an array is still supported, but not recommended.
36009  * <br><br>
36010  *
36011  * A server request is sent, and child nodes are loaded only when a node is expanded.
36012  * The loading node's id is passed to the server under the parameter name "node" to
36013  * enable the server to produce the correct child nodes.
36014  * <br><br>
36015  * To pass extra parameters, an event handler may be attached to the "beforeload"
36016  * event, and the parameters specified in the TreeLoader's baseParams property:
36017  * <pre><code>
36018     myTreeLoader.on("beforeload", function(treeLoader, node) {
36019         this.baseParams.category = node.attributes.category;
36020     }, this);
36021     
36022 </code></pre>
36023  *
36024  * This would pass an HTTP parameter called "category" to the server containing
36025  * the value of the Node's "category" attribute.
36026  * @constructor
36027  * Creates a new Treeloader.
36028  * @param {Object} config A config object containing config properties.
36029  */
36030 Roo.tree.TreeLoader = function(config){
36031     this.baseParams = {};
36032     this.requestMethod = "POST";
36033     Roo.apply(this, config);
36034
36035     this.addEvents({
36036     
36037         /**
36038          * @event beforeload
36039          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36040          * @param {Object} This TreeLoader object.
36041          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36042          * @param {Object} callback The callback function specified in the {@link #load} call.
36043          */
36044         beforeload : true,
36045         /**
36046          * @event load
36047          * Fires when the node has been successfuly loaded.
36048          * @param {Object} This TreeLoader object.
36049          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36050          * @param {Object} response The response object containing the data from the server.
36051          */
36052         load : true,
36053         /**
36054          * @event loadexception
36055          * Fires if the network request failed.
36056          * @param {Object} This TreeLoader object.
36057          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36058          * @param {Object} response The response object containing the data from the server.
36059          */
36060         loadexception : true,
36061         /**
36062          * @event create
36063          * Fires before a node is created, enabling you to return custom Node types 
36064          * @param {Object} This TreeLoader object.
36065          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36066          */
36067         create : true
36068     });
36069
36070     Roo.tree.TreeLoader.superclass.constructor.call(this);
36071 };
36072
36073 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36074     /**
36075     * @cfg {String} dataUrl The URL from which to request a Json string which
36076     * specifies an array of node definition object representing the child nodes
36077     * to be loaded.
36078     */
36079     /**
36080     * @cfg {String} requestMethod either GET or POST
36081     * defaults to POST (due to BC)
36082     * to be loaded.
36083     */
36084     /**
36085     * @cfg {Object} baseParams (optional) An object containing properties which
36086     * specify HTTP parameters to be passed to each request for child nodes.
36087     */
36088     /**
36089     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36090     * created by this loader. If the attributes sent by the server have an attribute in this object,
36091     * they take priority.
36092     */
36093     /**
36094     * @cfg {Object} uiProviders (optional) An object containing properties which
36095     * 
36096     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36097     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36098     * <i>uiProvider</i> attribute of a returned child node is a string rather
36099     * than a reference to a TreeNodeUI implementation, this that string value
36100     * is used as a property name in the uiProviders object. You can define the provider named
36101     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36102     */
36103     uiProviders : {},
36104
36105     /**
36106     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36107     * child nodes before loading.
36108     */
36109     clearOnLoad : true,
36110
36111     /**
36112     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36113     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36114     * Grid query { data : [ .....] }
36115     */
36116     
36117     root : false,
36118      /**
36119     * @cfg {String} queryParam (optional) 
36120     * Name of the query as it will be passed on the querystring (defaults to 'node')
36121     * eg. the request will be ?node=[id]
36122     */
36123     
36124     
36125     queryParam: false,
36126     
36127     /**
36128      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36129      * This is called automatically when a node is expanded, but may be used to reload
36130      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36131      * @param {Roo.tree.TreeNode} node
36132      * @param {Function} callback
36133      */
36134     load : function(node, callback){
36135         if(this.clearOnLoad){
36136             while(node.firstChild){
36137                 node.removeChild(node.firstChild);
36138             }
36139         }
36140         if(node.attributes.children){ // preloaded json children
36141             var cs = node.attributes.children;
36142             for(var i = 0, len = cs.length; i < len; i++){
36143                 node.appendChild(this.createNode(cs[i]));
36144             }
36145             if(typeof callback == "function"){
36146                 callback();
36147             }
36148         }else if(this.dataUrl){
36149             this.requestData(node, callback);
36150         }
36151     },
36152
36153     getParams: function(node){
36154         var buf = [], bp = this.baseParams;
36155         for(var key in bp){
36156             if(typeof bp[key] != "function"){
36157                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36158             }
36159         }
36160         var n = this.queryParam === false ? 'node' : this.queryParam;
36161         buf.push(n + "=", encodeURIComponent(node.id));
36162         return buf.join("");
36163     },
36164
36165     requestData : function(node, callback){
36166         if(this.fireEvent("beforeload", this, node, callback) !== false){
36167             this.transId = Roo.Ajax.request({
36168                 method:this.requestMethod,
36169                 url: this.dataUrl||this.url,
36170                 success: this.handleResponse,
36171                 failure: this.handleFailure,
36172                 scope: this,
36173                 argument: {callback: callback, node: node},
36174                 params: this.getParams(node)
36175             });
36176         }else{
36177             // if the load is cancelled, make sure we notify
36178             // the node that we are done
36179             if(typeof callback == "function"){
36180                 callback();
36181             }
36182         }
36183     },
36184
36185     isLoading : function(){
36186         return this.transId ? true : false;
36187     },
36188
36189     abort : function(){
36190         if(this.isLoading()){
36191             Roo.Ajax.abort(this.transId);
36192         }
36193     },
36194
36195     // private
36196     createNode : function(attr)
36197     {
36198         // apply baseAttrs, nice idea Corey!
36199         if(this.baseAttrs){
36200             Roo.applyIf(attr, this.baseAttrs);
36201         }
36202         if(this.applyLoader !== false){
36203             attr.loader = this;
36204         }
36205         // uiProvider = depreciated..
36206         
36207         if(typeof(attr.uiProvider) == 'string'){
36208            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36209                 /**  eval:var:attr */ eval(attr.uiProvider);
36210         }
36211         if(typeof(this.uiProviders['default']) != 'undefined') {
36212             attr.uiProvider = this.uiProviders['default'];
36213         }
36214         
36215         this.fireEvent('create', this, attr);
36216         
36217         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36218         return(attr.leaf ?
36219                         new Roo.tree.TreeNode(attr) :
36220                         new Roo.tree.AsyncTreeNode(attr));
36221     },
36222
36223     processResponse : function(response, node, callback)
36224     {
36225         var json = response.responseText;
36226         try {
36227             
36228             var o = Roo.decode(json);
36229             
36230             if (this.root === false && typeof(o.success) != undefined) {
36231                 this.root = 'data'; // the default behaviour for list like data..
36232                 }
36233                 
36234             if (this.root !== false &&  !o.success) {
36235                 // it's a failure condition.
36236                 var a = response.argument;
36237                 this.fireEvent("loadexception", this, a.node, response);
36238                 Roo.log("Load failed - should have a handler really");
36239                 return;
36240             }
36241             
36242             
36243             
36244             if (this.root !== false) {
36245                  o = o[this.root];
36246             }
36247             
36248             for(var i = 0, len = o.length; i < len; i++){
36249                 var n = this.createNode(o[i]);
36250                 if(n){
36251                     node.appendChild(n);
36252                 }
36253             }
36254             if(typeof callback == "function"){
36255                 callback(this, node);
36256             }
36257         }catch(e){
36258             this.handleFailure(response);
36259         }
36260     },
36261
36262     handleResponse : function(response){
36263         this.transId = false;
36264         var a = response.argument;
36265         this.processResponse(response, a.node, a.callback);
36266         this.fireEvent("load", this, a.node, response);
36267     },
36268
36269     handleFailure : function(response)
36270     {
36271         // should handle failure better..
36272         this.transId = false;
36273         var a = response.argument;
36274         this.fireEvent("loadexception", this, a.node, response);
36275         if(typeof a.callback == "function"){
36276             a.callback(this, a.node);
36277         }
36278     }
36279 });/*
36280  * Based on:
36281  * Ext JS Library 1.1.1
36282  * Copyright(c) 2006-2007, Ext JS, LLC.
36283  *
36284  * Originally Released Under LGPL - original licence link has changed is not relivant.
36285  *
36286  * Fork - LGPL
36287  * <script type="text/javascript">
36288  */
36289
36290 /**
36291 * @class Roo.tree.TreeFilter
36292 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36293 * @param {TreePanel} tree
36294 * @param {Object} config (optional)
36295  */
36296 Roo.tree.TreeFilter = function(tree, config){
36297     this.tree = tree;
36298     this.filtered = {};
36299     Roo.apply(this, config);
36300 };
36301
36302 Roo.tree.TreeFilter.prototype = {
36303     clearBlank:false,
36304     reverse:false,
36305     autoClear:false,
36306     remove:false,
36307
36308      /**
36309      * Filter the data by a specific attribute.
36310      * @param {String/RegExp} value Either string that the attribute value
36311      * should start with or a RegExp to test against the attribute
36312      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36313      * @param {TreeNode} startNode (optional) The node to start the filter at.
36314      */
36315     filter : function(value, attr, startNode){
36316         attr = attr || "text";
36317         var f;
36318         if(typeof value == "string"){
36319             var vlen = value.length;
36320             // auto clear empty filter
36321             if(vlen == 0 && this.clearBlank){
36322                 this.clear();
36323                 return;
36324             }
36325             value = value.toLowerCase();
36326             f = function(n){
36327                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36328             };
36329         }else if(value.exec){ // regex?
36330             f = function(n){
36331                 return value.test(n.attributes[attr]);
36332             };
36333         }else{
36334             throw 'Illegal filter type, must be string or regex';
36335         }
36336         this.filterBy(f, null, startNode);
36337         },
36338
36339     /**
36340      * Filter by a function. The passed function will be called with each
36341      * node in the tree (or from the startNode). If the function returns true, the node is kept
36342      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36343      * @param {Function} fn The filter function
36344      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36345      */
36346     filterBy : function(fn, scope, startNode){
36347         startNode = startNode || this.tree.root;
36348         if(this.autoClear){
36349             this.clear();
36350         }
36351         var af = this.filtered, rv = this.reverse;
36352         var f = function(n){
36353             if(n == startNode){
36354                 return true;
36355             }
36356             if(af[n.id]){
36357                 return false;
36358             }
36359             var m = fn.call(scope || n, n);
36360             if(!m || rv){
36361                 af[n.id] = n;
36362                 n.ui.hide();
36363                 return false;
36364             }
36365             return true;
36366         };
36367         startNode.cascade(f);
36368         if(this.remove){
36369            for(var id in af){
36370                if(typeof id != "function"){
36371                    var n = af[id];
36372                    if(n && n.parentNode){
36373                        n.parentNode.removeChild(n);
36374                    }
36375                }
36376            }
36377         }
36378     },
36379
36380     /**
36381      * Clears the current filter. Note: with the "remove" option
36382      * set a filter cannot be cleared.
36383      */
36384     clear : function(){
36385         var t = this.tree;
36386         var af = this.filtered;
36387         for(var id in af){
36388             if(typeof id != "function"){
36389                 var n = af[id];
36390                 if(n){
36391                     n.ui.show();
36392                 }
36393             }
36394         }
36395         this.filtered = {};
36396     }
36397 };
36398 /*
36399  * Based on:
36400  * Ext JS Library 1.1.1
36401  * Copyright(c) 2006-2007, Ext JS, LLC.
36402  *
36403  * Originally Released Under LGPL - original licence link has changed is not relivant.
36404  *
36405  * Fork - LGPL
36406  * <script type="text/javascript">
36407  */
36408  
36409
36410 /**
36411  * @class Roo.tree.TreeSorter
36412  * Provides sorting of nodes in a TreePanel
36413  * 
36414  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36415  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36416  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36417  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36418  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36419  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36420  * @constructor
36421  * @param {TreePanel} tree
36422  * @param {Object} config
36423  */
36424 Roo.tree.TreeSorter = function(tree, config){
36425     Roo.apply(this, config);
36426     tree.on("beforechildrenrendered", this.doSort, this);
36427     tree.on("append", this.updateSort, this);
36428     tree.on("insert", this.updateSort, this);
36429     
36430     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36431     var p = this.property || "text";
36432     var sortType = this.sortType;
36433     var fs = this.folderSort;
36434     var cs = this.caseSensitive === true;
36435     var leafAttr = this.leafAttr || 'leaf';
36436
36437     this.sortFn = function(n1, n2){
36438         if(fs){
36439             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36440                 return 1;
36441             }
36442             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36443                 return -1;
36444             }
36445         }
36446         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36447         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36448         if(v1 < v2){
36449                         return dsc ? +1 : -1;
36450                 }else if(v1 > v2){
36451                         return dsc ? -1 : +1;
36452         }else{
36453                 return 0;
36454         }
36455     };
36456 };
36457
36458 Roo.tree.TreeSorter.prototype = {
36459     doSort : function(node){
36460         node.sort(this.sortFn);
36461     },
36462     
36463     compareNodes : function(n1, n2){
36464         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36465     },
36466     
36467     updateSort : function(tree, node){
36468         if(node.childrenRendered){
36469             this.doSort.defer(1, this, [node]);
36470         }
36471     }
36472 };/*
36473  * Based on:
36474  * Ext JS Library 1.1.1
36475  * Copyright(c) 2006-2007, Ext JS, LLC.
36476  *
36477  * Originally Released Under LGPL - original licence link has changed is not relivant.
36478  *
36479  * Fork - LGPL
36480  * <script type="text/javascript">
36481  */
36482
36483 if(Roo.dd.DropZone){
36484     
36485 Roo.tree.TreeDropZone = function(tree, config){
36486     this.allowParentInsert = false;
36487     this.allowContainerDrop = false;
36488     this.appendOnly = false;
36489     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36490     this.tree = tree;
36491     this.lastInsertClass = "x-tree-no-status";
36492     this.dragOverData = {};
36493 };
36494
36495 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36496     ddGroup : "TreeDD",
36497     scroll:  true,
36498     
36499     expandDelay : 1000,
36500     
36501     expandNode : function(node){
36502         if(node.hasChildNodes() && !node.isExpanded()){
36503             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36504         }
36505     },
36506     
36507     queueExpand : function(node){
36508         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36509     },
36510     
36511     cancelExpand : function(){
36512         if(this.expandProcId){
36513             clearTimeout(this.expandProcId);
36514             this.expandProcId = false;
36515         }
36516     },
36517     
36518     isValidDropPoint : function(n, pt, dd, e, data){
36519         if(!n || !data){ return false; }
36520         var targetNode = n.node;
36521         var dropNode = data.node;
36522         // default drop rules
36523         if(!(targetNode && targetNode.isTarget && pt)){
36524             return false;
36525         }
36526         if(pt == "append" && targetNode.allowChildren === false){
36527             return false;
36528         }
36529         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36530             return false;
36531         }
36532         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36533             return false;
36534         }
36535         // reuse the object
36536         var overEvent = this.dragOverData;
36537         overEvent.tree = this.tree;
36538         overEvent.target = targetNode;
36539         overEvent.data = data;
36540         overEvent.point = pt;
36541         overEvent.source = dd;
36542         overEvent.rawEvent = e;
36543         overEvent.dropNode = dropNode;
36544         overEvent.cancel = false;  
36545         var result = this.tree.fireEvent("nodedragover", overEvent);
36546         return overEvent.cancel === false && result !== false;
36547     },
36548     
36549     getDropPoint : function(e, n, dd)
36550     {
36551         var tn = n.node;
36552         if(tn.isRoot){
36553             return tn.allowChildren !== false ? "append" : false; // always append for root
36554         }
36555         var dragEl = n.ddel;
36556         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36557         var y = Roo.lib.Event.getPageY(e);
36558         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36559         
36560         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36561         var noAppend = tn.allowChildren === false;
36562         if(this.appendOnly || tn.parentNode.allowChildren === false){
36563             return noAppend ? false : "append";
36564         }
36565         var noBelow = false;
36566         if(!this.allowParentInsert){
36567             noBelow = tn.hasChildNodes() && tn.isExpanded();
36568         }
36569         var q = (b - t) / (noAppend ? 2 : 3);
36570         if(y >= t && y < (t + q)){
36571             return "above";
36572         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36573             return "below";
36574         }else{
36575             return "append";
36576         }
36577     },
36578     
36579     onNodeEnter : function(n, dd, e, data)
36580     {
36581         this.cancelExpand();
36582     },
36583     
36584     onNodeOver : function(n, dd, e, data)
36585     {
36586        
36587         var pt = this.getDropPoint(e, n, dd);
36588         var node = n.node;
36589         
36590         // auto node expand check
36591         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36592             this.queueExpand(node);
36593         }else if(pt != "append"){
36594             this.cancelExpand();
36595         }
36596         
36597         // set the insert point style on the target node
36598         var returnCls = this.dropNotAllowed;
36599         if(this.isValidDropPoint(n, pt, dd, e, data)){
36600            if(pt){
36601                var el = n.ddel;
36602                var cls;
36603                if(pt == "above"){
36604                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36605                    cls = "x-tree-drag-insert-above";
36606                }else if(pt == "below"){
36607                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36608                    cls = "x-tree-drag-insert-below";
36609                }else{
36610                    returnCls = "x-tree-drop-ok-append";
36611                    cls = "x-tree-drag-append";
36612                }
36613                if(this.lastInsertClass != cls){
36614                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36615                    this.lastInsertClass = cls;
36616                }
36617            }
36618        }
36619        return returnCls;
36620     },
36621     
36622     onNodeOut : function(n, dd, e, data){
36623         
36624         this.cancelExpand();
36625         this.removeDropIndicators(n);
36626     },
36627     
36628     onNodeDrop : function(n, dd, e, data){
36629         var point = this.getDropPoint(e, n, dd);
36630         var targetNode = n.node;
36631         targetNode.ui.startDrop();
36632         if(!this.isValidDropPoint(n, point, dd, e, data)){
36633             targetNode.ui.endDrop();
36634             return false;
36635         }
36636         // first try to find the drop node
36637         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36638         var dropEvent = {
36639             tree : this.tree,
36640             target: targetNode,
36641             data: data,
36642             point: point,
36643             source: dd,
36644             rawEvent: e,
36645             dropNode: dropNode,
36646             cancel: !dropNode   
36647         };
36648         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36649         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36650             targetNode.ui.endDrop();
36651             return false;
36652         }
36653         // allow target changing
36654         targetNode = dropEvent.target;
36655         if(point == "append" && !targetNode.isExpanded()){
36656             targetNode.expand(false, null, function(){
36657                 this.completeDrop(dropEvent);
36658             }.createDelegate(this));
36659         }else{
36660             this.completeDrop(dropEvent);
36661         }
36662         return true;
36663     },
36664     
36665     completeDrop : function(de){
36666         var ns = de.dropNode, p = de.point, t = de.target;
36667         if(!(ns instanceof Array)){
36668             ns = [ns];
36669         }
36670         var n;
36671         for(var i = 0, len = ns.length; i < len; i++){
36672             n = ns[i];
36673             if(p == "above"){
36674                 t.parentNode.insertBefore(n, t);
36675             }else if(p == "below"){
36676                 t.parentNode.insertBefore(n, t.nextSibling);
36677             }else{
36678                 t.appendChild(n);
36679             }
36680         }
36681         n.ui.focus();
36682         if(this.tree.hlDrop){
36683             n.ui.highlight();
36684         }
36685         t.ui.endDrop();
36686         this.tree.fireEvent("nodedrop", de);
36687     },
36688     
36689     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36690         if(this.tree.hlDrop){
36691             dropNode.ui.focus();
36692             dropNode.ui.highlight();
36693         }
36694         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36695     },
36696     
36697     getTree : function(){
36698         return this.tree;
36699     },
36700     
36701     removeDropIndicators : function(n){
36702         if(n && n.ddel){
36703             var el = n.ddel;
36704             Roo.fly(el).removeClass([
36705                     "x-tree-drag-insert-above",
36706                     "x-tree-drag-insert-below",
36707                     "x-tree-drag-append"]);
36708             this.lastInsertClass = "_noclass";
36709         }
36710     },
36711     
36712     beforeDragDrop : function(target, e, id){
36713         this.cancelExpand();
36714         return true;
36715     },
36716     
36717     afterRepair : function(data){
36718         if(data && Roo.enableFx){
36719             data.node.ui.highlight();
36720         }
36721         this.hideProxy();
36722     } 
36723     
36724 });
36725
36726 }
36727 /*
36728  * Based on:
36729  * Ext JS Library 1.1.1
36730  * Copyright(c) 2006-2007, Ext JS, LLC.
36731  *
36732  * Originally Released Under LGPL - original licence link has changed is not relivant.
36733  *
36734  * Fork - LGPL
36735  * <script type="text/javascript">
36736  */
36737  
36738
36739 if(Roo.dd.DragZone){
36740 Roo.tree.TreeDragZone = function(tree, config){
36741     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36742     this.tree = tree;
36743 };
36744
36745 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36746     ddGroup : "TreeDD",
36747    
36748     onBeforeDrag : function(data, e){
36749         var n = data.node;
36750         return n && n.draggable && !n.disabled;
36751     },
36752      
36753     
36754     onInitDrag : function(e){
36755         var data = this.dragData;
36756         this.tree.getSelectionModel().select(data.node);
36757         this.proxy.update("");
36758         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36759         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36760     },
36761     
36762     getRepairXY : function(e, data){
36763         return data.node.ui.getDDRepairXY();
36764     },
36765     
36766     onEndDrag : function(data, e){
36767         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36768         
36769         
36770     },
36771     
36772     onValidDrop : function(dd, e, id){
36773         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36774         this.hideProxy();
36775     },
36776     
36777     beforeInvalidDrop : function(e, id){
36778         // this scrolls the original position back into view
36779         var sm = this.tree.getSelectionModel();
36780         sm.clearSelections();
36781         sm.select(this.dragData.node);
36782     }
36783 });
36784 }/*
36785  * Based on:
36786  * Ext JS Library 1.1.1
36787  * Copyright(c) 2006-2007, Ext JS, LLC.
36788  *
36789  * Originally Released Under LGPL - original licence link has changed is not relivant.
36790  *
36791  * Fork - LGPL
36792  * <script type="text/javascript">
36793  */
36794 /**
36795  * @class Roo.tree.TreeEditor
36796  * @extends Roo.Editor
36797  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36798  * as the editor field.
36799  * @constructor
36800  * @param {Object} config (used to be the tree panel.)
36801  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36802  * 
36803  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36804  * @cfg {Roo.form.TextField|Object} field The field configuration
36805  *
36806  * 
36807  */
36808 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36809     var tree = config;
36810     var field;
36811     if (oldconfig) { // old style..
36812         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36813     } else {
36814         // new style..
36815         tree = config.tree;
36816         config.field = config.field  || {};
36817         config.field.xtype = 'TextField';
36818         field = Roo.factory(config.field, Roo.form);
36819     }
36820     config = config || {};
36821     
36822     
36823     this.addEvents({
36824         /**
36825          * @event beforenodeedit
36826          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36827          * false from the handler of this event.
36828          * @param {Editor} this
36829          * @param {Roo.tree.Node} node 
36830          */
36831         "beforenodeedit" : true
36832     });
36833     
36834     //Roo.log(config);
36835     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36836
36837     this.tree = tree;
36838
36839     tree.on('beforeclick', this.beforeNodeClick, this);
36840     tree.getTreeEl().on('mousedown', this.hide, this);
36841     this.on('complete', this.updateNode, this);
36842     this.on('beforestartedit', this.fitToTree, this);
36843     this.on('startedit', this.bindScroll, this, {delay:10});
36844     this.on('specialkey', this.onSpecialKey, this);
36845 };
36846
36847 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36848     /**
36849      * @cfg {String} alignment
36850      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36851      */
36852     alignment: "l-l",
36853     // inherit
36854     autoSize: false,
36855     /**
36856      * @cfg {Boolean} hideEl
36857      * True to hide the bound element while the editor is displayed (defaults to false)
36858      */
36859     hideEl : false,
36860     /**
36861      * @cfg {String} cls
36862      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36863      */
36864     cls: "x-small-editor x-tree-editor",
36865     /**
36866      * @cfg {Boolean} shim
36867      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36868      */
36869     shim:false,
36870     // inherit
36871     shadow:"frame",
36872     /**
36873      * @cfg {Number} maxWidth
36874      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36875      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36876      * scroll and client offsets into account prior to each edit.
36877      */
36878     maxWidth: 250,
36879
36880     editDelay : 350,
36881
36882     // private
36883     fitToTree : function(ed, el){
36884         var td = this.tree.getTreeEl().dom, nd = el.dom;
36885         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36886             td.scrollLeft = nd.offsetLeft;
36887         }
36888         var w = Math.min(
36889                 this.maxWidth,
36890                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36891         this.setSize(w, '');
36892         
36893         return this.fireEvent('beforenodeedit', this, this.editNode);
36894         
36895     },
36896
36897     // private
36898     triggerEdit : function(node){
36899         this.completeEdit();
36900         this.editNode = node;
36901         this.startEdit(node.ui.textNode, node.text);
36902     },
36903
36904     // private
36905     bindScroll : function(){
36906         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36907     },
36908
36909     // private
36910     beforeNodeClick : function(node, e){
36911         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36912         this.lastClick = new Date();
36913         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36914             e.stopEvent();
36915             this.triggerEdit(node);
36916             return false;
36917         }
36918         return true;
36919     },
36920
36921     // private
36922     updateNode : function(ed, value){
36923         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36924         this.editNode.setText(value);
36925     },
36926
36927     // private
36928     onHide : function(){
36929         Roo.tree.TreeEditor.superclass.onHide.call(this);
36930         if(this.editNode){
36931             this.editNode.ui.focus();
36932         }
36933     },
36934
36935     // private
36936     onSpecialKey : function(field, e){
36937         var k = e.getKey();
36938         if(k == e.ESC){
36939             e.stopEvent();
36940             this.cancelEdit();
36941         }else if(k == e.ENTER && !e.hasModifier()){
36942             e.stopEvent();
36943             this.completeEdit();
36944         }
36945     }
36946 });//<Script type="text/javascript">
36947 /*
36948  * Based on:
36949  * Ext JS Library 1.1.1
36950  * Copyright(c) 2006-2007, Ext JS, LLC.
36951  *
36952  * Originally Released Under LGPL - original licence link has changed is not relivant.
36953  *
36954  * Fork - LGPL
36955  * <script type="text/javascript">
36956  */
36957  
36958 /**
36959  * Not documented??? - probably should be...
36960  */
36961
36962 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
36963     //focus: Roo.emptyFn, // prevent odd scrolling behavior
36964     
36965     renderElements : function(n, a, targetNode, bulkRender){
36966         //consel.log("renderElements?");
36967         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
36968
36969         var t = n.getOwnerTree();
36970         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
36971         
36972         var cols = t.columns;
36973         var bw = t.borderWidth;
36974         var c = cols[0];
36975         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
36976          var cb = typeof a.checked == "boolean";
36977         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
36978         var colcls = 'x-t-' + tid + '-c0';
36979         var buf = [
36980             '<li class="x-tree-node">',
36981             
36982                 
36983                 '<div class="x-tree-node-el ', a.cls,'">',
36984                     // extran...
36985                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
36986                 
36987                 
36988                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
36989                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
36990                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
36991                            (a.icon ? ' x-tree-node-inline-icon' : ''),
36992                            (a.iconCls ? ' '+a.iconCls : ''),
36993                            '" unselectable="on" />',
36994                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
36995                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
36996                              
36997                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
36998                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
36999                             '<span unselectable="on" qtip="' + tx + '">',
37000                              tx,
37001                              '</span></a>' ,
37002                     '</div>',
37003                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37004                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37005                  ];
37006         for(var i = 1, len = cols.length; i < len; i++){
37007             c = cols[i];
37008             colcls = 'x-t-' + tid + '-c' +i;
37009             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37010             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37011                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37012                       "</div>");
37013          }
37014          
37015          buf.push(
37016             '</a>',
37017             '<div class="x-clear"></div></div>',
37018             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37019             "</li>");
37020         
37021         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37022             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37023                                 n.nextSibling.ui.getEl(), buf.join(""));
37024         }else{
37025             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37026         }
37027         var el = this.wrap.firstChild;
37028         this.elRow = el;
37029         this.elNode = el.firstChild;
37030         this.ranchor = el.childNodes[1];
37031         this.ctNode = this.wrap.childNodes[1];
37032         var cs = el.firstChild.childNodes;
37033         this.indentNode = cs[0];
37034         this.ecNode = cs[1];
37035         this.iconNode = cs[2];
37036         var index = 3;
37037         if(cb){
37038             this.checkbox = cs[3];
37039             index++;
37040         }
37041         this.anchor = cs[index];
37042         
37043         this.textNode = cs[index].firstChild;
37044         
37045         //el.on("click", this.onClick, this);
37046         //el.on("dblclick", this.onDblClick, this);
37047         
37048         
37049        // console.log(this);
37050     },
37051     initEvents : function(){
37052         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37053         
37054             
37055         var a = this.ranchor;
37056
37057         var el = Roo.get(a);
37058
37059         if(Roo.isOpera){ // opera render bug ignores the CSS
37060             el.setStyle("text-decoration", "none");
37061         }
37062
37063         el.on("click", this.onClick, this);
37064         el.on("dblclick", this.onDblClick, this);
37065         el.on("contextmenu", this.onContextMenu, this);
37066         
37067     },
37068     
37069     /*onSelectedChange : function(state){
37070         if(state){
37071             this.focus();
37072             this.addClass("x-tree-selected");
37073         }else{
37074             //this.blur();
37075             this.removeClass("x-tree-selected");
37076         }
37077     },*/
37078     addClass : function(cls){
37079         if(this.elRow){
37080             Roo.fly(this.elRow).addClass(cls);
37081         }
37082         
37083     },
37084     
37085     
37086     removeClass : function(cls){
37087         if(this.elRow){
37088             Roo.fly(this.elRow).removeClass(cls);
37089         }
37090     }
37091
37092     
37093     
37094 });//<Script type="text/javascript">
37095
37096 /*
37097  * Based on:
37098  * Ext JS Library 1.1.1
37099  * Copyright(c) 2006-2007, Ext JS, LLC.
37100  *
37101  * Originally Released Under LGPL - original licence link has changed is not relivant.
37102  *
37103  * Fork - LGPL
37104  * <script type="text/javascript">
37105  */
37106  
37107
37108 /**
37109  * @class Roo.tree.ColumnTree
37110  * @extends Roo.data.TreePanel
37111  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37112  * @cfg {int} borderWidth  compined right/left border allowance
37113  * @constructor
37114  * @param {String/HTMLElement/Element} el The container element
37115  * @param {Object} config
37116  */
37117 Roo.tree.ColumnTree =  function(el, config)
37118 {
37119    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37120    this.addEvents({
37121         /**
37122         * @event resize
37123         * Fire this event on a container when it resizes
37124         * @param {int} w Width
37125         * @param {int} h Height
37126         */
37127        "resize" : true
37128     });
37129     this.on('resize', this.onResize, this);
37130 };
37131
37132 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37133     //lines:false,
37134     
37135     
37136     borderWidth: Roo.isBorderBox ? 0 : 2, 
37137     headEls : false,
37138     
37139     render : function(){
37140         // add the header.....
37141        
37142         Roo.tree.ColumnTree.superclass.render.apply(this);
37143         
37144         this.el.addClass('x-column-tree');
37145         
37146         this.headers = this.el.createChild(
37147             {cls:'x-tree-headers'},this.innerCt.dom);
37148    
37149         var cols = this.columns, c;
37150         var totalWidth = 0;
37151         this.headEls = [];
37152         var  len = cols.length;
37153         for(var i = 0; i < len; i++){
37154              c = cols[i];
37155              totalWidth += c.width;
37156             this.headEls.push(this.headers.createChild({
37157                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37158                  cn: {
37159                      cls:'x-tree-hd-text',
37160                      html: c.header
37161                  },
37162                  style:'width:'+(c.width-this.borderWidth)+'px;'
37163              }));
37164         }
37165         this.headers.createChild({cls:'x-clear'});
37166         // prevent floats from wrapping when clipped
37167         this.headers.setWidth(totalWidth);
37168         //this.innerCt.setWidth(totalWidth);
37169         this.innerCt.setStyle({ overflow: 'auto' });
37170         this.onResize(this.width, this.height);
37171              
37172         
37173     },
37174     onResize : function(w,h)
37175     {
37176         this.height = h;
37177         this.width = w;
37178         // resize cols..
37179         this.innerCt.setWidth(this.width);
37180         this.innerCt.setHeight(this.height-20);
37181         
37182         // headers...
37183         var cols = this.columns, c;
37184         var totalWidth = 0;
37185         var expEl = false;
37186         var len = cols.length;
37187         for(var i = 0; i < len; i++){
37188             c = cols[i];
37189             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37190                 // it's the expander..
37191                 expEl  = this.headEls[i];
37192                 continue;
37193             }
37194             totalWidth += c.width;
37195             
37196         }
37197         if (expEl) {
37198             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37199         }
37200         this.headers.setWidth(w-20);
37201
37202         
37203         
37204         
37205     }
37206 });
37207 /*
37208  * Based on:
37209  * Ext JS Library 1.1.1
37210  * Copyright(c) 2006-2007, Ext JS, LLC.
37211  *
37212  * Originally Released Under LGPL - original licence link has changed is not relivant.
37213  *
37214  * Fork - LGPL
37215  * <script type="text/javascript">
37216  */
37217  
37218 /**
37219  * @class Roo.menu.Menu
37220  * @extends Roo.util.Observable
37221  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37222  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37223  * @constructor
37224  * Creates a new Menu
37225  * @param {Object} config Configuration options
37226  */
37227 Roo.menu.Menu = function(config){
37228     Roo.apply(this, config);
37229     this.id = this.id || Roo.id();
37230     this.addEvents({
37231         /**
37232          * @event beforeshow
37233          * Fires before this menu is displayed
37234          * @param {Roo.menu.Menu} this
37235          */
37236         beforeshow : true,
37237         /**
37238          * @event beforehide
37239          * Fires before this menu is hidden
37240          * @param {Roo.menu.Menu} this
37241          */
37242         beforehide : true,
37243         /**
37244          * @event show
37245          * Fires after this menu is displayed
37246          * @param {Roo.menu.Menu} this
37247          */
37248         show : true,
37249         /**
37250          * @event hide
37251          * Fires after this menu is hidden
37252          * @param {Roo.menu.Menu} this
37253          */
37254         hide : true,
37255         /**
37256          * @event click
37257          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37258          * @param {Roo.menu.Menu} this
37259          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37260          * @param {Roo.EventObject} e
37261          */
37262         click : true,
37263         /**
37264          * @event mouseover
37265          * Fires when the mouse is hovering over this menu
37266          * @param {Roo.menu.Menu} this
37267          * @param {Roo.EventObject} e
37268          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37269          */
37270         mouseover : true,
37271         /**
37272          * @event mouseout
37273          * Fires when the mouse exits this menu
37274          * @param {Roo.menu.Menu} this
37275          * @param {Roo.EventObject} e
37276          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37277          */
37278         mouseout : true,
37279         /**
37280          * @event itemclick
37281          * Fires when a menu item contained in this menu is clicked
37282          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37283          * @param {Roo.EventObject} e
37284          */
37285         itemclick: true
37286     });
37287     if (this.registerMenu) {
37288         Roo.menu.MenuMgr.register(this);
37289     }
37290     
37291     var mis = this.items;
37292     this.items = new Roo.util.MixedCollection();
37293     if(mis){
37294         this.add.apply(this, mis);
37295     }
37296 };
37297
37298 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37299     /**
37300      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37301      */
37302     minWidth : 120,
37303     /**
37304      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37305      * for bottom-right shadow (defaults to "sides")
37306      */
37307     shadow : "sides",
37308     /**
37309      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37310      * this menu (defaults to "tl-tr?")
37311      */
37312     subMenuAlign : "tl-tr?",
37313     /**
37314      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37315      * relative to its element of origin (defaults to "tl-bl?")
37316      */
37317     defaultAlign : "tl-bl?",
37318     /**
37319      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37320      */
37321     allowOtherMenus : false,
37322     /**
37323      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37324      */
37325     registerMenu : true,
37326
37327     hidden:true,
37328
37329     // private
37330     render : function(){
37331         if(this.el){
37332             return;
37333         }
37334         var el = this.el = new Roo.Layer({
37335             cls: "x-menu",
37336             shadow:this.shadow,
37337             constrain: false,
37338             parentEl: this.parentEl || document.body,
37339             zindex:15000
37340         });
37341
37342         this.keyNav = new Roo.menu.MenuNav(this);
37343
37344         if(this.plain){
37345             el.addClass("x-menu-plain");
37346         }
37347         if(this.cls){
37348             el.addClass(this.cls);
37349         }
37350         // generic focus element
37351         this.focusEl = el.createChild({
37352             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37353         });
37354         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37355         //disabling touch- as it's causing issues ..
37356         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37357         ul.on('click'   , this.onClick, this);
37358         
37359         
37360         ul.on("mouseover", this.onMouseOver, this);
37361         ul.on("mouseout", this.onMouseOut, this);
37362         this.items.each(function(item){
37363             if (item.hidden) {
37364                 return;
37365             }
37366             
37367             var li = document.createElement("li");
37368             li.className = "x-menu-list-item";
37369             ul.dom.appendChild(li);
37370             item.render(li, this);
37371         }, this);
37372         this.ul = ul;
37373         this.autoWidth();
37374     },
37375
37376     // private
37377     autoWidth : function(){
37378         var el = this.el, ul = this.ul;
37379         if(!el){
37380             return;
37381         }
37382         var w = this.width;
37383         if(w){
37384             el.setWidth(w);
37385         }else if(Roo.isIE){
37386             el.setWidth(this.minWidth);
37387             var t = el.dom.offsetWidth; // force recalc
37388             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37389         }
37390     },
37391
37392     // private
37393     delayAutoWidth : function(){
37394         if(this.rendered){
37395             if(!this.awTask){
37396                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37397             }
37398             this.awTask.delay(20);
37399         }
37400     },
37401
37402     // private
37403     findTargetItem : function(e){
37404         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37405         if(t && t.menuItemId){
37406             return this.items.get(t.menuItemId);
37407         }
37408     },
37409
37410     // private
37411     onClick : function(e){
37412         Roo.log("menu.onClick");
37413         var t = this.findTargetItem(e);
37414         if(!t){
37415             return;
37416         }
37417         Roo.log(e);
37418         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37419             if(t == this.activeItem && t.shouldDeactivate(e)){
37420                 this.activeItem.deactivate();
37421                 delete this.activeItem;
37422                 return;
37423             }
37424             if(t.canActivate){
37425                 this.setActiveItem(t, true);
37426             }
37427             return;
37428             
37429             
37430         }
37431         
37432         t.onClick(e);
37433         this.fireEvent("click", this, t, e);
37434     },
37435
37436     // private
37437     setActiveItem : function(item, autoExpand){
37438         if(item != this.activeItem){
37439             if(this.activeItem){
37440                 this.activeItem.deactivate();
37441             }
37442             this.activeItem = item;
37443             item.activate(autoExpand);
37444         }else if(autoExpand){
37445             item.expandMenu();
37446         }
37447     },
37448
37449     // private
37450     tryActivate : function(start, step){
37451         var items = this.items;
37452         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37453             var item = items.get(i);
37454             if(!item.disabled && item.canActivate){
37455                 this.setActiveItem(item, false);
37456                 return item;
37457             }
37458         }
37459         return false;
37460     },
37461
37462     // private
37463     onMouseOver : function(e){
37464         var t;
37465         if(t = this.findTargetItem(e)){
37466             if(t.canActivate && !t.disabled){
37467                 this.setActiveItem(t, true);
37468             }
37469         }
37470         this.fireEvent("mouseover", this, e, t);
37471     },
37472
37473     // private
37474     onMouseOut : function(e){
37475         var t;
37476         if(t = this.findTargetItem(e)){
37477             if(t == this.activeItem && t.shouldDeactivate(e)){
37478                 this.activeItem.deactivate();
37479                 delete this.activeItem;
37480             }
37481         }
37482         this.fireEvent("mouseout", this, e, t);
37483     },
37484
37485     /**
37486      * Read-only.  Returns true if the menu is currently displayed, else false.
37487      * @type Boolean
37488      */
37489     isVisible : function(){
37490         return this.el && !this.hidden;
37491     },
37492
37493     /**
37494      * Displays this menu relative to another element
37495      * @param {String/HTMLElement/Roo.Element} element The element to align to
37496      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37497      * the element (defaults to this.defaultAlign)
37498      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37499      */
37500     show : function(el, pos, parentMenu){
37501         this.parentMenu = parentMenu;
37502         if(!this.el){
37503             this.render();
37504         }
37505         this.fireEvent("beforeshow", this);
37506         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37507     },
37508
37509     /**
37510      * Displays this menu at a specific xy position
37511      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37512      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37513      */
37514     showAt : function(xy, parentMenu, /* private: */_e){
37515         this.parentMenu = parentMenu;
37516         if(!this.el){
37517             this.render();
37518         }
37519         if(_e !== false){
37520             this.fireEvent("beforeshow", this);
37521             xy = this.el.adjustForConstraints(xy);
37522         }
37523         this.el.setXY(xy);
37524         this.el.show();
37525         this.hidden = false;
37526         this.focus();
37527         this.fireEvent("show", this);
37528     },
37529
37530     focus : function(){
37531         if(!this.hidden){
37532             this.doFocus.defer(50, this);
37533         }
37534     },
37535
37536     doFocus : function(){
37537         if(!this.hidden){
37538             this.focusEl.focus();
37539         }
37540     },
37541
37542     /**
37543      * Hides this menu and optionally all parent menus
37544      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37545      */
37546     hide : function(deep){
37547         if(this.el && this.isVisible()){
37548             this.fireEvent("beforehide", this);
37549             if(this.activeItem){
37550                 this.activeItem.deactivate();
37551                 this.activeItem = null;
37552             }
37553             this.el.hide();
37554             this.hidden = true;
37555             this.fireEvent("hide", this);
37556         }
37557         if(deep === true && this.parentMenu){
37558             this.parentMenu.hide(true);
37559         }
37560     },
37561
37562     /**
37563      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37564      * Any of the following are valid:
37565      * <ul>
37566      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37567      * <li>An HTMLElement object which will be converted to a menu item</li>
37568      * <li>A menu item config object that will be created as a new menu item</li>
37569      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37570      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37571      * </ul>
37572      * Usage:
37573      * <pre><code>
37574 // Create the menu
37575 var menu = new Roo.menu.Menu();
37576
37577 // Create a menu item to add by reference
37578 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37579
37580 // Add a bunch of items at once using different methods.
37581 // Only the last item added will be returned.
37582 var item = menu.add(
37583     menuItem,                // add existing item by ref
37584     'Dynamic Item',          // new TextItem
37585     '-',                     // new separator
37586     { text: 'Config Item' }  // new item by config
37587 );
37588 </code></pre>
37589      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37590      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37591      */
37592     add : function(){
37593         var a = arguments, l = a.length, item;
37594         for(var i = 0; i < l; i++){
37595             var el = a[i];
37596             if ((typeof(el) == "object") && el.xtype && el.xns) {
37597                 el = Roo.factory(el, Roo.menu);
37598             }
37599             
37600             if(el.render){ // some kind of Item
37601                 item = this.addItem(el);
37602             }else if(typeof el == "string"){ // string
37603                 if(el == "separator" || el == "-"){
37604                     item = this.addSeparator();
37605                 }else{
37606                     item = this.addText(el);
37607                 }
37608             }else if(el.tagName || el.el){ // element
37609                 item = this.addElement(el);
37610             }else if(typeof el == "object"){ // must be menu item config?
37611                 item = this.addMenuItem(el);
37612             }
37613         }
37614         return item;
37615     },
37616
37617     /**
37618      * Returns this menu's underlying {@link Roo.Element} object
37619      * @return {Roo.Element} The element
37620      */
37621     getEl : function(){
37622         if(!this.el){
37623             this.render();
37624         }
37625         return this.el;
37626     },
37627
37628     /**
37629      * Adds a separator bar to the menu
37630      * @return {Roo.menu.Item} The menu item that was added
37631      */
37632     addSeparator : function(){
37633         return this.addItem(new Roo.menu.Separator());
37634     },
37635
37636     /**
37637      * Adds an {@link Roo.Element} object to the menu
37638      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37639      * @return {Roo.menu.Item} The menu item that was added
37640      */
37641     addElement : function(el){
37642         return this.addItem(new Roo.menu.BaseItem(el));
37643     },
37644
37645     /**
37646      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37647      * @param {Roo.menu.Item} item The menu item to add
37648      * @return {Roo.menu.Item} The menu item that was added
37649      */
37650     addItem : function(item){
37651         this.items.add(item);
37652         if(this.ul){
37653             var li = document.createElement("li");
37654             li.className = "x-menu-list-item";
37655             this.ul.dom.appendChild(li);
37656             item.render(li, this);
37657             this.delayAutoWidth();
37658         }
37659         return item;
37660     },
37661
37662     /**
37663      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37664      * @param {Object} config A MenuItem config object
37665      * @return {Roo.menu.Item} The menu item that was added
37666      */
37667     addMenuItem : function(config){
37668         if(!(config instanceof Roo.menu.Item)){
37669             if(typeof config.checked == "boolean"){ // must be check menu item config?
37670                 config = new Roo.menu.CheckItem(config);
37671             }else{
37672                 config = new Roo.menu.Item(config);
37673             }
37674         }
37675         return this.addItem(config);
37676     },
37677
37678     /**
37679      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37680      * @param {String} text The text to display in the menu item
37681      * @return {Roo.menu.Item} The menu item that was added
37682      */
37683     addText : function(text){
37684         return this.addItem(new Roo.menu.TextItem({ text : text }));
37685     },
37686
37687     /**
37688      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37689      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37690      * @param {Roo.menu.Item} item The menu item to add
37691      * @return {Roo.menu.Item} The menu item that was added
37692      */
37693     insert : function(index, item){
37694         this.items.insert(index, item);
37695         if(this.ul){
37696             var li = document.createElement("li");
37697             li.className = "x-menu-list-item";
37698             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37699             item.render(li, this);
37700             this.delayAutoWidth();
37701         }
37702         return item;
37703     },
37704
37705     /**
37706      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37707      * @param {Roo.menu.Item} item The menu item to remove
37708      */
37709     remove : function(item){
37710         this.items.removeKey(item.id);
37711         item.destroy();
37712     },
37713
37714     /**
37715      * Removes and destroys all items in the menu
37716      */
37717     removeAll : function(){
37718         var f;
37719         while(f = this.items.first()){
37720             this.remove(f);
37721         }
37722     }
37723 });
37724
37725 // MenuNav is a private utility class used internally by the Menu
37726 Roo.menu.MenuNav = function(menu){
37727     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37728     this.scope = this.menu = menu;
37729 };
37730
37731 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37732     doRelay : function(e, h){
37733         var k = e.getKey();
37734         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37735             this.menu.tryActivate(0, 1);
37736             return false;
37737         }
37738         return h.call(this.scope || this, e, this.menu);
37739     },
37740
37741     up : function(e, m){
37742         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37743             m.tryActivate(m.items.length-1, -1);
37744         }
37745     },
37746
37747     down : function(e, m){
37748         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37749             m.tryActivate(0, 1);
37750         }
37751     },
37752
37753     right : function(e, m){
37754         if(m.activeItem){
37755             m.activeItem.expandMenu(true);
37756         }
37757     },
37758
37759     left : function(e, m){
37760         m.hide();
37761         if(m.parentMenu && m.parentMenu.activeItem){
37762             m.parentMenu.activeItem.activate();
37763         }
37764     },
37765
37766     enter : function(e, m){
37767         if(m.activeItem){
37768             e.stopPropagation();
37769             m.activeItem.onClick(e);
37770             m.fireEvent("click", this, m.activeItem);
37771             return true;
37772         }
37773     }
37774 });/*
37775  * Based on:
37776  * Ext JS Library 1.1.1
37777  * Copyright(c) 2006-2007, Ext JS, LLC.
37778  *
37779  * Originally Released Under LGPL - original licence link has changed is not relivant.
37780  *
37781  * Fork - LGPL
37782  * <script type="text/javascript">
37783  */
37784  
37785 /**
37786  * @class Roo.menu.MenuMgr
37787  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37788  * @singleton
37789  */
37790 Roo.menu.MenuMgr = function(){
37791    var menus, active, groups = {}, attached = false, lastShow = new Date();
37792
37793    // private - called when first menu is created
37794    function init(){
37795        menus = {};
37796        active = new Roo.util.MixedCollection();
37797        Roo.get(document).addKeyListener(27, function(){
37798            if(active.length > 0){
37799                hideAll();
37800            }
37801        });
37802    }
37803
37804    // private
37805    function hideAll(){
37806        if(active && active.length > 0){
37807            var c = active.clone();
37808            c.each(function(m){
37809                m.hide();
37810            });
37811        }
37812    }
37813
37814    // private
37815    function onHide(m){
37816        active.remove(m);
37817        if(active.length < 1){
37818            Roo.get(document).un("mousedown", onMouseDown);
37819            attached = false;
37820        }
37821    }
37822
37823    // private
37824    function onShow(m){
37825        var last = active.last();
37826        lastShow = new Date();
37827        active.add(m);
37828        if(!attached){
37829            Roo.get(document).on("mousedown", onMouseDown);
37830            attached = true;
37831        }
37832        if(m.parentMenu){
37833           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37834           m.parentMenu.activeChild = m;
37835        }else if(last && last.isVisible()){
37836           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37837        }
37838    }
37839
37840    // private
37841    function onBeforeHide(m){
37842        if(m.activeChild){
37843            m.activeChild.hide();
37844        }
37845        if(m.autoHideTimer){
37846            clearTimeout(m.autoHideTimer);
37847            delete m.autoHideTimer;
37848        }
37849    }
37850
37851    // private
37852    function onBeforeShow(m){
37853        var pm = m.parentMenu;
37854        if(!pm && !m.allowOtherMenus){
37855            hideAll();
37856        }else if(pm && pm.activeChild && active != m){
37857            pm.activeChild.hide();
37858        }
37859    }
37860
37861    // private
37862    function onMouseDown(e){
37863        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37864            hideAll();
37865        }
37866    }
37867
37868    // private
37869    function onBeforeCheck(mi, state){
37870        if(state){
37871            var g = groups[mi.group];
37872            for(var i = 0, l = g.length; i < l; i++){
37873                if(g[i] != mi){
37874                    g[i].setChecked(false);
37875                }
37876            }
37877        }
37878    }
37879
37880    return {
37881
37882        /**
37883         * Hides all menus that are currently visible
37884         */
37885        hideAll : function(){
37886             hideAll();  
37887        },
37888
37889        // private
37890        register : function(menu){
37891            if(!menus){
37892                init();
37893            }
37894            menus[menu.id] = menu;
37895            menu.on("beforehide", onBeforeHide);
37896            menu.on("hide", onHide);
37897            menu.on("beforeshow", onBeforeShow);
37898            menu.on("show", onShow);
37899            var g = menu.group;
37900            if(g && menu.events["checkchange"]){
37901                if(!groups[g]){
37902                    groups[g] = [];
37903                }
37904                groups[g].push(menu);
37905                menu.on("checkchange", onCheck);
37906            }
37907        },
37908
37909         /**
37910          * Returns a {@link Roo.menu.Menu} object
37911          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37912          * be used to generate and return a new Menu instance.
37913          */
37914        get : function(menu){
37915            if(typeof menu == "string"){ // menu id
37916                return menus[menu];
37917            }else if(menu.events){  // menu instance
37918                return menu;
37919            }else if(typeof menu.length == 'number'){ // array of menu items?
37920                return new Roo.menu.Menu({items:menu});
37921            }else{ // otherwise, must be a config
37922                return new Roo.menu.Menu(menu);
37923            }
37924        },
37925
37926        // private
37927        unregister : function(menu){
37928            delete menus[menu.id];
37929            menu.un("beforehide", onBeforeHide);
37930            menu.un("hide", onHide);
37931            menu.un("beforeshow", onBeforeShow);
37932            menu.un("show", onShow);
37933            var g = menu.group;
37934            if(g && menu.events["checkchange"]){
37935                groups[g].remove(menu);
37936                menu.un("checkchange", onCheck);
37937            }
37938        },
37939
37940        // private
37941        registerCheckable : function(menuItem){
37942            var g = menuItem.group;
37943            if(g){
37944                if(!groups[g]){
37945                    groups[g] = [];
37946                }
37947                groups[g].push(menuItem);
37948                menuItem.on("beforecheckchange", onBeforeCheck);
37949            }
37950        },
37951
37952        // private
37953        unregisterCheckable : function(menuItem){
37954            var g = menuItem.group;
37955            if(g){
37956                groups[g].remove(menuItem);
37957                menuItem.un("beforecheckchange", onBeforeCheck);
37958            }
37959        }
37960    };
37961 }();/*
37962  * Based on:
37963  * Ext JS Library 1.1.1
37964  * Copyright(c) 2006-2007, Ext JS, LLC.
37965  *
37966  * Originally Released Under LGPL - original licence link has changed is not relivant.
37967  *
37968  * Fork - LGPL
37969  * <script type="text/javascript">
37970  */
37971  
37972
37973 /**
37974  * @class Roo.menu.BaseItem
37975  * @extends Roo.Component
37976  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
37977  * management and base configuration options shared by all menu components.
37978  * @constructor
37979  * Creates a new BaseItem
37980  * @param {Object} config Configuration options
37981  */
37982 Roo.menu.BaseItem = function(config){
37983     Roo.menu.BaseItem.superclass.constructor.call(this, config);
37984
37985     this.addEvents({
37986         /**
37987          * @event click
37988          * Fires when this item is clicked
37989          * @param {Roo.menu.BaseItem} this
37990          * @param {Roo.EventObject} e
37991          */
37992         click: true,
37993         /**
37994          * @event activate
37995          * Fires when this item is activated
37996          * @param {Roo.menu.BaseItem} this
37997          */
37998         activate : true,
37999         /**
38000          * @event deactivate
38001          * Fires when this item is deactivated
38002          * @param {Roo.menu.BaseItem} this
38003          */
38004         deactivate : true
38005     });
38006
38007     if(this.handler){
38008         this.on("click", this.handler, this.scope, true);
38009     }
38010 };
38011
38012 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38013     /**
38014      * @cfg {Function} handler
38015      * A function that will handle the click event of this menu item (defaults to undefined)
38016      */
38017     /**
38018      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38019      */
38020     canActivate : false,
38021     
38022      /**
38023      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38024      */
38025     hidden: false,
38026     
38027     /**
38028      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38029      */
38030     activeClass : "x-menu-item-active",
38031     /**
38032      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38033      */
38034     hideOnClick : true,
38035     /**
38036      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38037      */
38038     hideDelay : 100,
38039
38040     // private
38041     ctype: "Roo.menu.BaseItem",
38042
38043     // private
38044     actionMode : "container",
38045
38046     // private
38047     render : function(container, parentMenu){
38048         this.parentMenu = parentMenu;
38049         Roo.menu.BaseItem.superclass.render.call(this, container);
38050         this.container.menuItemId = this.id;
38051     },
38052
38053     // private
38054     onRender : function(container, position){
38055         this.el = Roo.get(this.el);
38056         container.dom.appendChild(this.el.dom);
38057     },
38058
38059     // private
38060     onClick : function(e){
38061         if(!this.disabled && this.fireEvent("click", this, e) !== false
38062                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38063             this.handleClick(e);
38064         }else{
38065             e.stopEvent();
38066         }
38067     },
38068
38069     // private
38070     activate : function(){
38071         if(this.disabled){
38072             return false;
38073         }
38074         var li = this.container;
38075         li.addClass(this.activeClass);
38076         this.region = li.getRegion().adjust(2, 2, -2, -2);
38077         this.fireEvent("activate", this);
38078         return true;
38079     },
38080
38081     // private
38082     deactivate : function(){
38083         this.container.removeClass(this.activeClass);
38084         this.fireEvent("deactivate", this);
38085     },
38086
38087     // private
38088     shouldDeactivate : function(e){
38089         return !this.region || !this.region.contains(e.getPoint());
38090     },
38091
38092     // private
38093     handleClick : function(e){
38094         if(this.hideOnClick){
38095             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38096         }
38097     },
38098
38099     // private
38100     expandMenu : function(autoActivate){
38101         // do nothing
38102     },
38103
38104     // private
38105     hideMenu : function(){
38106         // do nothing
38107     }
38108 });/*
38109  * Based on:
38110  * Ext JS Library 1.1.1
38111  * Copyright(c) 2006-2007, Ext JS, LLC.
38112  *
38113  * Originally Released Under LGPL - original licence link has changed is not relivant.
38114  *
38115  * Fork - LGPL
38116  * <script type="text/javascript">
38117  */
38118  
38119 /**
38120  * @class Roo.menu.Adapter
38121  * @extends Roo.menu.BaseItem
38122  * A base utility class that adapts a non-menu component so that it can be wrapped by a menu item and added to a menu.
38123  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38124  * @constructor
38125  * Creates a new Adapter
38126  * @param {Object} config Configuration options
38127  */
38128 Roo.menu.Adapter = function(component, config){
38129     Roo.menu.Adapter.superclass.constructor.call(this, config);
38130     this.component = component;
38131 };
38132 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38133     // private
38134     canActivate : true,
38135
38136     // private
38137     onRender : function(container, position){
38138         this.component.render(container);
38139         this.el = this.component.getEl();
38140     },
38141
38142     // private
38143     activate : function(){
38144         if(this.disabled){
38145             return false;
38146         }
38147         this.component.focus();
38148         this.fireEvent("activate", this);
38149         return true;
38150     },
38151
38152     // private
38153     deactivate : function(){
38154         this.fireEvent("deactivate", this);
38155     },
38156
38157     // private
38158     disable : function(){
38159         this.component.disable();
38160         Roo.menu.Adapter.superclass.disable.call(this);
38161     },
38162
38163     // private
38164     enable : function(){
38165         this.component.enable();
38166         Roo.menu.Adapter.superclass.enable.call(this);
38167     }
38168 });/*
38169  * Based on:
38170  * Ext JS Library 1.1.1
38171  * Copyright(c) 2006-2007, Ext JS, LLC.
38172  *
38173  * Originally Released Under LGPL - original licence link has changed is not relivant.
38174  *
38175  * Fork - LGPL
38176  * <script type="text/javascript">
38177  */
38178
38179 /**
38180  * @class Roo.menu.TextItem
38181  * @extends Roo.menu.BaseItem
38182  * Adds a static text string to a menu, usually used as either a heading or group separator.
38183  * Note: old style constructor with text is still supported.
38184  * 
38185  * @constructor
38186  * Creates a new TextItem
38187  * @param {Object} cfg Configuration
38188  */
38189 Roo.menu.TextItem = function(cfg){
38190     if (typeof(cfg) == 'string') {
38191         this.text = cfg;
38192     } else {
38193         Roo.apply(this,cfg);
38194     }
38195     
38196     Roo.menu.TextItem.superclass.constructor.call(this);
38197 };
38198
38199 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38200     /**
38201      * @cfg {Boolean} text Text to show on item.
38202      */
38203     text : '',
38204     
38205     /**
38206      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38207      */
38208     hideOnClick : false,
38209     /**
38210      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38211      */
38212     itemCls : "x-menu-text",
38213
38214     // private
38215     onRender : function(){
38216         var s = document.createElement("span");
38217         s.className = this.itemCls;
38218         s.innerHTML = this.text;
38219         this.el = s;
38220         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38221     }
38222 });/*
38223  * Based on:
38224  * Ext JS Library 1.1.1
38225  * Copyright(c) 2006-2007, Ext JS, LLC.
38226  *
38227  * Originally Released Under LGPL - original licence link has changed is not relivant.
38228  *
38229  * Fork - LGPL
38230  * <script type="text/javascript">
38231  */
38232
38233 /**
38234  * @class Roo.menu.Separator
38235  * @extends Roo.menu.BaseItem
38236  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38237  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38238  * @constructor
38239  * @param {Object} config Configuration options
38240  */
38241 Roo.menu.Separator = function(config){
38242     Roo.menu.Separator.superclass.constructor.call(this, config);
38243 };
38244
38245 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38246     /**
38247      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38248      */
38249     itemCls : "x-menu-sep",
38250     /**
38251      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38252      */
38253     hideOnClick : false,
38254
38255     // private
38256     onRender : function(li){
38257         var s = document.createElement("span");
38258         s.className = this.itemCls;
38259         s.innerHTML = "&#160;";
38260         this.el = s;
38261         li.addClass("x-menu-sep-li");
38262         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38263     }
38264 });/*
38265  * Based on:
38266  * Ext JS Library 1.1.1
38267  * Copyright(c) 2006-2007, Ext JS, LLC.
38268  *
38269  * Originally Released Under LGPL - original licence link has changed is not relivant.
38270  *
38271  * Fork - LGPL
38272  * <script type="text/javascript">
38273  */
38274 /**
38275  * @class Roo.menu.Item
38276  * @extends Roo.menu.BaseItem
38277  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38278  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38279  * activation and click handling.
38280  * @constructor
38281  * Creates a new Item
38282  * @param {Object} config Configuration options
38283  */
38284 Roo.menu.Item = function(config){
38285     Roo.menu.Item.superclass.constructor.call(this, config);
38286     if(this.menu){
38287         this.menu = Roo.menu.MenuMgr.get(this.menu);
38288     }
38289 };
38290 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38291     
38292     /**
38293      * @cfg {String} text
38294      * The text to show on the menu item.
38295      */
38296     text: '',
38297      /**
38298      * @cfg {String} HTML to render in menu
38299      * The text to show on the menu item (HTML version).
38300      */
38301     html: '',
38302     /**
38303      * @cfg {String} icon
38304      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38305      */
38306     icon: undefined,
38307     /**
38308      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38309      */
38310     itemCls : "x-menu-item",
38311     /**
38312      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38313      */
38314     canActivate : true,
38315     /**
38316      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38317      */
38318     showDelay: 200,
38319     // doc'd in BaseItem
38320     hideDelay: 200,
38321
38322     // private
38323     ctype: "Roo.menu.Item",
38324     
38325     // private
38326     onRender : function(container, position){
38327         var el = document.createElement("a");
38328         el.hideFocus = true;
38329         el.unselectable = "on";
38330         el.href = this.href || "#";
38331         if(this.hrefTarget){
38332             el.target = this.hrefTarget;
38333         }
38334         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38335         
38336         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38337         
38338         el.innerHTML = String.format(
38339                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38340                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38341         this.el = el;
38342         Roo.menu.Item.superclass.onRender.call(this, container, position);
38343     },
38344
38345     /**
38346      * Sets the text to display in this menu item
38347      * @param {String} text The text to display
38348      * @param {Boolean} isHTML true to indicate text is pure html.
38349      */
38350     setText : function(text, isHTML){
38351         if (isHTML) {
38352             this.html = text;
38353         } else {
38354             this.text = text;
38355             this.html = '';
38356         }
38357         if(this.rendered){
38358             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38359      
38360             this.el.update(String.format(
38361                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38362                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38363             this.parentMenu.autoWidth();
38364         }
38365     },
38366
38367     // private
38368     handleClick : function(e){
38369         if(!this.href){ // if no link defined, stop the event automatically
38370             e.stopEvent();
38371         }
38372         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38373     },
38374
38375     // private
38376     activate : function(autoExpand){
38377         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38378             this.focus();
38379             if(autoExpand){
38380                 this.expandMenu();
38381             }
38382         }
38383         return true;
38384     },
38385
38386     // private
38387     shouldDeactivate : function(e){
38388         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38389             if(this.menu && this.menu.isVisible()){
38390                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38391             }
38392             return true;
38393         }
38394         return false;
38395     },
38396
38397     // private
38398     deactivate : function(){
38399         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38400         this.hideMenu();
38401     },
38402
38403     // private
38404     expandMenu : function(autoActivate){
38405         if(!this.disabled && this.menu){
38406             clearTimeout(this.hideTimer);
38407             delete this.hideTimer;
38408             if(!this.menu.isVisible() && !this.showTimer){
38409                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38410             }else if (this.menu.isVisible() && autoActivate){
38411                 this.menu.tryActivate(0, 1);
38412             }
38413         }
38414     },
38415
38416     // private
38417     deferExpand : function(autoActivate){
38418         delete this.showTimer;
38419         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38420         if(autoActivate){
38421             this.menu.tryActivate(0, 1);
38422         }
38423     },
38424
38425     // private
38426     hideMenu : function(){
38427         clearTimeout(this.showTimer);
38428         delete this.showTimer;
38429         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38430             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38431         }
38432     },
38433
38434     // private
38435     deferHide : function(){
38436         delete this.hideTimer;
38437         this.menu.hide();
38438     }
38439 });/*
38440  * Based on:
38441  * Ext JS Library 1.1.1
38442  * Copyright(c) 2006-2007, Ext JS, LLC.
38443  *
38444  * Originally Released Under LGPL - original licence link has changed is not relivant.
38445  *
38446  * Fork - LGPL
38447  * <script type="text/javascript">
38448  */
38449  
38450 /**
38451  * @class Roo.menu.CheckItem
38452  * @extends Roo.menu.Item
38453  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38454  * @constructor
38455  * Creates a new CheckItem
38456  * @param {Object} config Configuration options
38457  */
38458 Roo.menu.CheckItem = function(config){
38459     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38460     this.addEvents({
38461         /**
38462          * @event beforecheckchange
38463          * Fires before the checked value is set, providing an opportunity to cancel if needed
38464          * @param {Roo.menu.CheckItem} this
38465          * @param {Boolean} checked The new checked value that will be set
38466          */
38467         "beforecheckchange" : true,
38468         /**
38469          * @event checkchange
38470          * Fires after the checked value has been set
38471          * @param {Roo.menu.CheckItem} this
38472          * @param {Boolean} checked The checked value that was set
38473          */
38474         "checkchange" : true
38475     });
38476     if(this.checkHandler){
38477         this.on('checkchange', this.checkHandler, this.scope);
38478     }
38479 };
38480 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38481     /**
38482      * @cfg {String} group
38483      * All check items with the same group name will automatically be grouped into a single-select
38484      * radio button group (defaults to '')
38485      */
38486     /**
38487      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38488      */
38489     itemCls : "x-menu-item x-menu-check-item",
38490     /**
38491      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38492      */
38493     groupClass : "x-menu-group-item",
38494
38495     /**
38496      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38497      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38498      * initialized with checked = true will be rendered as checked.
38499      */
38500     checked: false,
38501
38502     // private
38503     ctype: "Roo.menu.CheckItem",
38504
38505     // private
38506     onRender : function(c){
38507         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38508         if(this.group){
38509             this.el.addClass(this.groupClass);
38510         }
38511         Roo.menu.MenuMgr.registerCheckable(this);
38512         if(this.checked){
38513             this.checked = false;
38514             this.setChecked(true, true);
38515         }
38516     },
38517
38518     // private
38519     destroy : function(){
38520         if(this.rendered){
38521             Roo.menu.MenuMgr.unregisterCheckable(this);
38522         }
38523         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38524     },
38525
38526     /**
38527      * Set the checked state of this item
38528      * @param {Boolean} checked The new checked value
38529      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38530      */
38531     setChecked : function(state, suppressEvent){
38532         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38533             if(this.container){
38534                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38535             }
38536             this.checked = state;
38537             if(suppressEvent !== true){
38538                 this.fireEvent("checkchange", this, state);
38539             }
38540         }
38541     },
38542
38543     // private
38544     handleClick : function(e){
38545        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38546            this.setChecked(!this.checked);
38547        }
38548        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38549     }
38550 });/*
38551  * Based on:
38552  * Ext JS Library 1.1.1
38553  * Copyright(c) 2006-2007, Ext JS, LLC.
38554  *
38555  * Originally Released Under LGPL - original licence link has changed is not relivant.
38556  *
38557  * Fork - LGPL
38558  * <script type="text/javascript">
38559  */
38560  
38561 /**
38562  * @class Roo.menu.DateItem
38563  * @extends Roo.menu.Adapter
38564  * A menu item that wraps the {@link Roo.DatPicker} component.
38565  * @constructor
38566  * Creates a new DateItem
38567  * @param {Object} config Configuration options
38568  */
38569 Roo.menu.DateItem = function(config){
38570     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38571     /** The Roo.DatePicker object @type Roo.DatePicker */
38572     this.picker = this.component;
38573     this.addEvents({select: true});
38574     
38575     this.picker.on("render", function(picker){
38576         picker.getEl().swallowEvent("click");
38577         picker.container.addClass("x-menu-date-item");
38578     });
38579
38580     this.picker.on("select", this.onSelect, this);
38581 };
38582
38583 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38584     // private
38585     onSelect : function(picker, date){
38586         this.fireEvent("select", this, date, picker);
38587         Roo.menu.DateItem.superclass.handleClick.call(this);
38588     }
38589 });/*
38590  * Based on:
38591  * Ext JS Library 1.1.1
38592  * Copyright(c) 2006-2007, Ext JS, LLC.
38593  *
38594  * Originally Released Under LGPL - original licence link has changed is not relivant.
38595  *
38596  * Fork - LGPL
38597  * <script type="text/javascript">
38598  */
38599  
38600 /**
38601  * @class Roo.menu.ColorItem
38602  * @extends Roo.menu.Adapter
38603  * A menu item that wraps the {@link Roo.ColorPalette} component.
38604  * @constructor
38605  * Creates a new ColorItem
38606  * @param {Object} config Configuration options
38607  */
38608 Roo.menu.ColorItem = function(config){
38609     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38610     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38611     this.palette = this.component;
38612     this.relayEvents(this.palette, ["select"]);
38613     if(this.selectHandler){
38614         this.on('select', this.selectHandler, this.scope);
38615     }
38616 };
38617 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38618  * Based on:
38619  * Ext JS Library 1.1.1
38620  * Copyright(c) 2006-2007, Ext JS, LLC.
38621  *
38622  * Originally Released Under LGPL - original licence link has changed is not relivant.
38623  *
38624  * Fork - LGPL
38625  * <script type="text/javascript">
38626  */
38627  
38628
38629 /**
38630  * @class Roo.menu.DateMenu
38631  * @extends Roo.menu.Menu
38632  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38633  * @constructor
38634  * Creates a new DateMenu
38635  * @param {Object} config Configuration options
38636  */
38637 Roo.menu.DateMenu = function(config){
38638     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38639     this.plain = true;
38640     var di = new Roo.menu.DateItem(config);
38641     this.add(di);
38642     /**
38643      * The {@link Roo.DatePicker} instance for this DateMenu
38644      * @type DatePicker
38645      */
38646     this.picker = di.picker;
38647     /**
38648      * @event select
38649      * @param {DatePicker} picker
38650      * @param {Date} date
38651      */
38652     this.relayEvents(di, ["select"]);
38653     this.on('beforeshow', function(){
38654         if(this.picker){
38655             this.picker.hideMonthPicker(false);
38656         }
38657     }, this);
38658 };
38659 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38660     cls:'x-date-menu'
38661 });/*
38662  * Based on:
38663  * Ext JS Library 1.1.1
38664  * Copyright(c) 2006-2007, Ext JS, LLC.
38665  *
38666  * Originally Released Under LGPL - original licence link has changed is not relivant.
38667  *
38668  * Fork - LGPL
38669  * <script type="text/javascript">
38670  */
38671  
38672
38673 /**
38674  * @class Roo.menu.ColorMenu
38675  * @extends Roo.menu.Menu
38676  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38677  * @constructor
38678  * Creates a new ColorMenu
38679  * @param {Object} config Configuration options
38680  */
38681 Roo.menu.ColorMenu = function(config){
38682     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38683     this.plain = true;
38684     var ci = new Roo.menu.ColorItem(config);
38685     this.add(ci);
38686     /**
38687      * The {@link Roo.ColorPalette} instance for this ColorMenu
38688      * @type ColorPalette
38689      */
38690     this.palette = ci.palette;
38691     /**
38692      * @event select
38693      * @param {ColorPalette} palette
38694      * @param {String} color
38695      */
38696     this.relayEvents(ci, ["select"]);
38697 };
38698 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38699  * Based on:
38700  * Ext JS Library 1.1.1
38701  * Copyright(c) 2006-2007, Ext JS, LLC.
38702  *
38703  * Originally Released Under LGPL - original licence link has changed is not relivant.
38704  *
38705  * Fork - LGPL
38706  * <script type="text/javascript">
38707  */
38708  
38709 /**
38710  * @class Roo.form.Field
38711  * @extends Roo.BoxComponent
38712  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38713  * @constructor
38714  * Creates a new Field
38715  * @param {Object} config Configuration options
38716  */
38717 Roo.form.Field = function(config){
38718     Roo.form.Field.superclass.constructor.call(this, config);
38719 };
38720
38721 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38722     /**
38723      * @cfg {String} fieldLabel Label to use when rendering a form.
38724      */
38725        /**
38726      * @cfg {String} qtip Mouse over tip
38727      */
38728      
38729     /**
38730      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38731      */
38732     invalidClass : "x-form-invalid",
38733     /**
38734      * @cfg {String} invalidText The error text to use when marking a field invalid and no message is provided (defaults to "The value in this field is invalid")
38735      */
38736     invalidText : "The value in this field is invalid",
38737     /**
38738      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38739      */
38740     focusClass : "x-form-focus",
38741     /**
38742      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38743       automatic validation (defaults to "keyup").
38744      */
38745     validationEvent : "keyup",
38746     /**
38747      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38748      */
38749     validateOnBlur : true,
38750     /**
38751      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38752      */
38753     validationDelay : 250,
38754     /**
38755      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38756      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38757      */
38758     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38759     /**
38760      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38761      */
38762     fieldClass : "x-form-field",
38763     /**
38764      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38765      *<pre>
38766 Value         Description
38767 -----------   ----------------------------------------------------------------------
38768 qtip          Display a quick tip when the user hovers over the field
38769 title         Display a default browser title attribute popup
38770 under         Add a block div beneath the field containing the error text
38771 side          Add an error icon to the right of the field with a popup on hover
38772 [element id]  Add the error text directly to the innerHTML of the specified element
38773 </pre>
38774      */
38775     msgTarget : 'qtip',
38776     /**
38777      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38778      */
38779     msgFx : 'normal',
38780
38781     /**
38782      * @cfg {Boolean} readOnly True to mark the field as readOnly in HTML (defaults to false) -- Note: this only sets the element's readOnly DOM attribute.
38783      */
38784     readOnly : false,
38785
38786     /**
38787      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38788      */
38789     disabled : false,
38790
38791     /**
38792      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38793      */
38794     inputType : undefined,
38795     
38796     /**
38797      * @cfg {Number} tabIndex The tabIndex for this field. Note this only applies to fields that are rendered, not those which are built via applyTo (defaults to undefined).
38798          */
38799         tabIndex : undefined,
38800         
38801     // private
38802     isFormField : true,
38803
38804     // private
38805     hasFocus : false,
38806     /**
38807      * @property {Roo.Element} fieldEl
38808      * Element Containing the rendered Field (with label etc.)
38809      */
38810     /**
38811      * @cfg {Mixed} value A value to initialize this field with.
38812      */
38813     value : undefined,
38814
38815     /**
38816      * @cfg {String} name The field's HTML name attribute.
38817      */
38818     /**
38819      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38820      */
38821     // private
38822     loadedValue : false,
38823      
38824      
38825         // private ??
38826         initComponent : function(){
38827         Roo.form.Field.superclass.initComponent.call(this);
38828         this.addEvents({
38829             /**
38830              * @event focus
38831              * Fires when this field receives input focus.
38832              * @param {Roo.form.Field} this
38833              */
38834             focus : true,
38835             /**
38836              * @event blur
38837              * Fires when this field loses input focus.
38838              * @param {Roo.form.Field} this
38839              */
38840             blur : true,
38841             /**
38842              * @event specialkey
38843              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38844              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38845              * @param {Roo.form.Field} this
38846              * @param {Roo.EventObject} e The event object
38847              */
38848             specialkey : true,
38849             /**
38850              * @event change
38851              * Fires just before the field blurs if the field value has changed.
38852              * @param {Roo.form.Field} this
38853              * @param {Mixed} newValue The new value
38854              * @param {Mixed} oldValue The original value
38855              */
38856             change : true,
38857             /**
38858              * @event invalid
38859              * Fires after the field has been marked as invalid.
38860              * @param {Roo.form.Field} this
38861              * @param {String} msg The validation message
38862              */
38863             invalid : true,
38864             /**
38865              * @event valid
38866              * Fires after the field has been validated with no errors.
38867              * @param {Roo.form.Field} this
38868              */
38869             valid : true,
38870              /**
38871              * @event keyup
38872              * Fires after the key up
38873              * @param {Roo.form.Field} this
38874              * @param {Roo.EventObject}  e The event Object
38875              */
38876             keyup : true
38877         });
38878     },
38879
38880     /**
38881      * Returns the name attribute of the field if available
38882      * @return {String} name The field name
38883      */
38884     getName: function(){
38885          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
38886     },
38887
38888     // private
38889     onRender : function(ct, position){
38890         Roo.form.Field.superclass.onRender.call(this, ct, position);
38891         if(!this.el){
38892             var cfg = this.getAutoCreate();
38893             if(!cfg.name){
38894                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38895             }
38896             if (!cfg.name.length) {
38897                 delete cfg.name;
38898             }
38899             if(this.inputType){
38900                 cfg.type = this.inputType;
38901             }
38902             this.el = ct.createChild(cfg, position);
38903         }
38904         var type = this.el.dom.type;
38905         if(type){
38906             if(type == 'password'){
38907                 type = 'text';
38908             }
38909             this.el.addClass('x-form-'+type);
38910         }
38911         if(this.readOnly){
38912             this.el.dom.readOnly = true;
38913         }
38914         if(this.tabIndex !== undefined){
38915             this.el.dom.setAttribute('tabIndex', this.tabIndex);
38916         }
38917
38918         this.el.addClass([this.fieldClass, this.cls]);
38919         this.initValue();
38920     },
38921
38922     /**
38923      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
38924      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
38925      * @return {Roo.form.Field} this
38926      */
38927     applyTo : function(target){
38928         this.allowDomMove = false;
38929         this.el = Roo.get(target);
38930         this.render(this.el.dom.parentNode);
38931         return this;
38932     },
38933
38934     // private
38935     initValue : function(){
38936         if(this.value !== undefined){
38937             this.setValue(this.value);
38938         }else if(this.el.dom.value.length > 0){
38939             this.setValue(this.el.dom.value);
38940         }
38941     },
38942
38943     /**
38944      * Returns true if this field has been changed since it was originally loaded and is not disabled.
38945      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
38946      */
38947     isDirty : function() {
38948         if(this.disabled) {
38949             return false;
38950         }
38951         return String(this.getValue()) !== String(this.originalValue);
38952     },
38953
38954     /**
38955      * stores the current value in loadedValue
38956      */
38957     resetHasChanged : function()
38958     {
38959         this.loadedValue = String(this.getValue());
38960     },
38961     /**
38962      * checks the current value against the 'loaded' value.
38963      * Note - will return false if 'resetHasChanged' has not been called first.
38964      */
38965     hasChanged : function()
38966     {
38967         if(this.disabled || this.readOnly) {
38968             return false;
38969         }
38970         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
38971     },
38972     
38973     
38974     
38975     // private
38976     afterRender : function(){
38977         Roo.form.Field.superclass.afterRender.call(this);
38978         this.initEvents();
38979     },
38980
38981     // private
38982     fireKey : function(e){
38983         //Roo.log('field ' + e.getKey());
38984         if(e.isNavKeyPress()){
38985             this.fireEvent("specialkey", this, e);
38986         }
38987     },
38988
38989     /**
38990      * Resets the current field value to the originally loaded value and clears any validation messages
38991      */
38992     reset : function(){
38993         this.setValue(this.resetValue);
38994         this.originalValue = this.getValue();
38995         this.clearInvalid();
38996     },
38997
38998     // private
38999     initEvents : function(){
39000         // safari killled keypress - so keydown is now used..
39001         this.el.on("keydown" , this.fireKey,  this);
39002         this.el.on("focus", this.onFocus,  this);
39003         this.el.on("blur", this.onBlur,  this);
39004         this.el.relayEvent('keyup', this);
39005
39006         // reference to original value for reset
39007         this.originalValue = this.getValue();
39008         this.resetValue =  this.getValue();
39009     },
39010
39011     // private
39012     onFocus : function(){
39013         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39014             this.el.addClass(this.focusClass);
39015         }
39016         if(!this.hasFocus){
39017             this.hasFocus = true;
39018             this.startValue = this.getValue();
39019             this.fireEvent("focus", this);
39020         }
39021     },
39022
39023     beforeBlur : Roo.emptyFn,
39024
39025     // private
39026     onBlur : function(){
39027         this.beforeBlur();
39028         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39029             this.el.removeClass(this.focusClass);
39030         }
39031         this.hasFocus = false;
39032         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39033             this.validate();
39034         }
39035         var v = this.getValue();
39036         if(String(v) !== String(this.startValue)){
39037             this.fireEvent('change', this, v, this.startValue);
39038         }
39039         this.fireEvent("blur", this);
39040     },
39041
39042     /**
39043      * Returns whether or not the field value is currently valid
39044      * @param {Boolean} preventMark True to disable marking the field invalid
39045      * @return {Boolean} True if the value is valid, else false
39046      */
39047     isValid : function(preventMark){
39048         if(this.disabled){
39049             return true;
39050         }
39051         var restore = this.preventMark;
39052         this.preventMark = preventMark === true;
39053         var v = this.validateValue(this.processValue(this.getRawValue()));
39054         this.preventMark = restore;
39055         return v;
39056     },
39057
39058     /**
39059      * Validates the field value
39060      * @return {Boolean} True if the value is valid, else false
39061      */
39062     validate : function(){
39063         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39064             this.clearInvalid();
39065             return true;
39066         }
39067         return false;
39068     },
39069
39070     processValue : function(value){
39071         return value;
39072     },
39073
39074     // private
39075     // Subclasses should provide the validation implementation by overriding this
39076     validateValue : function(value){
39077         return true;
39078     },
39079
39080     /**
39081      * Mark this field as invalid
39082      * @param {String} msg The validation message
39083      */
39084     markInvalid : function(msg){
39085         if(!this.rendered || this.preventMark){ // not rendered
39086             return;
39087         }
39088         
39089         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39090         
39091         obj.el.addClass(this.invalidClass);
39092         msg = msg || this.invalidText;
39093         switch(this.msgTarget){
39094             case 'qtip':
39095                 obj.el.dom.qtip = msg;
39096                 obj.el.dom.qclass = 'x-form-invalid-tip';
39097                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39098                     Roo.QuickTips.enable();
39099                 }
39100                 break;
39101             case 'title':
39102                 this.el.dom.title = msg;
39103                 break;
39104             case 'under':
39105                 if(!this.errorEl){
39106                     var elp = this.el.findParent('.x-form-element', 5, true);
39107                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39108                     this.errorEl.setWidth(elp.getWidth(true)-20);
39109                 }
39110                 this.errorEl.update(msg);
39111                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39112                 break;
39113             case 'side':
39114                 if(!this.errorIcon){
39115                     var elp = this.el.findParent('.x-form-element', 5, true);
39116                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39117                 }
39118                 this.alignErrorIcon();
39119                 this.errorIcon.dom.qtip = msg;
39120                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39121                 this.errorIcon.show();
39122                 this.on('resize', this.alignErrorIcon, this);
39123                 break;
39124             default:
39125                 var t = Roo.getDom(this.msgTarget);
39126                 t.innerHTML = msg;
39127                 t.style.display = this.msgDisplay;
39128                 break;
39129         }
39130         this.fireEvent('invalid', this, msg);
39131     },
39132
39133     // private
39134     alignErrorIcon : function(){
39135         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39136     },
39137
39138     /**
39139      * Clear any invalid styles/messages for this field
39140      */
39141     clearInvalid : function(){
39142         if(!this.rendered || this.preventMark){ // not rendered
39143             return;
39144         }
39145         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39146         
39147         obj.el.removeClass(this.invalidClass);
39148         switch(this.msgTarget){
39149             case 'qtip':
39150                 obj.el.dom.qtip = '';
39151                 break;
39152             case 'title':
39153                 this.el.dom.title = '';
39154                 break;
39155             case 'under':
39156                 if(this.errorEl){
39157                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39158                 }
39159                 break;
39160             case 'side':
39161                 if(this.errorIcon){
39162                     this.errorIcon.dom.qtip = '';
39163                     this.errorIcon.hide();
39164                     this.un('resize', this.alignErrorIcon, this);
39165                 }
39166                 break;
39167             default:
39168                 var t = Roo.getDom(this.msgTarget);
39169                 t.innerHTML = '';
39170                 t.style.display = 'none';
39171                 break;
39172         }
39173         this.fireEvent('valid', this);
39174     },
39175
39176     /**
39177      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39178      * @return {Mixed} value The field value
39179      */
39180     getRawValue : function(){
39181         var v = this.el.getValue();
39182         
39183         return v;
39184     },
39185
39186     /**
39187      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39188      * @return {Mixed} value The field value
39189      */
39190     getValue : function(){
39191         var v = this.el.getValue();
39192          
39193         return v;
39194     },
39195
39196     /**
39197      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39198      * @param {Mixed} value The value to set
39199      */
39200     setRawValue : function(v){
39201         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39202     },
39203
39204     /**
39205      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39206      * @param {Mixed} value The value to set
39207      */
39208     setValue : function(v){
39209         this.value = v;
39210         if(this.rendered){
39211             this.el.dom.value = (v === null || v === undefined ? '' : v);
39212              this.validate();
39213         }
39214     },
39215
39216     adjustSize : function(w, h){
39217         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39218         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39219         return s;
39220     },
39221
39222     adjustWidth : function(tag, w){
39223         tag = tag.toLowerCase();
39224         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39225             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39226                 if(tag == 'input'){
39227                     return w + 2;
39228                 }
39229                 if(tag == 'textarea'){
39230                     return w-2;
39231                 }
39232             }else if(Roo.isOpera){
39233                 if(tag == 'input'){
39234                     return w + 2;
39235                 }
39236                 if(tag == 'textarea'){
39237                     return w-2;
39238                 }
39239             }
39240         }
39241         return w;
39242     }
39243 });
39244
39245
39246 // anything other than normal should be considered experimental
39247 Roo.form.Field.msgFx = {
39248     normal : {
39249         show: function(msgEl, f){
39250             msgEl.setDisplayed('block');
39251         },
39252
39253         hide : function(msgEl, f){
39254             msgEl.setDisplayed(false).update('');
39255         }
39256     },
39257
39258     slide : {
39259         show: function(msgEl, f){
39260             msgEl.slideIn('t', {stopFx:true});
39261         },
39262
39263         hide : function(msgEl, f){
39264             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39265         }
39266     },
39267
39268     slideRight : {
39269         show: function(msgEl, f){
39270             msgEl.fixDisplay();
39271             msgEl.alignTo(f.el, 'tl-tr');
39272             msgEl.slideIn('l', {stopFx:true});
39273         },
39274
39275         hide : function(msgEl, f){
39276             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39277         }
39278     }
39279 };/*
39280  * Based on:
39281  * Ext JS Library 1.1.1
39282  * Copyright(c) 2006-2007, Ext JS, LLC.
39283  *
39284  * Originally Released Under LGPL - original licence link has changed is not relivant.
39285  *
39286  * Fork - LGPL
39287  * <script type="text/javascript">
39288  */
39289  
39290
39291 /**
39292  * @class Roo.form.TextField
39293  * @extends Roo.form.Field
39294  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39295  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39296  * @constructor
39297  * Creates a new TextField
39298  * @param {Object} config Configuration options
39299  */
39300 Roo.form.TextField = function(config){
39301     Roo.form.TextField.superclass.constructor.call(this, config);
39302     this.addEvents({
39303         /**
39304          * @event autosize
39305          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39306          * according to the default logic, but this event provides a hook for the developer to apply additional
39307          * logic at runtime to resize the field if needed.
39308              * @param {Roo.form.Field} this This text field
39309              * @param {Number} width The new field width
39310              */
39311         autosize : true
39312     });
39313 };
39314
39315 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39316     /**
39317      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39318      */
39319     grow : false,
39320     /**
39321      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39322      */
39323     growMin : 30,
39324     /**
39325      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39326      */
39327     growMax : 800,
39328     /**
39329      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39330      */
39331     vtype : null,
39332     /**
39333      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39334      */
39335     maskRe : null,
39336     /**
39337      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39338      */
39339     disableKeyFilter : false,
39340     /**
39341      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39342      */
39343     allowBlank : true,
39344     /**
39345      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39346      */
39347     minLength : 0,
39348     /**
39349      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39350      */
39351     maxLength : Number.MAX_VALUE,
39352     /**
39353      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39354      */
39355     minLengthText : "The minimum length for this field is {0}",
39356     /**
39357      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39358      */
39359     maxLengthText : "The maximum length for this field is {0}",
39360     /**
39361      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39362      */
39363     selectOnFocus : false,
39364     /**
39365      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39366      */
39367     blankText : "This field is required",
39368     /**
39369      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39370      * If available, this function will be called only after the basic validators all return true, and will be passed the
39371      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39372      */
39373     validator : null,
39374     /**
39375      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39376      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39377      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39378      */
39379     regex : null,
39380     /**
39381      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39382      */
39383     regexText : "",
39384     /**
39385      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39386      */
39387     emptyText : null,
39388    
39389
39390     // private
39391     initEvents : function()
39392     {
39393         if (this.emptyText) {
39394             this.el.attr('placeholder', this.emptyText);
39395         }
39396         
39397         Roo.form.TextField.superclass.initEvents.call(this);
39398         if(this.validationEvent == 'keyup'){
39399             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39400             this.el.on('keyup', this.filterValidation, this);
39401         }
39402         else if(this.validationEvent !== false){
39403             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39404         }
39405         
39406         if(this.selectOnFocus){
39407             this.on("focus", this.preFocus, this);
39408             
39409         }
39410         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39411             this.el.on("keypress", this.filterKeys, this);
39412         }
39413         if(this.grow){
39414             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39415             this.el.on("click", this.autoSize,  this);
39416         }
39417         if(this.el.is('input[type=password]') && Roo.isSafari){
39418             this.el.on('keydown', this.SafariOnKeyDown, this);
39419         }
39420     },
39421
39422     processValue : function(value){
39423         if(this.stripCharsRe){
39424             var newValue = value.replace(this.stripCharsRe, '');
39425             if(newValue !== value){
39426                 this.setRawValue(newValue);
39427                 return newValue;
39428             }
39429         }
39430         return value;
39431     },
39432
39433     filterValidation : function(e){
39434         if(!e.isNavKeyPress()){
39435             this.validationTask.delay(this.validationDelay);
39436         }
39437     },
39438
39439     // private
39440     onKeyUp : function(e){
39441         if(!e.isNavKeyPress()){
39442             this.autoSize();
39443         }
39444     },
39445
39446     /**
39447      * Resets the current field value to the originally-loaded value and clears any validation messages.
39448      *  
39449      */
39450     reset : function(){
39451         Roo.form.TextField.superclass.reset.call(this);
39452        
39453     },
39454
39455     
39456     // private
39457     preFocus : function(){
39458         
39459         if(this.selectOnFocus){
39460             this.el.dom.select();
39461         }
39462     },
39463
39464     
39465     // private
39466     filterKeys : function(e){
39467         var k = e.getKey();
39468         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39469             return;
39470         }
39471         var c = e.getCharCode(), cc = String.fromCharCode(c);
39472         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39473             return;
39474         }
39475         if(!this.maskRe.test(cc)){
39476             e.stopEvent();
39477         }
39478     },
39479
39480     setValue : function(v){
39481         
39482         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39483         
39484         this.autoSize();
39485     },
39486
39487     /**
39488      * Validates a value according to the field's validation rules and marks the field as invalid
39489      * if the validation fails
39490      * @param {Mixed} value The value to validate
39491      * @return {Boolean} True if the value is valid, else false
39492      */
39493     validateValue : function(value){
39494         if(value.length < 1)  { // if it's blank
39495              if(this.allowBlank){
39496                 this.clearInvalid();
39497                 return true;
39498              }else{
39499                 this.markInvalid(this.blankText);
39500                 return false;
39501              }
39502         }
39503         if(value.length < this.minLength){
39504             this.markInvalid(String.format(this.minLengthText, this.minLength));
39505             return false;
39506         }
39507         if(value.length > this.maxLength){
39508             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39509             return false;
39510         }
39511         if(this.vtype){
39512             var vt = Roo.form.VTypes;
39513             if(!vt[this.vtype](value, this)){
39514                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39515                 return false;
39516             }
39517         }
39518         if(typeof this.validator == "function"){
39519             var msg = this.validator(value);
39520             if(msg !== true){
39521                 this.markInvalid(msg);
39522                 return false;
39523             }
39524         }
39525         if(this.regex && !this.regex.test(value)){
39526             this.markInvalid(this.regexText);
39527             return false;
39528         }
39529         return true;
39530     },
39531
39532     /**
39533      * Selects text in this field
39534      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39535      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39536      */
39537     selectText : function(start, end){
39538         var v = this.getRawValue();
39539         if(v.length > 0){
39540             start = start === undefined ? 0 : start;
39541             end = end === undefined ? v.length : end;
39542             var d = this.el.dom;
39543             if(d.setSelectionRange){
39544                 d.setSelectionRange(start, end);
39545             }else if(d.createTextRange){
39546                 var range = d.createTextRange();
39547                 range.moveStart("character", start);
39548                 range.moveEnd("character", v.length-end);
39549                 range.select();
39550             }
39551         }
39552     },
39553
39554     /**
39555      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39556      * This only takes effect if grow = true, and fires the autosize event.
39557      */
39558     autoSize : function(){
39559         if(!this.grow || !this.rendered){
39560             return;
39561         }
39562         if(!this.metrics){
39563             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39564         }
39565         var el = this.el;
39566         var v = el.dom.value;
39567         var d = document.createElement('div');
39568         d.appendChild(document.createTextNode(v));
39569         v = d.innerHTML;
39570         d = null;
39571         v += "&#160;";
39572         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39573         this.el.setWidth(w);
39574         this.fireEvent("autosize", this, w);
39575     },
39576     
39577     // private
39578     SafariOnKeyDown : function(event)
39579     {
39580         // this is a workaround for a password hang bug on chrome/ webkit.
39581         
39582         var isSelectAll = false;
39583         
39584         if(this.el.dom.selectionEnd > 0){
39585             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39586         }
39587         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39588             event.preventDefault();
39589             this.setValue('');
39590             return;
39591         }
39592         
39593         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39594             
39595             event.preventDefault();
39596             // this is very hacky as keydown always get's upper case.
39597             
39598             var cc = String.fromCharCode(event.getCharCode());
39599             
39600             
39601             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39602             
39603         }
39604         
39605         
39606     }
39607 });/*
39608  * Based on:
39609  * Ext JS Library 1.1.1
39610  * Copyright(c) 2006-2007, Ext JS, LLC.
39611  *
39612  * Originally Released Under LGPL - original licence link has changed is not relivant.
39613  *
39614  * Fork - LGPL
39615  * <script type="text/javascript">
39616  */
39617  
39618 /**
39619  * @class Roo.form.Hidden
39620  * @extends Roo.form.TextField
39621  * Simple Hidden element used on forms 
39622  * 
39623  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39624  * 
39625  * @constructor
39626  * Creates a new Hidden form element.
39627  * @param {Object} config Configuration options
39628  */
39629
39630
39631
39632 // easy hidden field...
39633 Roo.form.Hidden = function(config){
39634     Roo.form.Hidden.superclass.constructor.call(this, config);
39635 };
39636   
39637 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39638     fieldLabel:      '',
39639     inputType:      'hidden',
39640     width:          50,
39641     allowBlank:     true,
39642     labelSeparator: '',
39643     hidden:         true,
39644     itemCls :       'x-form-item-display-none'
39645
39646
39647 });
39648
39649
39650 /*
39651  * Based on:
39652  * Ext JS Library 1.1.1
39653  * Copyright(c) 2006-2007, Ext JS, LLC.
39654  *
39655  * Originally Released Under LGPL - original licence link has changed is not relivant.
39656  *
39657  * Fork - LGPL
39658  * <script type="text/javascript">
39659  */
39660  
39661 /**
39662  * @class Roo.form.TriggerField
39663  * @extends Roo.form.TextField
39664  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39665  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39666  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39667  * for which you can provide a custom implementation.  For example:
39668  * <pre><code>
39669 var trigger = new Roo.form.TriggerField();
39670 trigger.onTriggerClick = myTriggerFn;
39671 trigger.applyTo('my-field');
39672 </code></pre>
39673  *
39674  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39675  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39676  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39677  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39678  * @constructor
39679  * Create a new TriggerField.
39680  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39681  * to the base TextField)
39682  */
39683 Roo.form.TriggerField = function(config){
39684     this.mimicing = false;
39685     Roo.form.TriggerField.superclass.constructor.call(this, config);
39686 };
39687
39688 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39689     /**
39690      * @cfg {String} triggerClass A CSS class to apply to the trigger
39691      */
39692     /**
39693      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39694      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39695      */
39696     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39697     /**
39698      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39699      */
39700     hideTrigger:false,
39701
39702     /** @cfg {Boolean} grow @hide */
39703     /** @cfg {Number} growMin @hide */
39704     /** @cfg {Number} growMax @hide */
39705
39706     /**
39707      * @hide 
39708      * @method
39709      */
39710     autoSize: Roo.emptyFn,
39711     // private
39712     monitorTab : true,
39713     // private
39714     deferHeight : true,
39715
39716     
39717     actionMode : 'wrap',
39718     // private
39719     onResize : function(w, h){
39720         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39721         if(typeof w == 'number'){
39722             var x = w - this.trigger.getWidth();
39723             this.el.setWidth(this.adjustWidth('input', x));
39724             this.trigger.setStyle('left', x+'px');
39725         }
39726     },
39727
39728     // private
39729     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39730
39731     // private
39732     getResizeEl : function(){
39733         return this.wrap;
39734     },
39735
39736     // private
39737     getPositionEl : function(){
39738         return this.wrap;
39739     },
39740
39741     // private
39742     alignErrorIcon : function(){
39743         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39744     },
39745
39746     // private
39747     onRender : function(ct, position){
39748         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39749         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39750         this.trigger = this.wrap.createChild(this.triggerConfig ||
39751                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39752         if(this.hideTrigger){
39753             this.trigger.setDisplayed(false);
39754         }
39755         this.initTrigger();
39756         if(!this.width){
39757             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39758         }
39759     },
39760
39761     // private
39762     initTrigger : function(){
39763         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39764         this.trigger.addClassOnOver('x-form-trigger-over');
39765         this.trigger.addClassOnClick('x-form-trigger-click');
39766     },
39767
39768     // private
39769     onDestroy : function(){
39770         if(this.trigger){
39771             this.trigger.removeAllListeners();
39772             this.trigger.remove();
39773         }
39774         if(this.wrap){
39775             this.wrap.remove();
39776         }
39777         Roo.form.TriggerField.superclass.onDestroy.call(this);
39778     },
39779
39780     // private
39781     onFocus : function(){
39782         Roo.form.TriggerField.superclass.onFocus.call(this);
39783         if(!this.mimicing){
39784             this.wrap.addClass('x-trigger-wrap-focus');
39785             this.mimicing = true;
39786             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39787             if(this.monitorTab){
39788                 this.el.on("keydown", this.checkTab, this);
39789             }
39790         }
39791     },
39792
39793     // private
39794     checkTab : function(e){
39795         if(e.getKey() == e.TAB){
39796             this.triggerBlur();
39797         }
39798     },
39799
39800     // private
39801     onBlur : function(){
39802         // do nothing
39803     },
39804
39805     // private
39806     mimicBlur : function(e, t){
39807         if(!this.wrap.contains(t) && this.validateBlur()){
39808             this.triggerBlur();
39809         }
39810     },
39811
39812     // private
39813     triggerBlur : function(){
39814         this.mimicing = false;
39815         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39816         if(this.monitorTab){
39817             this.el.un("keydown", this.checkTab, this);
39818         }
39819         this.wrap.removeClass('x-trigger-wrap-focus');
39820         Roo.form.TriggerField.superclass.onBlur.call(this);
39821     },
39822
39823     // private
39824     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39825     validateBlur : function(e, t){
39826         return true;
39827     },
39828
39829     // private
39830     onDisable : function(){
39831         Roo.form.TriggerField.superclass.onDisable.call(this);
39832         if(this.wrap){
39833             this.wrap.addClass('x-item-disabled');
39834         }
39835     },
39836
39837     // private
39838     onEnable : function(){
39839         Roo.form.TriggerField.superclass.onEnable.call(this);
39840         if(this.wrap){
39841             this.wrap.removeClass('x-item-disabled');
39842         }
39843     },
39844
39845     // private
39846     onShow : function(){
39847         var ae = this.getActionEl();
39848         
39849         if(ae){
39850             ae.dom.style.display = '';
39851             ae.dom.style.visibility = 'visible';
39852         }
39853     },
39854
39855     // private
39856     
39857     onHide : function(){
39858         var ae = this.getActionEl();
39859         ae.dom.style.display = 'none';
39860     },
39861
39862     /**
39863      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39864      * by an implementing function.
39865      * @method
39866      * @param {EventObject} e
39867      */
39868     onTriggerClick : Roo.emptyFn
39869 });
39870
39871 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
39872 // to be extended by an implementing class.  For an example of implementing this class, see the custom
39873 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
39874 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
39875     initComponent : function(){
39876         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
39877
39878         this.triggerConfig = {
39879             tag:'span', cls:'x-form-twin-triggers', cn:[
39880             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
39881             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
39882         ]};
39883     },
39884
39885     getTrigger : function(index){
39886         return this.triggers[index];
39887     },
39888
39889     initTrigger : function(){
39890         var ts = this.trigger.select('.x-form-trigger', true);
39891         this.wrap.setStyle('overflow', 'hidden');
39892         var triggerField = this;
39893         ts.each(function(t, all, index){
39894             t.hide = function(){
39895                 var w = triggerField.wrap.getWidth();
39896                 this.dom.style.display = 'none';
39897                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39898             };
39899             t.show = function(){
39900                 var w = triggerField.wrap.getWidth();
39901                 this.dom.style.display = '';
39902                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
39903             };
39904             var triggerIndex = 'Trigger'+(index+1);
39905
39906             if(this['hide'+triggerIndex]){
39907                 t.dom.style.display = 'none';
39908             }
39909             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
39910             t.addClassOnOver('x-form-trigger-over');
39911             t.addClassOnClick('x-form-trigger-click');
39912         }, this);
39913         this.triggers = ts.elements;
39914     },
39915
39916     onTrigger1Click : Roo.emptyFn,
39917     onTrigger2Click : Roo.emptyFn
39918 });/*
39919  * Based on:
39920  * Ext JS Library 1.1.1
39921  * Copyright(c) 2006-2007, Ext JS, LLC.
39922  *
39923  * Originally Released Under LGPL - original licence link has changed is not relivant.
39924  *
39925  * Fork - LGPL
39926  * <script type="text/javascript">
39927  */
39928  
39929 /**
39930  * @class Roo.form.TextArea
39931  * @extends Roo.form.TextField
39932  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
39933  * support for auto-sizing.
39934  * @constructor
39935  * Creates a new TextArea
39936  * @param {Object} config Configuration options
39937  */
39938 Roo.form.TextArea = function(config){
39939     Roo.form.TextArea.superclass.constructor.call(this, config);
39940     // these are provided exchanges for backwards compat
39941     // minHeight/maxHeight were replaced by growMin/growMax to be
39942     // compatible with TextField growing config values
39943     if(this.minHeight !== undefined){
39944         this.growMin = this.minHeight;
39945     }
39946     if(this.maxHeight !== undefined){
39947         this.growMax = this.maxHeight;
39948     }
39949 };
39950
39951 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
39952     /**
39953      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
39954      */
39955     growMin : 60,
39956     /**
39957      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
39958      */
39959     growMax: 1000,
39960     /**
39961      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
39962      * in the field (equivalent to setting overflow: hidden, defaults to false)
39963      */
39964     preventScrollbars: false,
39965     /**
39966      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39967      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
39968      */
39969
39970     // private
39971     onRender : function(ct, position){
39972         if(!this.el){
39973             this.defaultAutoCreate = {
39974                 tag: "textarea",
39975                 style:"width:300px;height:60px;",
39976                 autocomplete: "new-password"
39977             };
39978         }
39979         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
39980         if(this.grow){
39981             this.textSizeEl = Roo.DomHelper.append(document.body, {
39982                 tag: "pre", cls: "x-form-grow-sizer"
39983             });
39984             if(this.preventScrollbars){
39985                 this.el.setStyle("overflow", "hidden");
39986             }
39987             this.el.setHeight(this.growMin);
39988         }
39989     },
39990
39991     onDestroy : function(){
39992         if(this.textSizeEl){
39993             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
39994         }
39995         Roo.form.TextArea.superclass.onDestroy.call(this);
39996     },
39997
39998     // private
39999     onKeyUp : function(e){
40000         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40001             this.autoSize();
40002         }
40003     },
40004
40005     /**
40006      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40007      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40008      */
40009     autoSize : function(){
40010         if(!this.grow || !this.textSizeEl){
40011             return;
40012         }
40013         var el = this.el;
40014         var v = el.dom.value;
40015         var ts = this.textSizeEl;
40016
40017         ts.innerHTML = '';
40018         ts.appendChild(document.createTextNode(v));
40019         v = ts.innerHTML;
40020
40021         Roo.fly(ts).setWidth(this.el.getWidth());
40022         if(v.length < 1){
40023             v = "&#160;&#160;";
40024         }else{
40025             if(Roo.isIE){
40026                 v = v.replace(/\n/g, '<p>&#160;</p>');
40027             }
40028             v += "&#160;\n&#160;";
40029         }
40030         ts.innerHTML = v;
40031         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40032         if(h != this.lastHeight){
40033             this.lastHeight = h;
40034             this.el.setHeight(h);
40035             this.fireEvent("autosize", this, h);
40036         }
40037     }
40038 });/*
40039  * Based on:
40040  * Ext JS Library 1.1.1
40041  * Copyright(c) 2006-2007, Ext JS, LLC.
40042  *
40043  * Originally Released Under LGPL - original licence link has changed is not relivant.
40044  *
40045  * Fork - LGPL
40046  * <script type="text/javascript">
40047  */
40048  
40049
40050 /**
40051  * @class Roo.form.NumberField
40052  * @extends Roo.form.TextField
40053  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40054  * @constructor
40055  * Creates a new NumberField
40056  * @param {Object} config Configuration options
40057  */
40058 Roo.form.NumberField = function(config){
40059     Roo.form.NumberField.superclass.constructor.call(this, config);
40060 };
40061
40062 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40063     /**
40064      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40065      */
40066     fieldClass: "x-form-field x-form-num-field",
40067     /**
40068      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40069      */
40070     allowDecimals : true,
40071     /**
40072      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40073      */
40074     decimalSeparator : ".",
40075     /**
40076      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40077      */
40078     decimalPrecision : 2,
40079     /**
40080      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40081      */
40082     allowNegative : true,
40083     /**
40084      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40085      */
40086     minValue : Number.NEGATIVE_INFINITY,
40087     /**
40088      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40089      */
40090     maxValue : Number.MAX_VALUE,
40091     /**
40092      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40093      */
40094     minText : "The minimum value for this field is {0}",
40095     /**
40096      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40097      */
40098     maxText : "The maximum value for this field is {0}",
40099     /**
40100      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40101      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40102      */
40103     nanText : "{0} is not a valid number",
40104
40105     // private
40106     initEvents : function(){
40107         Roo.form.NumberField.superclass.initEvents.call(this);
40108         var allowed = "0123456789";
40109         if(this.allowDecimals){
40110             allowed += this.decimalSeparator;
40111         }
40112         if(this.allowNegative){
40113             allowed += "-";
40114         }
40115         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40116         var keyPress = function(e){
40117             var k = e.getKey();
40118             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40119                 return;
40120             }
40121             var c = e.getCharCode();
40122             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40123                 e.stopEvent();
40124             }
40125         };
40126         this.el.on("keypress", keyPress, this);
40127     },
40128
40129     // private
40130     validateValue : function(value){
40131         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40132             return false;
40133         }
40134         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40135              return true;
40136         }
40137         var num = this.parseValue(value);
40138         if(isNaN(num)){
40139             this.markInvalid(String.format(this.nanText, value));
40140             return false;
40141         }
40142         if(num < this.minValue){
40143             this.markInvalid(String.format(this.minText, this.minValue));
40144             return false;
40145         }
40146         if(num > this.maxValue){
40147             this.markInvalid(String.format(this.maxText, this.maxValue));
40148             return false;
40149         }
40150         return true;
40151     },
40152
40153     getValue : function(){
40154         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40155     },
40156
40157     // private
40158     parseValue : function(value){
40159         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40160         return isNaN(value) ? '' : value;
40161     },
40162
40163     // private
40164     fixPrecision : function(value){
40165         var nan = isNaN(value);
40166         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40167             return nan ? '' : value;
40168         }
40169         return parseFloat(value).toFixed(this.decimalPrecision);
40170     },
40171
40172     setValue : function(v){
40173         v = this.fixPrecision(v);
40174         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40175     },
40176
40177     // private
40178     decimalPrecisionFcn : function(v){
40179         return Math.floor(v);
40180     },
40181
40182     beforeBlur : function(){
40183         var v = this.parseValue(this.getRawValue());
40184         if(v){
40185             this.setValue(v);
40186         }
40187     }
40188 });/*
40189  * Based on:
40190  * Ext JS Library 1.1.1
40191  * Copyright(c) 2006-2007, Ext JS, LLC.
40192  *
40193  * Originally Released Under LGPL - original licence link has changed is not relivant.
40194  *
40195  * Fork - LGPL
40196  * <script type="text/javascript">
40197  */
40198  
40199 /**
40200  * @class Roo.form.DateField
40201  * @extends Roo.form.TriggerField
40202  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40203 * @constructor
40204 * Create a new DateField
40205 * @param {Object} config
40206  */
40207 Roo.form.DateField = function(config){
40208     Roo.form.DateField.superclass.constructor.call(this, config);
40209     
40210       this.addEvents({
40211          
40212         /**
40213          * @event select
40214          * Fires when a date is selected
40215              * @param {Roo.form.DateField} combo This combo box
40216              * @param {Date} date The date selected
40217              */
40218         'select' : true
40219          
40220     });
40221     
40222     
40223     if(typeof this.minValue == "string") {
40224         this.minValue = this.parseDate(this.minValue);
40225     }
40226     if(typeof this.maxValue == "string") {
40227         this.maxValue = this.parseDate(this.maxValue);
40228     }
40229     this.ddMatch = null;
40230     if(this.disabledDates){
40231         var dd = this.disabledDates;
40232         var re = "(?:";
40233         for(var i = 0; i < dd.length; i++){
40234             re += dd[i];
40235             if(i != dd.length-1) {
40236                 re += "|";
40237             }
40238         }
40239         this.ddMatch = new RegExp(re + ")");
40240     }
40241 };
40242
40243 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40244     /**
40245      * @cfg {String} format
40246      * The default date format string which can be overriden for localization support.  The format must be
40247      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40248      */
40249     format : "m/d/y",
40250     /**
40251      * @cfg {String} altFormats
40252      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40253      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40254      */
40255     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40256     /**
40257      * @cfg {Array} disabledDays
40258      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40259      */
40260     disabledDays : null,
40261     /**
40262      * @cfg {String} disabledDaysText
40263      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40264      */
40265     disabledDaysText : "Disabled",
40266     /**
40267      * @cfg {Array} disabledDates
40268      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40269      * expression so they are very powerful. Some examples:
40270      * <ul>
40271      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40272      * <li>["03/08", "09/16"] would disable those days for every year</li>
40273      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40274      * <li>["03/../2006"] would disable every day in March 2006</li>
40275      * <li>["^03"] would disable every day in every March</li>
40276      * </ul>
40277      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40278      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40279      */
40280     disabledDates : null,
40281     /**
40282      * @cfg {String} disabledDatesText
40283      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40284      */
40285     disabledDatesText : "Disabled",
40286     /**
40287      * @cfg {Date/String} minValue
40288      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40289      * valid format (defaults to null).
40290      */
40291     minValue : null,
40292     /**
40293      * @cfg {Date/String} maxValue
40294      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40295      * valid format (defaults to null).
40296      */
40297     maxValue : null,
40298     /**
40299      * @cfg {String} minText
40300      * The error text to display when the date in the cell is before minValue (defaults to
40301      * 'The date in this field must be after {minValue}').
40302      */
40303     minText : "The date in this field must be equal to or after {0}",
40304     /**
40305      * @cfg {String} maxText
40306      * The error text to display when the date in the cell is after maxValue (defaults to
40307      * 'The date in this field must be before {maxValue}').
40308      */
40309     maxText : "The date in this field must be equal to or before {0}",
40310     /**
40311      * @cfg {String} invalidText
40312      * The error text to display when the date in the field is invalid (defaults to
40313      * '{value} is not a valid date - it must be in the format {format}').
40314      */
40315     invalidText : "{0} is not a valid date - it must be in the format {1}",
40316     /**
40317      * @cfg {String} triggerClass
40318      * An additional CSS class used to style the trigger button.  The trigger will always get the
40319      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40320      * which displays a calendar icon).
40321      */
40322     triggerClass : 'x-form-date-trigger',
40323     
40324
40325     /**
40326      * @cfg {Boolean} useIso
40327      * if enabled, then the date field will use a hidden field to store the 
40328      * real value as iso formated date. default (false)
40329      */ 
40330     useIso : false,
40331     /**
40332      * @cfg {String/Object} autoCreate
40333      * A DomHelper element spec, or true for a default element spec (defaults to
40334      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40335      */ 
40336     // private
40337     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40338     
40339     // private
40340     hiddenField: false,
40341     
40342     onRender : function(ct, position)
40343     {
40344         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40345         if (this.useIso) {
40346             //this.el.dom.removeAttribute('name'); 
40347             Roo.log("Changing name?");
40348             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40349             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40350                     'before', true);
40351             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40352             // prevent input submission
40353             this.hiddenName = this.name;
40354         }
40355             
40356             
40357     },
40358     
40359     // private
40360     validateValue : function(value)
40361     {
40362         value = this.formatDate(value);
40363         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40364             Roo.log('super failed');
40365             return false;
40366         }
40367         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40368              return true;
40369         }
40370         var svalue = value;
40371         value = this.parseDate(value);
40372         if(!value){
40373             Roo.log('parse date failed' + svalue);
40374             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40375             return false;
40376         }
40377         var time = value.getTime();
40378         if(this.minValue && time < this.minValue.getTime()){
40379             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40380             return false;
40381         }
40382         if(this.maxValue && time > this.maxValue.getTime()){
40383             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40384             return false;
40385         }
40386         if(this.disabledDays){
40387             var day = value.getDay();
40388             for(var i = 0; i < this.disabledDays.length; i++) {
40389                 if(day === this.disabledDays[i]){
40390                     this.markInvalid(this.disabledDaysText);
40391                     return false;
40392                 }
40393             }
40394         }
40395         var fvalue = this.formatDate(value);
40396         if(this.ddMatch && this.ddMatch.test(fvalue)){
40397             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40398             return false;
40399         }
40400         return true;
40401     },
40402
40403     // private
40404     // Provides logic to override the default TriggerField.validateBlur which just returns true
40405     validateBlur : function(){
40406         return !this.menu || !this.menu.isVisible();
40407     },
40408     
40409     getName: function()
40410     {
40411         // returns hidden if it's set..
40412         if (!this.rendered) {return ''};
40413         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40414         
40415     },
40416
40417     /**
40418      * Returns the current date value of the date field.
40419      * @return {Date} The date value
40420      */
40421     getValue : function(){
40422         
40423         return  this.hiddenField ?
40424                 this.hiddenField.value :
40425                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40426     },
40427
40428     /**
40429      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40430      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40431      * (the default format used is "m/d/y").
40432      * <br />Usage:
40433      * <pre><code>
40434 //All of these calls set the same date value (May 4, 2006)
40435
40436 //Pass a date object:
40437 var dt = new Date('5/4/06');
40438 dateField.setValue(dt);
40439
40440 //Pass a date string (default format):
40441 dateField.setValue('5/4/06');
40442
40443 //Pass a date string (custom format):
40444 dateField.format = 'Y-m-d';
40445 dateField.setValue('2006-5-4');
40446 </code></pre>
40447      * @param {String/Date} date The date or valid date string
40448      */
40449     setValue : function(date){
40450         if (this.hiddenField) {
40451             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40452         }
40453         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40454         // make sure the value field is always stored as a date..
40455         this.value = this.parseDate(date);
40456         
40457         
40458     },
40459
40460     // private
40461     parseDate : function(value){
40462         if(!value || value instanceof Date){
40463             return value;
40464         }
40465         var v = Date.parseDate(value, this.format);
40466          if (!v && this.useIso) {
40467             v = Date.parseDate(value, 'Y-m-d');
40468         }
40469         if(!v && this.altFormats){
40470             if(!this.altFormatsArray){
40471                 this.altFormatsArray = this.altFormats.split("|");
40472             }
40473             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40474                 v = Date.parseDate(value, this.altFormatsArray[i]);
40475             }
40476         }
40477         return v;
40478     },
40479
40480     // private
40481     formatDate : function(date, fmt){
40482         return (!date || !(date instanceof Date)) ?
40483                date : date.dateFormat(fmt || this.format);
40484     },
40485
40486     // private
40487     menuListeners : {
40488         select: function(m, d){
40489             
40490             this.setValue(d);
40491             this.fireEvent('select', this, d);
40492         },
40493         show : function(){ // retain focus styling
40494             this.onFocus();
40495         },
40496         hide : function(){
40497             this.focus.defer(10, this);
40498             var ml = this.menuListeners;
40499             this.menu.un("select", ml.select,  this);
40500             this.menu.un("show", ml.show,  this);
40501             this.menu.un("hide", ml.hide,  this);
40502         }
40503     },
40504
40505     // private
40506     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40507     onTriggerClick : function(){
40508         if(this.disabled){
40509             return;
40510         }
40511         if(this.menu == null){
40512             this.menu = new Roo.menu.DateMenu();
40513         }
40514         Roo.apply(this.menu.picker,  {
40515             showClear: this.allowBlank,
40516             minDate : this.minValue,
40517             maxDate : this.maxValue,
40518             disabledDatesRE : this.ddMatch,
40519             disabledDatesText : this.disabledDatesText,
40520             disabledDays : this.disabledDays,
40521             disabledDaysText : this.disabledDaysText,
40522             format : this.useIso ? 'Y-m-d' : this.format,
40523             minText : String.format(this.minText, this.formatDate(this.minValue)),
40524             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40525         });
40526         this.menu.on(Roo.apply({}, this.menuListeners, {
40527             scope:this
40528         }));
40529         this.menu.picker.setValue(this.getValue() || new Date());
40530         this.menu.show(this.el, "tl-bl?");
40531     },
40532
40533     beforeBlur : function(){
40534         var v = this.parseDate(this.getRawValue());
40535         if(v){
40536             this.setValue(v);
40537         }
40538     },
40539
40540     /*@
40541      * overide
40542      * 
40543      */
40544     isDirty : function() {
40545         if(this.disabled) {
40546             return false;
40547         }
40548         
40549         if(typeof(this.startValue) === 'undefined'){
40550             return false;
40551         }
40552         
40553         return String(this.getValue()) !== String(this.startValue);
40554         
40555     }
40556 });/*
40557  * Based on:
40558  * Ext JS Library 1.1.1
40559  * Copyright(c) 2006-2007, Ext JS, LLC.
40560  *
40561  * Originally Released Under LGPL - original licence link has changed is not relivant.
40562  *
40563  * Fork - LGPL
40564  * <script type="text/javascript">
40565  */
40566  
40567 /**
40568  * @class Roo.form.MonthField
40569  * @extends Roo.form.TriggerField
40570  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40571 * @constructor
40572 * Create a new MonthField
40573 * @param {Object} config
40574  */
40575 Roo.form.MonthField = function(config){
40576     
40577     Roo.form.MonthField.superclass.constructor.call(this, config);
40578     
40579       this.addEvents({
40580          
40581         /**
40582          * @event select
40583          * Fires when a date is selected
40584              * @param {Roo.form.MonthFieeld} combo This combo box
40585              * @param {Date} date The date selected
40586              */
40587         'select' : true
40588          
40589     });
40590     
40591     
40592     if(typeof this.minValue == "string") {
40593         this.minValue = this.parseDate(this.minValue);
40594     }
40595     if(typeof this.maxValue == "string") {
40596         this.maxValue = this.parseDate(this.maxValue);
40597     }
40598     this.ddMatch = null;
40599     if(this.disabledDates){
40600         var dd = this.disabledDates;
40601         var re = "(?:";
40602         for(var i = 0; i < dd.length; i++){
40603             re += dd[i];
40604             if(i != dd.length-1) {
40605                 re += "|";
40606             }
40607         }
40608         this.ddMatch = new RegExp(re + ")");
40609     }
40610 };
40611
40612 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40613     /**
40614      * @cfg {String} format
40615      * The default date format string which can be overriden for localization support.  The format must be
40616      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40617      */
40618     format : "M Y",
40619     /**
40620      * @cfg {String} altFormats
40621      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40622      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40623      */
40624     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40625     /**
40626      * @cfg {Array} disabledDays
40627      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40628      */
40629     disabledDays : [0,1,2,3,4,5,6],
40630     /**
40631      * @cfg {String} disabledDaysText
40632      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40633      */
40634     disabledDaysText : "Disabled",
40635     /**
40636      * @cfg {Array} disabledDates
40637      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40638      * expression so they are very powerful. Some examples:
40639      * <ul>
40640      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40641      * <li>["03/08", "09/16"] would disable those days for every year</li>
40642      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40643      * <li>["03/../2006"] would disable every day in March 2006</li>
40644      * <li>["^03"] would disable every day in every March</li>
40645      * </ul>
40646      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40647      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40648      */
40649     disabledDates : null,
40650     /**
40651      * @cfg {String} disabledDatesText
40652      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40653      */
40654     disabledDatesText : "Disabled",
40655     /**
40656      * @cfg {Date/String} minValue
40657      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40658      * valid format (defaults to null).
40659      */
40660     minValue : null,
40661     /**
40662      * @cfg {Date/String} maxValue
40663      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40664      * valid format (defaults to null).
40665      */
40666     maxValue : null,
40667     /**
40668      * @cfg {String} minText
40669      * The error text to display when the date in the cell is before minValue (defaults to
40670      * 'The date in this field must be after {minValue}').
40671      */
40672     minText : "The date in this field must be equal to or after {0}",
40673     /**
40674      * @cfg {String} maxTextf
40675      * The error text to display when the date in the cell is after maxValue (defaults to
40676      * 'The date in this field must be before {maxValue}').
40677      */
40678     maxText : "The date in this field must be equal to or before {0}",
40679     /**
40680      * @cfg {String} invalidText
40681      * The error text to display when the date in the field is invalid (defaults to
40682      * '{value} is not a valid date - it must be in the format {format}').
40683      */
40684     invalidText : "{0} is not a valid date - it must be in the format {1}",
40685     /**
40686      * @cfg {String} triggerClass
40687      * An additional CSS class used to style the trigger button.  The trigger will always get the
40688      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40689      * which displays a calendar icon).
40690      */
40691     triggerClass : 'x-form-date-trigger',
40692     
40693
40694     /**
40695      * @cfg {Boolean} useIso
40696      * if enabled, then the date field will use a hidden field to store the 
40697      * real value as iso formated date. default (true)
40698      */ 
40699     useIso : true,
40700     /**
40701      * @cfg {String/Object} autoCreate
40702      * A DomHelper element spec, or true for a default element spec (defaults to
40703      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40704      */ 
40705     // private
40706     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40707     
40708     // private
40709     hiddenField: false,
40710     
40711     hideMonthPicker : false,
40712     
40713     onRender : function(ct, position)
40714     {
40715         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40716         if (this.useIso) {
40717             this.el.dom.removeAttribute('name'); 
40718             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40719                     'before', true);
40720             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40721             // prevent input submission
40722             this.hiddenName = this.name;
40723         }
40724             
40725             
40726     },
40727     
40728     // private
40729     validateValue : function(value)
40730     {
40731         value = this.formatDate(value);
40732         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40733             return false;
40734         }
40735         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40736              return true;
40737         }
40738         var svalue = value;
40739         value = this.parseDate(value);
40740         if(!value){
40741             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40742             return false;
40743         }
40744         var time = value.getTime();
40745         if(this.minValue && time < this.minValue.getTime()){
40746             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40747             return false;
40748         }
40749         if(this.maxValue && time > this.maxValue.getTime()){
40750             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40751             return false;
40752         }
40753         /*if(this.disabledDays){
40754             var day = value.getDay();
40755             for(var i = 0; i < this.disabledDays.length; i++) {
40756                 if(day === this.disabledDays[i]){
40757                     this.markInvalid(this.disabledDaysText);
40758                     return false;
40759                 }
40760             }
40761         }
40762         */
40763         var fvalue = this.formatDate(value);
40764         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40765             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40766             return false;
40767         }
40768         */
40769         return true;
40770     },
40771
40772     // private
40773     // Provides logic to override the default TriggerField.validateBlur which just returns true
40774     validateBlur : function(){
40775         return !this.menu || !this.menu.isVisible();
40776     },
40777
40778     /**
40779      * Returns the current date value of the date field.
40780      * @return {Date} The date value
40781      */
40782     getValue : function(){
40783         
40784         
40785         
40786         return  this.hiddenField ?
40787                 this.hiddenField.value :
40788                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40789     },
40790
40791     /**
40792      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40793      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40794      * (the default format used is "m/d/y").
40795      * <br />Usage:
40796      * <pre><code>
40797 //All of these calls set the same date value (May 4, 2006)
40798
40799 //Pass a date object:
40800 var dt = new Date('5/4/06');
40801 monthField.setValue(dt);
40802
40803 //Pass a date string (default format):
40804 monthField.setValue('5/4/06');
40805
40806 //Pass a date string (custom format):
40807 monthField.format = 'Y-m-d';
40808 monthField.setValue('2006-5-4');
40809 </code></pre>
40810      * @param {String/Date} date The date or valid date string
40811      */
40812     setValue : function(date){
40813         Roo.log('month setValue' + date);
40814         // can only be first of month..
40815         
40816         var val = this.parseDate(date);
40817         
40818         if (this.hiddenField) {
40819             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40820         }
40821         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40822         this.value = this.parseDate(date);
40823     },
40824
40825     // private
40826     parseDate : function(value){
40827         if(!value || value instanceof Date){
40828             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40829             return value;
40830         }
40831         var v = Date.parseDate(value, this.format);
40832         if (!v && this.useIso) {
40833             v = Date.parseDate(value, 'Y-m-d');
40834         }
40835         if (v) {
40836             // 
40837             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40838         }
40839         
40840         
40841         if(!v && this.altFormats){
40842             if(!this.altFormatsArray){
40843                 this.altFormatsArray = this.altFormats.split("|");
40844             }
40845             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40846                 v = Date.parseDate(value, this.altFormatsArray[i]);
40847             }
40848         }
40849         return v;
40850     },
40851
40852     // private
40853     formatDate : function(date, fmt){
40854         return (!date || !(date instanceof Date)) ?
40855                date : date.dateFormat(fmt || this.format);
40856     },
40857
40858     // private
40859     menuListeners : {
40860         select: function(m, d){
40861             this.setValue(d);
40862             this.fireEvent('select', this, d);
40863         },
40864         show : function(){ // retain focus styling
40865             this.onFocus();
40866         },
40867         hide : function(){
40868             this.focus.defer(10, this);
40869             var ml = this.menuListeners;
40870             this.menu.un("select", ml.select,  this);
40871             this.menu.un("show", ml.show,  this);
40872             this.menu.un("hide", ml.hide,  this);
40873         }
40874     },
40875     // private
40876     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40877     onTriggerClick : function(){
40878         if(this.disabled){
40879             return;
40880         }
40881         if(this.menu == null){
40882             this.menu = new Roo.menu.DateMenu();
40883            
40884         }
40885         
40886         Roo.apply(this.menu.picker,  {
40887             
40888             showClear: this.allowBlank,
40889             minDate : this.minValue,
40890             maxDate : this.maxValue,
40891             disabledDatesRE : this.ddMatch,
40892             disabledDatesText : this.disabledDatesText,
40893             
40894             format : this.useIso ? 'Y-m-d' : this.format,
40895             minText : String.format(this.minText, this.formatDate(this.minValue)),
40896             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40897             
40898         });
40899          this.menu.on(Roo.apply({}, this.menuListeners, {
40900             scope:this
40901         }));
40902        
40903         
40904         var m = this.menu;
40905         var p = m.picker;
40906         
40907         // hide month picker get's called when we called by 'before hide';
40908         
40909         var ignorehide = true;
40910         p.hideMonthPicker  = function(disableAnim){
40911             if (ignorehide) {
40912                 return;
40913             }
40914              if(this.monthPicker){
40915                 Roo.log("hideMonthPicker called");
40916                 if(disableAnim === true){
40917                     this.monthPicker.hide();
40918                 }else{
40919                     this.monthPicker.slideOut('t', {duration:.2});
40920                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
40921                     p.fireEvent("select", this, this.value);
40922                     m.hide();
40923                 }
40924             }
40925         }
40926         
40927         Roo.log('picker set value');
40928         Roo.log(this.getValue());
40929         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
40930         m.show(this.el, 'tl-bl?');
40931         ignorehide  = false;
40932         // this will trigger hideMonthPicker..
40933         
40934         
40935         // hidden the day picker
40936         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
40937         
40938         
40939         
40940       
40941         
40942         p.showMonthPicker.defer(100, p);
40943     
40944         
40945        
40946     },
40947
40948     beforeBlur : function(){
40949         var v = this.parseDate(this.getRawValue());
40950         if(v){
40951             this.setValue(v);
40952         }
40953     }
40954
40955     /** @cfg {Boolean} grow @hide */
40956     /** @cfg {Number} growMin @hide */
40957     /** @cfg {Number} growMax @hide */
40958     /**
40959      * @hide
40960      * @method autoSize
40961      */
40962 });/*
40963  * Based on:
40964  * Ext JS Library 1.1.1
40965  * Copyright(c) 2006-2007, Ext JS, LLC.
40966  *
40967  * Originally Released Under LGPL - original licence link has changed is not relivant.
40968  *
40969  * Fork - LGPL
40970  * <script type="text/javascript">
40971  */
40972  
40973
40974 /**
40975  * @class Roo.form.ComboBox
40976  * @extends Roo.form.TriggerField
40977  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
40978  * @constructor
40979  * Create a new ComboBox.
40980  * @param {Object} config Configuration options
40981  */
40982 Roo.form.ComboBox = function(config){
40983     Roo.form.ComboBox.superclass.constructor.call(this, config);
40984     this.addEvents({
40985         /**
40986          * @event expand
40987          * Fires when the dropdown list is expanded
40988              * @param {Roo.form.ComboBox} combo This combo box
40989              */
40990         'expand' : true,
40991         /**
40992          * @event collapse
40993          * Fires when the dropdown list is collapsed
40994              * @param {Roo.form.ComboBox} combo This combo box
40995              */
40996         'collapse' : true,
40997         /**
40998          * @event beforeselect
40999          * Fires before a list item is selected. Return false to cancel the selection.
41000              * @param {Roo.form.ComboBox} combo This combo box
41001              * @param {Roo.data.Record} record The data record returned from the underlying store
41002              * @param {Number} index The index of the selected item in the dropdown list
41003              */
41004         'beforeselect' : true,
41005         /**
41006          * @event select
41007          * Fires when a list item is selected
41008              * @param {Roo.form.ComboBox} combo This combo box
41009              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41010              * @param {Number} index The index of the selected item in the dropdown list
41011              */
41012         'select' : true,
41013         /**
41014          * @event beforequery
41015          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41016          * The event object passed has these properties:
41017              * @param {Roo.form.ComboBox} combo This combo box
41018              * @param {String} query The query
41019              * @param {Boolean} forceAll true to force "all" query
41020              * @param {Boolean} cancel true to cancel the query
41021              * @param {Object} e The query event object
41022              */
41023         'beforequery': true,
41024          /**
41025          * @event add
41026          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41027              * @param {Roo.form.ComboBox} combo This combo box
41028              */
41029         'add' : true,
41030         /**
41031          * @event edit
41032          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41033              * @param {Roo.form.ComboBox} combo This combo box
41034              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41035              */
41036         'edit' : true
41037         
41038         
41039     });
41040     if(this.transform){
41041         this.allowDomMove = false;
41042         var s = Roo.getDom(this.transform);
41043         if(!this.hiddenName){
41044             this.hiddenName = s.name;
41045         }
41046         if(!this.store){
41047             this.mode = 'local';
41048             var d = [], opts = s.options;
41049             for(var i = 0, len = opts.length;i < len; i++){
41050                 var o = opts[i];
41051                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41052                 if(o.selected) {
41053                     this.value = value;
41054                 }
41055                 d.push([value, o.text]);
41056             }
41057             this.store = new Roo.data.SimpleStore({
41058                 'id': 0,
41059                 fields: ['value', 'text'],
41060                 data : d
41061             });
41062             this.valueField = 'value';
41063             this.displayField = 'text';
41064         }
41065         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41066         if(!this.lazyRender){
41067             this.target = true;
41068             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41069             s.parentNode.removeChild(s); // remove it
41070             this.render(this.el.parentNode);
41071         }else{
41072             s.parentNode.removeChild(s); // remove it
41073         }
41074
41075     }
41076     if (this.store) {
41077         this.store = Roo.factory(this.store, Roo.data);
41078     }
41079     
41080     this.selectedIndex = -1;
41081     if(this.mode == 'local'){
41082         if(config.queryDelay === undefined){
41083             this.queryDelay = 10;
41084         }
41085         if(config.minChars === undefined){
41086             this.minChars = 0;
41087         }
41088     }
41089 };
41090
41091 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41092     /**
41093      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41094      */
41095     /**
41096      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41097      * rendering into an Roo.Editor, defaults to false)
41098      */
41099     /**
41100      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41101      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41102      */
41103     /**
41104      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41105      */
41106     /**
41107      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41108      * the dropdown list (defaults to undefined, with no header element)
41109      */
41110
41111      /**
41112      * @cfg {String/Roo.Template} tpl The template to use to render the output
41113      */
41114      
41115     // private
41116     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41117     /**
41118      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41119      */
41120     listWidth: undefined,
41121     /**
41122      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41123      * mode = 'remote' or 'text' if mode = 'local')
41124      */
41125     displayField: undefined,
41126     /**
41127      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41128      * mode = 'remote' or 'value' if mode = 'local'). 
41129      * Note: use of a valueField requires the user make a selection
41130      * in order for a value to be mapped.
41131      */
41132     valueField: undefined,
41133     
41134     
41135     /**
41136      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41137      * field's data value (defaults to the underlying DOM element's name)
41138      */
41139     hiddenName: undefined,
41140     /**
41141      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41142      */
41143     listClass: '',
41144     /**
41145      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41146      */
41147     selectedClass: 'x-combo-selected',
41148     /**
41149      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41150      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41151      * which displays a downward arrow icon).
41152      */
41153     triggerClass : 'x-form-arrow-trigger',
41154     /**
41155      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41156      */
41157     shadow:'sides',
41158     /**
41159      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41160      * anchor positions (defaults to 'tl-bl')
41161      */
41162     listAlign: 'tl-bl?',
41163     /**
41164      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41165      */
41166     maxHeight: 300,
41167     /**
41168      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41169      * query specified by the allQuery config option (defaults to 'query')
41170      */
41171     triggerAction: 'query',
41172     /**
41173      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41174      * (defaults to 4, does not apply if editable = false)
41175      */
41176     minChars : 4,
41177     /**
41178      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41179      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41180      */
41181     typeAhead: false,
41182     /**
41183      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41184      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41185      */
41186     queryDelay: 500,
41187     /**
41188      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41189      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41190      */
41191     pageSize: 0,
41192     /**
41193      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41194      * when editable = true (defaults to false)
41195      */
41196     selectOnFocus:false,
41197     /**
41198      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41199      */
41200     queryParam: 'query',
41201     /**
41202      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41203      * when mode = 'remote' (defaults to 'Loading...')
41204      */
41205     loadingText: 'Loading...',
41206     /**
41207      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41208      */
41209     resizable: false,
41210     /**
41211      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41212      */
41213     handleHeight : 8,
41214     /**
41215      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41216      * traditional select (defaults to true)
41217      */
41218     editable: true,
41219     /**
41220      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41221      */
41222     allQuery: '',
41223     /**
41224      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41225      */
41226     mode: 'remote',
41227     /**
41228      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41229      * listWidth has a higher value)
41230      */
41231     minListWidth : 70,
41232     /**
41233      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41234      * allow the user to set arbitrary text into the field (defaults to false)
41235      */
41236     forceSelection:false,
41237     /**
41238      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41239      * if typeAhead = true (defaults to 250)
41240      */
41241     typeAheadDelay : 250,
41242     /**
41243      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41244      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41245      */
41246     valueNotFoundText : undefined,
41247     /**
41248      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41249      */
41250     blockFocus : false,
41251     
41252     /**
41253      * @cfg {Boolean} disableClear Disable showing of clear button.
41254      */
41255     disableClear : false,
41256     /**
41257      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41258      */
41259     alwaysQuery : false,
41260     
41261     //private
41262     addicon : false,
41263     editicon: false,
41264     
41265     // element that contains real text value.. (when hidden is used..)
41266      
41267     // private
41268     onRender : function(ct, position){
41269         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41270         if(this.hiddenName){
41271             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41272                     'before', true);
41273             this.hiddenField.value =
41274                 this.hiddenValue !== undefined ? this.hiddenValue :
41275                 this.value !== undefined ? this.value : '';
41276
41277             // prevent input submission
41278             this.el.dom.removeAttribute('name');
41279              
41280              
41281         }
41282         if(Roo.isGecko){
41283             this.el.dom.setAttribute('autocomplete', 'off');
41284         }
41285
41286         var cls = 'x-combo-list';
41287
41288         this.list = new Roo.Layer({
41289             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41290         });
41291
41292         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41293         this.list.setWidth(lw);
41294         this.list.swallowEvent('mousewheel');
41295         this.assetHeight = 0;
41296
41297         if(this.title){
41298             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41299             this.assetHeight += this.header.getHeight();
41300         }
41301
41302         this.innerList = this.list.createChild({cls:cls+'-inner'});
41303         this.innerList.on('mouseover', this.onViewOver, this);
41304         this.innerList.on('mousemove', this.onViewMove, this);
41305         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41306         
41307         if(this.allowBlank && !this.pageSize && !this.disableClear){
41308             this.footer = this.list.createChild({cls:cls+'-ft'});
41309             this.pageTb = new Roo.Toolbar(this.footer);
41310            
41311         }
41312         if(this.pageSize){
41313             this.footer = this.list.createChild({cls:cls+'-ft'});
41314             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41315                     {pageSize: this.pageSize});
41316             
41317         }
41318         
41319         if (this.pageTb && this.allowBlank && !this.disableClear) {
41320             var _this = this;
41321             this.pageTb.add(new Roo.Toolbar.Fill(), {
41322                 cls: 'x-btn-icon x-btn-clear',
41323                 text: '&#160;',
41324                 handler: function()
41325                 {
41326                     _this.collapse();
41327                     _this.clearValue();
41328                     _this.onSelect(false, -1);
41329                 }
41330             });
41331         }
41332         if (this.footer) {
41333             this.assetHeight += this.footer.getHeight();
41334         }
41335         
41336
41337         if(!this.tpl){
41338             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41339         }
41340
41341         this.view = new Roo.View(this.innerList, this.tpl, {
41342             singleSelect:true, store: this.store, selectedClass: this.selectedClass
41343         });
41344
41345         this.view.on('click', this.onViewClick, this);
41346
41347         this.store.on('beforeload', this.onBeforeLoad, this);
41348         this.store.on('load', this.onLoad, this);
41349         this.store.on('loadexception', this.onLoadException, this);
41350
41351         if(this.resizable){
41352             this.resizer = new Roo.Resizable(this.list,  {
41353                pinned:true, handles:'se'
41354             });
41355             this.resizer.on('resize', function(r, w, h){
41356                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41357                 this.listWidth = w;
41358                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41359                 this.restrictHeight();
41360             }, this);
41361             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41362         }
41363         if(!this.editable){
41364             this.editable = true;
41365             this.setEditable(false);
41366         }  
41367         
41368         
41369         if (typeof(this.events.add.listeners) != 'undefined') {
41370             
41371             this.addicon = this.wrap.createChild(
41372                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41373        
41374             this.addicon.on('click', function(e) {
41375                 this.fireEvent('add', this);
41376             }, this);
41377         }
41378         if (typeof(this.events.edit.listeners) != 'undefined') {
41379             
41380             this.editicon = this.wrap.createChild(
41381                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41382             if (this.addicon) {
41383                 this.editicon.setStyle('margin-left', '40px');
41384             }
41385             this.editicon.on('click', function(e) {
41386                 
41387                 // we fire even  if inothing is selected..
41388                 this.fireEvent('edit', this, this.lastData );
41389                 
41390             }, this);
41391         }
41392         
41393         
41394         
41395     },
41396
41397     // private
41398     initEvents : function(){
41399         Roo.form.ComboBox.superclass.initEvents.call(this);
41400
41401         this.keyNav = new Roo.KeyNav(this.el, {
41402             "up" : function(e){
41403                 this.inKeyMode = true;
41404                 this.selectPrev();
41405             },
41406
41407             "down" : function(e){
41408                 if(!this.isExpanded()){
41409                     this.onTriggerClick();
41410                 }else{
41411                     this.inKeyMode = true;
41412                     this.selectNext();
41413                 }
41414             },
41415
41416             "enter" : function(e){
41417                 this.onViewClick();
41418                 //return true;
41419             },
41420
41421             "esc" : function(e){
41422                 this.collapse();
41423             },
41424
41425             "tab" : function(e){
41426                 this.onViewClick(false);
41427                 this.fireEvent("specialkey", this, e);
41428                 return true;
41429             },
41430
41431             scope : this,
41432
41433             doRelay : function(foo, bar, hname){
41434                 if(hname == 'down' || this.scope.isExpanded()){
41435                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41436                 }
41437                 return true;
41438             },
41439
41440             forceKeyDown: true
41441         });
41442         this.queryDelay = Math.max(this.queryDelay || 10,
41443                 this.mode == 'local' ? 10 : 250);
41444         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41445         if(this.typeAhead){
41446             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41447         }
41448         if(this.editable !== false){
41449             this.el.on("keyup", this.onKeyUp, this);
41450         }
41451         if(this.forceSelection){
41452             this.on('blur', this.doForce, this);
41453         }
41454     },
41455
41456     onDestroy : function(){
41457         if(this.view){
41458             this.view.setStore(null);
41459             this.view.el.removeAllListeners();
41460             this.view.el.remove();
41461             this.view.purgeListeners();
41462         }
41463         if(this.list){
41464             this.list.destroy();
41465         }
41466         if(this.store){
41467             this.store.un('beforeload', this.onBeforeLoad, this);
41468             this.store.un('load', this.onLoad, this);
41469             this.store.un('loadexception', this.onLoadException, this);
41470         }
41471         Roo.form.ComboBox.superclass.onDestroy.call(this);
41472     },
41473
41474     // private
41475     fireKey : function(e){
41476         if(e.isNavKeyPress() && !this.list.isVisible()){
41477             this.fireEvent("specialkey", this, e);
41478         }
41479     },
41480
41481     // private
41482     onResize: function(w, h){
41483         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41484         
41485         if(typeof w != 'number'){
41486             // we do not handle it!?!?
41487             return;
41488         }
41489         var tw = this.trigger.getWidth();
41490         tw += this.addicon ? this.addicon.getWidth() : 0;
41491         tw += this.editicon ? this.editicon.getWidth() : 0;
41492         var x = w - tw;
41493         this.el.setWidth( this.adjustWidth('input', x));
41494             
41495         this.trigger.setStyle('left', x+'px');
41496         
41497         if(this.list && this.listWidth === undefined){
41498             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41499             this.list.setWidth(lw);
41500             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41501         }
41502         
41503     
41504         
41505     },
41506
41507     /**
41508      * Allow or prevent the user from directly editing the field text.  If false is passed,
41509      * the user will only be able to select from the items defined in the dropdown list.  This method
41510      * is the runtime equivalent of setting the 'editable' config option at config time.
41511      * @param {Boolean} value True to allow the user to directly edit the field text
41512      */
41513     setEditable : function(value){
41514         if(value == this.editable){
41515             return;
41516         }
41517         this.editable = value;
41518         if(!value){
41519             this.el.dom.setAttribute('readOnly', true);
41520             this.el.on('mousedown', this.onTriggerClick,  this);
41521             this.el.addClass('x-combo-noedit');
41522         }else{
41523             this.el.dom.setAttribute('readOnly', false);
41524             this.el.un('mousedown', this.onTriggerClick,  this);
41525             this.el.removeClass('x-combo-noedit');
41526         }
41527     },
41528
41529     // private
41530     onBeforeLoad : function(){
41531         if(!this.hasFocus){
41532             return;
41533         }
41534         this.innerList.update(this.loadingText ?
41535                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41536         this.restrictHeight();
41537         this.selectedIndex = -1;
41538     },
41539
41540     // private
41541     onLoad : function(){
41542         if(!this.hasFocus){
41543             return;
41544         }
41545         if(this.store.getCount() > 0){
41546             this.expand();
41547             this.restrictHeight();
41548             if(this.lastQuery == this.allQuery){
41549                 if(this.editable){
41550                     this.el.dom.select();
41551                 }
41552                 if(!this.selectByValue(this.value, true)){
41553                     this.select(0, true);
41554                 }
41555             }else{
41556                 this.selectNext();
41557                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41558                     this.taTask.delay(this.typeAheadDelay);
41559                 }
41560             }
41561         }else{
41562             this.onEmptyResults();
41563         }
41564         //this.el.focus();
41565     },
41566     // private
41567     onLoadException : function()
41568     {
41569         this.collapse();
41570         Roo.log(this.store.reader.jsonData);
41571         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41572             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41573         }
41574         
41575         
41576     },
41577     // private
41578     onTypeAhead : function(){
41579         if(this.store.getCount() > 0){
41580             var r = this.store.getAt(0);
41581             var newValue = r.data[this.displayField];
41582             var len = newValue.length;
41583             var selStart = this.getRawValue().length;
41584             if(selStart != len){
41585                 this.setRawValue(newValue);
41586                 this.selectText(selStart, newValue.length);
41587             }
41588         }
41589     },
41590
41591     // private
41592     onSelect : function(record, index){
41593         if(this.fireEvent('beforeselect', this, record, index) !== false){
41594             this.setFromData(index > -1 ? record.data : false);
41595             this.collapse();
41596             this.fireEvent('select', this, record, index);
41597         }
41598     },
41599
41600     /**
41601      * Returns the currently selected field value or empty string if no value is set.
41602      * @return {String} value The selected value
41603      */
41604     getValue : function(){
41605         if(this.valueField){
41606             return typeof this.value != 'undefined' ? this.value : '';
41607         }
41608         return Roo.form.ComboBox.superclass.getValue.call(this);
41609     },
41610
41611     /**
41612      * Clears any text/value currently set in the field
41613      */
41614     clearValue : function(){
41615         if(this.hiddenField){
41616             this.hiddenField.value = '';
41617         }
41618         this.value = '';
41619         this.setRawValue('');
41620         this.lastSelectionText = '';
41621         
41622     },
41623
41624     /**
41625      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41626      * will be displayed in the field.  If the value does not match the data value of an existing item,
41627      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41628      * Otherwise the field will be blank (although the value will still be set).
41629      * @param {String} value The value to match
41630      */
41631     setValue : function(v){
41632         var text = v;
41633         if(this.valueField){
41634             var r = this.findRecord(this.valueField, v);
41635             if(r){
41636                 text = r.data[this.displayField];
41637             }else if(this.valueNotFoundText !== undefined){
41638                 text = this.valueNotFoundText;
41639             }
41640         }
41641         this.lastSelectionText = text;
41642         if(this.hiddenField){
41643             this.hiddenField.value = v;
41644         }
41645         Roo.form.ComboBox.superclass.setValue.call(this, text);
41646         this.value = v;
41647     },
41648     /**
41649      * @property {Object} the last set data for the element
41650      */
41651     
41652     lastData : false,
41653     /**
41654      * Sets the value of the field based on a object which is related to the record format for the store.
41655      * @param {Object} value the value to set as. or false on reset?
41656      */
41657     setFromData : function(o){
41658         var dv = ''; // display value
41659         var vv = ''; // value value..
41660         this.lastData = o;
41661         if (this.displayField) {
41662             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41663         } else {
41664             // this is an error condition!!!
41665             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41666         }
41667         
41668         if(this.valueField){
41669             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41670         }
41671         if(this.hiddenField){
41672             this.hiddenField.value = vv;
41673             
41674             this.lastSelectionText = dv;
41675             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41676             this.value = vv;
41677             return;
41678         }
41679         // no hidden field.. - we store the value in 'value', but still display
41680         // display field!!!!
41681         this.lastSelectionText = dv;
41682         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41683         this.value = vv;
41684         
41685         
41686     },
41687     // private
41688     reset : function(){
41689         // overridden so that last data is reset..
41690         this.setValue(this.resetValue);
41691         this.originalValue = this.getValue();
41692         this.clearInvalid();
41693         this.lastData = false;
41694         if (this.view) {
41695             this.view.clearSelections();
41696         }
41697     },
41698     // private
41699     findRecord : function(prop, value){
41700         var record;
41701         if(this.store.getCount() > 0){
41702             this.store.each(function(r){
41703                 if(r.data[prop] == value){
41704                     record = r;
41705                     return false;
41706                 }
41707                 return true;
41708             });
41709         }
41710         return record;
41711     },
41712     
41713     getName: function()
41714     {
41715         // returns hidden if it's set..
41716         if (!this.rendered) {return ''};
41717         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41718         
41719     },
41720     // private
41721     onViewMove : function(e, t){
41722         this.inKeyMode = false;
41723     },
41724
41725     // private
41726     onViewOver : function(e, t){
41727         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41728             return;
41729         }
41730         var item = this.view.findItemFromChild(t);
41731         if(item){
41732             var index = this.view.indexOf(item);
41733             this.select(index, false);
41734         }
41735     },
41736
41737     // private
41738     onViewClick : function(doFocus)
41739     {
41740         var index = this.view.getSelectedIndexes()[0];
41741         var r = this.store.getAt(index);
41742         if(r){
41743             this.onSelect(r, index);
41744         }
41745         if(doFocus !== false && !this.blockFocus){
41746             this.el.focus();
41747         }
41748     },
41749
41750     // private
41751     restrictHeight : function(){
41752         this.innerList.dom.style.height = '';
41753         var inner = this.innerList.dom;
41754         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41755         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41756         this.list.beginUpdate();
41757         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41758         this.list.alignTo(this.el, this.listAlign);
41759         this.list.endUpdate();
41760     },
41761
41762     // private
41763     onEmptyResults : function(){
41764         this.collapse();
41765     },
41766
41767     /**
41768      * Returns true if the dropdown list is expanded, else false.
41769      */
41770     isExpanded : function(){
41771         return this.list.isVisible();
41772     },
41773
41774     /**
41775      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41776      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41777      * @param {String} value The data value of the item to select
41778      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41779      * selected item if it is not currently in view (defaults to true)
41780      * @return {Boolean} True if the value matched an item in the list, else false
41781      */
41782     selectByValue : function(v, scrollIntoView){
41783         if(v !== undefined && v !== null){
41784             var r = this.findRecord(this.valueField || this.displayField, v);
41785             if(r){
41786                 this.select(this.store.indexOf(r), scrollIntoView);
41787                 return true;
41788             }
41789         }
41790         return false;
41791     },
41792
41793     /**
41794      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41795      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41796      * @param {Number} index The zero-based index of the list item to select
41797      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41798      * selected item if it is not currently in view (defaults to true)
41799      */
41800     select : function(index, scrollIntoView){
41801         this.selectedIndex = index;
41802         this.view.select(index);
41803         if(scrollIntoView !== false){
41804             var el = this.view.getNode(index);
41805             if(el){
41806                 this.innerList.scrollChildIntoView(el, false);
41807             }
41808         }
41809     },
41810
41811     // private
41812     selectNext : function(){
41813         var ct = this.store.getCount();
41814         if(ct > 0){
41815             if(this.selectedIndex == -1){
41816                 this.select(0);
41817             }else if(this.selectedIndex < ct-1){
41818                 this.select(this.selectedIndex+1);
41819             }
41820         }
41821     },
41822
41823     // private
41824     selectPrev : function(){
41825         var ct = this.store.getCount();
41826         if(ct > 0){
41827             if(this.selectedIndex == -1){
41828                 this.select(0);
41829             }else if(this.selectedIndex != 0){
41830                 this.select(this.selectedIndex-1);
41831             }
41832         }
41833     },
41834
41835     // private
41836     onKeyUp : function(e){
41837         if(this.editable !== false && !e.isSpecialKey()){
41838             this.lastKey = e.getKey();
41839             this.dqTask.delay(this.queryDelay);
41840         }
41841     },
41842
41843     // private
41844     validateBlur : function(){
41845         return !this.list || !this.list.isVisible();   
41846     },
41847
41848     // private
41849     initQuery : function(){
41850         this.doQuery(this.getRawValue());
41851     },
41852
41853     // private
41854     doForce : function(){
41855         if(this.el.dom.value.length > 0){
41856             this.el.dom.value =
41857                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
41858              
41859         }
41860     },
41861
41862     /**
41863      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
41864      * query allowing the query action to be canceled if needed.
41865      * @param {String} query The SQL query to execute
41866      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
41867      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
41868      * saved in the current store (defaults to false)
41869      */
41870     doQuery : function(q, forceAll){
41871         if(q === undefined || q === null){
41872             q = '';
41873         }
41874         var qe = {
41875             query: q,
41876             forceAll: forceAll,
41877             combo: this,
41878             cancel:false
41879         };
41880         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
41881             return false;
41882         }
41883         q = qe.query;
41884         forceAll = qe.forceAll;
41885         if(forceAll === true || (q.length >= this.minChars)){
41886             if(this.lastQuery != q || this.alwaysQuery){
41887                 this.lastQuery = q;
41888                 if(this.mode == 'local'){
41889                     this.selectedIndex = -1;
41890                     if(forceAll){
41891                         this.store.clearFilter();
41892                     }else{
41893                         this.store.filter(this.displayField, q);
41894                     }
41895                     this.onLoad();
41896                 }else{
41897                     this.store.baseParams[this.queryParam] = q;
41898                     this.store.load({
41899                         params: this.getParams(q)
41900                     });
41901                     this.expand();
41902                 }
41903             }else{
41904                 this.selectedIndex = -1;
41905                 this.onLoad();   
41906             }
41907         }
41908     },
41909
41910     // private
41911     getParams : function(q){
41912         var p = {};
41913         //p[this.queryParam] = q;
41914         if(this.pageSize){
41915             p.start = 0;
41916             p.limit = this.pageSize;
41917         }
41918         return p;
41919     },
41920
41921     /**
41922      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
41923      */
41924     collapse : function(){
41925         if(!this.isExpanded()){
41926             return;
41927         }
41928         this.list.hide();
41929         Roo.get(document).un('mousedown', this.collapseIf, this);
41930         Roo.get(document).un('mousewheel', this.collapseIf, this);
41931         if (!this.editable) {
41932             Roo.get(document).un('keydown', this.listKeyPress, this);
41933         }
41934         this.fireEvent('collapse', this);
41935     },
41936
41937     // private
41938     collapseIf : function(e){
41939         if(!e.within(this.wrap) && !e.within(this.list)){
41940             this.collapse();
41941         }
41942     },
41943
41944     /**
41945      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
41946      */
41947     expand : function(){
41948         if(this.isExpanded() || !this.hasFocus){
41949             return;
41950         }
41951         this.list.alignTo(this.el, this.listAlign);
41952         this.list.show();
41953         Roo.get(document).on('mousedown', this.collapseIf, this);
41954         Roo.get(document).on('mousewheel', this.collapseIf, this);
41955         if (!this.editable) {
41956             Roo.get(document).on('keydown', this.listKeyPress, this);
41957         }
41958         
41959         this.fireEvent('expand', this);
41960     },
41961
41962     // private
41963     // Implements the default empty TriggerField.onTriggerClick function
41964     onTriggerClick : function(){
41965         if(this.disabled){
41966             return;
41967         }
41968         if(this.isExpanded()){
41969             this.collapse();
41970             if (!this.blockFocus) {
41971                 this.el.focus();
41972             }
41973             
41974         }else {
41975             this.hasFocus = true;
41976             if(this.triggerAction == 'all') {
41977                 this.doQuery(this.allQuery, true);
41978             } else {
41979                 this.doQuery(this.getRawValue());
41980             }
41981             if (!this.blockFocus) {
41982                 this.el.focus();
41983             }
41984         }
41985     },
41986     listKeyPress : function(e)
41987     {
41988         //Roo.log('listkeypress');
41989         // scroll to first matching element based on key pres..
41990         if (e.isSpecialKey()) {
41991             return false;
41992         }
41993         var k = String.fromCharCode(e.getKey()).toUpperCase();
41994         //Roo.log(k);
41995         var match  = false;
41996         var csel = this.view.getSelectedNodes();
41997         var cselitem = false;
41998         if (csel.length) {
41999             var ix = this.view.indexOf(csel[0]);
42000             cselitem  = this.store.getAt(ix);
42001             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42002                 cselitem = false;
42003             }
42004             
42005         }
42006         
42007         this.store.each(function(v) { 
42008             if (cselitem) {
42009                 // start at existing selection.
42010                 if (cselitem.id == v.id) {
42011                     cselitem = false;
42012                 }
42013                 return;
42014             }
42015                 
42016             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42017                 match = this.store.indexOf(v);
42018                 return false;
42019             }
42020         }, this);
42021         
42022         if (match === false) {
42023             return true; // no more action?
42024         }
42025         // scroll to?
42026         this.view.select(match);
42027         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42028         sn.scrollIntoView(sn.dom.parentNode, false);
42029     }
42030
42031     /** 
42032     * @cfg {Boolean} grow 
42033     * @hide 
42034     */
42035     /** 
42036     * @cfg {Number} growMin 
42037     * @hide 
42038     */
42039     /** 
42040     * @cfg {Number} growMax 
42041     * @hide 
42042     */
42043     /**
42044      * @hide
42045      * @method autoSize
42046      */
42047 });/*
42048  * Copyright(c) 2010-2012, Roo J Solutions Limited
42049  *
42050  * Licence LGPL
42051  *
42052  */
42053
42054 /**
42055  * @class Roo.form.ComboBoxArray
42056  * @extends Roo.form.TextField
42057  * A facebook style adder... for lists of email / people / countries  etc...
42058  * pick multiple items from a combo box, and shows each one.
42059  *
42060  *  Fred [x]  Brian [x]  [Pick another |v]
42061  *
42062  *
42063  *  For this to work: it needs various extra information
42064  *    - normal combo problay has
42065  *      name, hiddenName
42066  *    + displayField, valueField
42067  *
42068  *    For our purpose...
42069  *
42070  *
42071  *   If we change from 'extends' to wrapping...
42072  *   
42073  *  
42074  *
42075  
42076  
42077  * @constructor
42078  * Create a new ComboBoxArray.
42079  * @param {Object} config Configuration options
42080  */
42081  
42082
42083 Roo.form.ComboBoxArray = function(config)
42084 {
42085     this.addEvents({
42086         /**
42087          * @event beforeremove
42088          * Fires before remove the value from the list
42089              * @param {Roo.form.ComboBoxArray} _self This combo box array
42090              * @param {Roo.form.ComboBoxArray.Item} item removed item
42091              */
42092         'beforeremove' : true,
42093         /**
42094          * @event remove
42095          * Fires when remove the value from the list
42096              * @param {Roo.form.ComboBoxArray} _self This combo box array
42097              * @param {Roo.form.ComboBoxArray.Item} item removed item
42098              */
42099         'remove' : true
42100         
42101         
42102     });
42103     
42104     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42105     
42106     this.items = new Roo.util.MixedCollection(false);
42107     
42108     // construct the child combo...
42109     
42110     
42111     
42112     
42113    
42114     
42115 }
42116
42117  
42118 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42119
42120     /**
42121      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42122      */
42123     
42124     lastData : false,
42125     
42126     // behavies liek a hiddne field
42127     inputType:      'hidden',
42128     /**
42129      * @cfg {Number} width The width of the box that displays the selected element
42130      */ 
42131     width:          300,
42132
42133     
42134     
42135     /**
42136      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42137      */
42138     name : false,
42139     /**
42140      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42141      */
42142     hiddenName : false,
42143     
42144     
42145     // private the array of items that are displayed..
42146     items  : false,
42147     // private - the hidden field el.
42148     hiddenEl : false,
42149     // private - the filed el..
42150     el : false,
42151     
42152     //validateValue : function() { return true; }, // all values are ok!
42153     //onAddClick: function() { },
42154     
42155     onRender : function(ct, position) 
42156     {
42157         
42158         // create the standard hidden element
42159         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42160         
42161         
42162         // give fake names to child combo;
42163         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42164         this.combo.name = this.name? (this.name+'-subcombo') : this.name;
42165         
42166         this.combo = Roo.factory(this.combo, Roo.form);
42167         this.combo.onRender(ct, position);
42168         if (typeof(this.combo.width) != 'undefined') {
42169             this.combo.onResize(this.combo.width,0);
42170         }
42171         
42172         this.combo.initEvents();
42173         
42174         // assigned so form know we need to do this..
42175         this.store          = this.combo.store;
42176         this.valueField     = this.combo.valueField;
42177         this.displayField   = this.combo.displayField ;
42178         
42179         
42180         this.combo.wrap.addClass('x-cbarray-grp');
42181         
42182         var cbwrap = this.combo.wrap.createChild(
42183             {tag: 'div', cls: 'x-cbarray-cb'},
42184             this.combo.el.dom
42185         );
42186         
42187              
42188         this.hiddenEl = this.combo.wrap.createChild({
42189             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42190         });
42191         this.el = this.combo.wrap.createChild({
42192             tag: 'input',  type:'hidden' , name: this.name, value : ''
42193         });
42194          //   this.el.dom.removeAttribute("name");
42195         
42196         
42197         this.outerWrap = this.combo.wrap;
42198         this.wrap = cbwrap;
42199         
42200         this.outerWrap.setWidth(this.width);
42201         this.outerWrap.dom.removeChild(this.el.dom);
42202         
42203         this.wrap.dom.appendChild(this.el.dom);
42204         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42205         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42206         
42207         this.combo.trigger.setStyle('position','relative');
42208         this.combo.trigger.setStyle('left', '0px');
42209         this.combo.trigger.setStyle('top', '2px');
42210         
42211         this.combo.el.setStyle('vertical-align', 'text-bottom');
42212         
42213         //this.trigger.setStyle('vertical-align', 'top');
42214         
42215         // this should use the code from combo really... on('add' ....)
42216         if (this.adder) {
42217             
42218         
42219             this.adder = this.outerWrap.createChild(
42220                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42221             var _t = this;
42222             this.adder.on('click', function(e) {
42223                 _t.fireEvent('adderclick', this, e);
42224             }, _t);
42225         }
42226         //var _t = this;
42227         //this.adder.on('click', this.onAddClick, _t);
42228         
42229         
42230         this.combo.on('select', function(cb, rec, ix) {
42231             this.addItem(rec.data);
42232             
42233             cb.setValue('');
42234             cb.el.dom.value = '';
42235             //cb.lastData = rec.data;
42236             // add to list
42237             
42238         }, this);
42239         
42240         
42241     },
42242     
42243     
42244     getName: function()
42245     {
42246         // returns hidden if it's set..
42247         if (!this.rendered) {return ''};
42248         return  this.hiddenName ? this.hiddenName : this.name;
42249         
42250     },
42251     
42252     
42253     onResize: function(w, h){
42254         
42255         return;
42256         // not sure if this is needed..
42257         //this.combo.onResize(w,h);
42258         
42259         if(typeof w != 'number'){
42260             // we do not handle it!?!?
42261             return;
42262         }
42263         var tw = this.combo.trigger.getWidth();
42264         tw += this.addicon ? this.addicon.getWidth() : 0;
42265         tw += this.editicon ? this.editicon.getWidth() : 0;
42266         var x = w - tw;
42267         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42268             
42269         this.combo.trigger.setStyle('left', '0px');
42270         
42271         if(this.list && this.listWidth === undefined){
42272             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42273             this.list.setWidth(lw);
42274             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42275         }
42276         
42277     
42278         
42279     },
42280     
42281     addItem: function(rec)
42282     {
42283         var valueField = this.combo.valueField;
42284         var displayField = this.combo.displayField;
42285         if (this.items.indexOfKey(rec[valueField]) > -1) {
42286             //console.log("GOT " + rec.data.id);
42287             return;
42288         }
42289         
42290         var x = new Roo.form.ComboBoxArray.Item({
42291             //id : rec[this.idField],
42292             data : rec,
42293             displayField : displayField ,
42294             tipField : displayField ,
42295             cb : this
42296         });
42297         // use the 
42298         this.items.add(rec[valueField],x);
42299         // add it before the element..
42300         this.updateHiddenEl();
42301         x.render(this.outerWrap, this.wrap.dom);
42302         // add the image handler..
42303     },
42304     
42305     updateHiddenEl : function()
42306     {
42307         this.validate();
42308         if (!this.hiddenEl) {
42309             return;
42310         }
42311         var ar = [];
42312         var idField = this.combo.valueField;
42313         
42314         this.items.each(function(f) {
42315             ar.push(f.data[idField]);
42316            
42317         });
42318         this.hiddenEl.dom.value = ar.join(',');
42319         this.validate();
42320     },
42321     
42322     reset : function()
42323     {
42324         this.items.clear();
42325         
42326         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42327            el.remove();
42328         });
42329         
42330         this.el.dom.value = '';
42331         if (this.hiddenEl) {
42332             this.hiddenEl.dom.value = '';
42333         }
42334         
42335     },
42336     getValue: function()
42337     {
42338         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42339     },
42340     setValue: function(v) // not a valid action - must use addItems..
42341     {
42342          
42343         this.reset();
42344         
42345         
42346         
42347         if (this.store.isLocal && (typeof(v) == 'string')) {
42348             // then we can use the store to find the values..
42349             // comma seperated at present.. this needs to allow JSON based encoding..
42350             this.hiddenEl.value  = v;
42351             var v_ar = [];
42352             Roo.each(v.split(','), function(k) {
42353                 Roo.log("CHECK " + this.valueField + ',' + k);
42354                 var li = this.store.query(this.valueField, k);
42355                 if (!li.length) {
42356                     return;
42357                 }
42358                 var add = {};
42359                 add[this.valueField] = k;
42360                 add[this.displayField] = li.item(0).data[this.displayField];
42361                 
42362                 this.addItem(add);
42363             }, this) 
42364              
42365         }
42366         if (typeof(v) == 'object' ) {
42367             // then let's assume it's an array of objects..
42368             Roo.each(v, function(l) {
42369                 this.addItem(l);
42370             }, this);
42371              
42372         }
42373         
42374         
42375     },
42376     setFromData: function(v)
42377     {
42378         // this recieves an object, if setValues is called.
42379         this.reset();
42380         this.el.dom.value = v[this.displayField];
42381         this.hiddenEl.dom.value = v[this.valueField];
42382         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42383             return;
42384         }
42385         var kv = v[this.valueField];
42386         var dv = v[this.displayField];
42387         kv = typeof(kv) != 'string' ? '' : kv;
42388         dv = typeof(dv) != 'string' ? '' : dv;
42389         
42390         
42391         var keys = kv.split(',');
42392         var display = dv.split(',');
42393         for (var i = 0 ; i < keys.length; i++) {
42394             
42395             add = {};
42396             add[this.valueField] = keys[i];
42397             add[this.displayField] = display[i];
42398             this.addItem(add);
42399         }
42400       
42401         
42402     },
42403     
42404     /**
42405      * Validates the combox array value
42406      * @return {Boolean} True if the value is valid, else false
42407      */
42408     validate : function(){
42409         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42410             this.clearInvalid();
42411             return true;
42412         }
42413         return false;
42414     },
42415     
42416     validateValue : function(value){
42417         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42418         
42419     },
42420     
42421     /*@
42422      * overide
42423      * 
42424      */
42425     isDirty : function() {
42426         if(this.disabled) {
42427             return false;
42428         }
42429         
42430         try {
42431             var d = Roo.decode(String(this.originalValue));
42432         } catch (e) {
42433             return String(this.getValue()) !== String(this.originalValue);
42434         }
42435         
42436         var originalValue = [];
42437         
42438         for (var i = 0; i < d.length; i++){
42439             originalValue.push(d[i][this.valueField]);
42440         }
42441         
42442         return String(this.getValue()) !== String(originalValue.join(','));
42443         
42444     }
42445     
42446 });
42447
42448
42449
42450 /**
42451  * @class Roo.form.ComboBoxArray.Item
42452  * @extends Roo.BoxComponent
42453  * A selected item in the list
42454  *  Fred [x]  Brian [x]  [Pick another |v]
42455  * 
42456  * @constructor
42457  * Create a new item.
42458  * @param {Object} config Configuration options
42459  */
42460  
42461 Roo.form.ComboBoxArray.Item = function(config) {
42462     config.id = Roo.id();
42463     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42464 }
42465
42466 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42467     data : {},
42468     cb: false,
42469     displayField : false,
42470     tipField : false,
42471     
42472     
42473     defaultAutoCreate : {
42474         tag: 'div',
42475         cls: 'x-cbarray-item',
42476         cn : [ 
42477             { tag: 'div' },
42478             {
42479                 tag: 'img',
42480                 width:16,
42481                 height : 16,
42482                 src : Roo.BLANK_IMAGE_URL ,
42483                 align: 'center'
42484             }
42485         ]
42486         
42487     },
42488     
42489  
42490     onRender : function(ct, position)
42491     {
42492         Roo.form.Field.superclass.onRender.call(this, ct, position);
42493         
42494         if(!this.el){
42495             var cfg = this.getAutoCreate();
42496             this.el = ct.createChild(cfg, position);
42497         }
42498         
42499         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42500         
42501         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42502             this.cb.renderer(this.data) :
42503             String.format('{0}',this.data[this.displayField]);
42504         
42505             
42506         this.el.child('div').dom.setAttribute('qtip',
42507                         String.format('{0}',this.data[this.tipField])
42508         );
42509         
42510         this.el.child('img').on('click', this.remove, this);
42511         
42512     },
42513    
42514     remove : function()
42515     {
42516         if(this.cb.disabled){
42517             return;
42518         }
42519         
42520         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42521             this.cb.items.remove(this);
42522             this.el.child('img').un('click', this.remove, this);
42523             this.el.remove();
42524             this.cb.updateHiddenEl();
42525
42526             this.cb.fireEvent('remove', this.cb, this);
42527         }
42528         
42529     }
42530 });/*
42531  * Based on:
42532  * Ext JS Library 1.1.1
42533  * Copyright(c) 2006-2007, Ext JS, LLC.
42534  *
42535  * Originally Released Under LGPL - original licence link has changed is not relivant.
42536  *
42537  * Fork - LGPL
42538  * <script type="text/javascript">
42539  */
42540 /**
42541  * @class Roo.form.Checkbox
42542  * @extends Roo.form.Field
42543  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
42544  * @constructor
42545  * Creates a new Checkbox
42546  * @param {Object} config Configuration options
42547  */
42548 Roo.form.Checkbox = function(config){
42549     Roo.form.Checkbox.superclass.constructor.call(this, config);
42550     this.addEvents({
42551         /**
42552          * @event check
42553          * Fires when the checkbox is checked or unchecked.
42554              * @param {Roo.form.Checkbox} this This checkbox
42555              * @param {Boolean} checked The new checked value
42556              */
42557         check : true
42558     });
42559 };
42560
42561 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
42562     /**
42563      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
42564      */
42565     focusClass : undefined,
42566     /**
42567      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
42568      */
42569     fieldClass: "x-form-field",
42570     /**
42571      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
42572      */
42573     checked: false,
42574     /**
42575      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
42576      * {tag: "input", type: "checkbox", autocomplete: "off"})
42577      */
42578     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
42579     /**
42580      * @cfg {String} boxLabel The text that appears beside the checkbox
42581      */
42582     boxLabel : "",
42583     /**
42584      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
42585      */  
42586     inputValue : '1',
42587     /**
42588      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
42589      */
42590      valueOff: '0', // value when not checked..
42591
42592     actionMode : 'viewEl', 
42593     //
42594     // private
42595     itemCls : 'x-menu-check-item x-form-item',
42596     groupClass : 'x-menu-group-item',
42597     inputType : 'hidden',
42598     
42599     
42600     inSetChecked: false, // check that we are not calling self...
42601     
42602     inputElement: false, // real input element?
42603     basedOn: false, // ????
42604     
42605     isFormField: true, // not sure where this is needed!!!!
42606
42607     onResize : function(){
42608         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
42609         if(!this.boxLabel){
42610             this.el.alignTo(this.wrap, 'c-c');
42611         }
42612     },
42613
42614     initEvents : function(){
42615         Roo.form.Checkbox.superclass.initEvents.call(this);
42616         this.el.on("click", this.onClick,  this);
42617         this.el.on("change", this.onClick,  this);
42618     },
42619
42620
42621     getResizeEl : function(){
42622         return this.wrap;
42623     },
42624
42625     getPositionEl : function(){
42626         return this.wrap;
42627     },
42628
42629     // private
42630     onRender : function(ct, position){
42631         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42632         /*
42633         if(this.inputValue !== undefined){
42634             this.el.dom.value = this.inputValue;
42635         }
42636         */
42637         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42638         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42639         var viewEl = this.wrap.createChild({ 
42640             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42641         this.viewEl = viewEl;   
42642         this.wrap.on('click', this.onClick,  this); 
42643         
42644         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42645         this.el.on('propertychange', this.setFromHidden,  this);  //ie
42646         
42647         
42648         
42649         if(this.boxLabel){
42650             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42651         //    viewEl.on('click', this.onClick,  this); 
42652         }
42653         //if(this.checked){
42654             this.setChecked(this.checked);
42655         //}else{
42656             //this.checked = this.el.dom;
42657         //}
42658
42659     },
42660
42661     // private
42662     initValue : Roo.emptyFn,
42663
42664     /**
42665      * Returns the checked state of the checkbox.
42666      * @return {Boolean} True if checked, else false
42667      */
42668     getValue : function(){
42669         if(this.el){
42670             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
42671         }
42672         return this.valueOff;
42673         
42674     },
42675
42676         // private
42677     onClick : function(){ 
42678         if (this.disabled) {
42679             return;
42680         }
42681         this.setChecked(!this.checked);
42682
42683         //if(this.el.dom.checked != this.checked){
42684         //    this.setValue(this.el.dom.checked);
42685        // }
42686     },
42687
42688     /**
42689      * Sets the checked state of the checkbox.
42690      * On is always based on a string comparison between inputValue and the param.
42691      * @param {Boolean/String} value - the value to set 
42692      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
42693      */
42694     setValue : function(v,suppressEvent){
42695         
42696         
42697         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
42698         //if(this.el && this.el.dom){
42699         //    this.el.dom.checked = this.checked;
42700         //    this.el.dom.defaultChecked = this.checked;
42701         //}
42702         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
42703         //this.fireEvent("check", this, this.checked);
42704     },
42705     // private..
42706     setChecked : function(state,suppressEvent)
42707     {
42708         if (this.inSetChecked) {
42709             this.checked = state;
42710             return;
42711         }
42712         
42713     
42714         if(this.wrap){
42715             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
42716         }
42717         this.checked = state;
42718         if(suppressEvent !== true){
42719             this.fireEvent('check', this, state);
42720         }
42721         this.inSetChecked = true;
42722         this.el.dom.value = state ? this.inputValue : this.valueOff;
42723         this.inSetChecked = false;
42724         
42725     },
42726     // handle setting of hidden value by some other method!!?!?
42727     setFromHidden: function()
42728     {
42729         if(!this.el){
42730             return;
42731         }
42732         //console.log("SET FROM HIDDEN");
42733         //alert('setFrom hidden');
42734         this.setValue(this.el.dom.value);
42735     },
42736     
42737     onDestroy : function()
42738     {
42739         if(this.viewEl){
42740             Roo.get(this.viewEl).remove();
42741         }
42742          
42743         Roo.form.Checkbox.superclass.onDestroy.call(this);
42744     },
42745     
42746     setBoxLabel : function(str)
42747     {
42748         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
42749     }
42750
42751 });/*
42752  * Based on:
42753  * Ext JS Library 1.1.1
42754  * Copyright(c) 2006-2007, Ext JS, LLC.
42755  *
42756  * Originally Released Under LGPL - original licence link has changed is not relivant.
42757  *
42758  * Fork - LGPL
42759  * <script type="text/javascript">
42760  */
42761  
42762 /**
42763  * @class Roo.form.Radio
42764  * @extends Roo.form.Checkbox
42765  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
42766  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
42767  * @constructor
42768  * Creates a new Radio
42769  * @param {Object} config Configuration options
42770  */
42771 Roo.form.Radio = function(){
42772     Roo.form.Radio.superclass.constructor.apply(this, arguments);
42773 };
42774 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
42775     inputType: 'radio',
42776
42777     /**
42778      * If this radio is part of a group, it will return the selected value
42779      * @return {String}
42780      */
42781     getGroupValue : function(){
42782         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
42783     },
42784     
42785     
42786     onRender : function(ct, position){
42787         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
42788         
42789         if(this.inputValue !== undefined){
42790             this.el.dom.value = this.inputValue;
42791         }
42792          
42793         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
42794         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
42795         //var viewEl = this.wrap.createChild({ 
42796         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
42797         //this.viewEl = viewEl;   
42798         //this.wrap.on('click', this.onClick,  this); 
42799         
42800         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
42801         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
42802         
42803         
42804         
42805         if(this.boxLabel){
42806             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
42807         //    viewEl.on('click', this.onClick,  this); 
42808         }
42809          if(this.checked){
42810             this.el.dom.checked =   'checked' ;
42811         }
42812          
42813     } 
42814     
42815     
42816 });//<script type="text/javascript">
42817
42818 /*
42819  * Based  Ext JS Library 1.1.1
42820  * Copyright(c) 2006-2007, Ext JS, LLC.
42821  * LGPL
42822  *
42823  */
42824  
42825 /**
42826  * @class Roo.HtmlEditorCore
42827  * @extends Roo.Component
42828  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
42829  *
42830  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
42831  */
42832
42833 Roo.HtmlEditorCore = function(config){
42834     
42835     
42836     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
42837     
42838     
42839     this.addEvents({
42840         /**
42841          * @event initialize
42842          * Fires when the editor is fully initialized (including the iframe)
42843          * @param {Roo.HtmlEditorCore} this
42844          */
42845         initialize: true,
42846         /**
42847          * @event activate
42848          * Fires when the editor is first receives the focus. Any insertion must wait
42849          * until after this event.
42850          * @param {Roo.HtmlEditorCore} this
42851          */
42852         activate: true,
42853          /**
42854          * @event beforesync
42855          * Fires before the textarea is updated with content from the editor iframe. Return false
42856          * to cancel the sync.
42857          * @param {Roo.HtmlEditorCore} this
42858          * @param {String} html
42859          */
42860         beforesync: true,
42861          /**
42862          * @event beforepush
42863          * Fires before the iframe editor is updated with content from the textarea. Return false
42864          * to cancel the push.
42865          * @param {Roo.HtmlEditorCore} this
42866          * @param {String} html
42867          */
42868         beforepush: true,
42869          /**
42870          * @event sync
42871          * Fires when the textarea is updated with content from the editor iframe.
42872          * @param {Roo.HtmlEditorCore} this
42873          * @param {String} html
42874          */
42875         sync: true,
42876          /**
42877          * @event push
42878          * Fires when the iframe editor is updated with content from the textarea.
42879          * @param {Roo.HtmlEditorCore} this
42880          * @param {String} html
42881          */
42882         push: true,
42883         
42884         /**
42885          * @event editorevent
42886          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
42887          * @param {Roo.HtmlEditorCore} this
42888          */
42889         editorevent: true
42890         
42891     });
42892     
42893     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
42894     
42895     // defaults : white / black...
42896     this.applyBlacklists();
42897     
42898     
42899     
42900 };
42901
42902
42903 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
42904
42905
42906      /**
42907      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
42908      */
42909     
42910     owner : false,
42911     
42912      /**
42913      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
42914      *                        Roo.resizable.
42915      */
42916     resizable : false,
42917      /**
42918      * @cfg {Number} height (in pixels)
42919      */   
42920     height: 300,
42921    /**
42922      * @cfg {Number} width (in pixels)
42923      */   
42924     width: 500,
42925     
42926     /**
42927      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
42928      * 
42929      */
42930     stylesheets: false,
42931     
42932     // id of frame..
42933     frameId: false,
42934     
42935     // private properties
42936     validationEvent : false,
42937     deferHeight: true,
42938     initialized : false,
42939     activated : false,
42940     sourceEditMode : false,
42941     onFocus : Roo.emptyFn,
42942     iframePad:3,
42943     hideMode:'offsets',
42944     
42945     clearUp: true,
42946     
42947     // blacklist + whitelisted elements..
42948     black: false,
42949     white: false,
42950      
42951     bodyCls : '',
42952
42953     /**
42954      * Protected method that will not generally be called directly. It
42955      * is called when the editor initializes the iframe with HTML contents. Override this method if you
42956      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
42957      */
42958     getDocMarkup : function(){
42959         // body styles..
42960         var st = '';
42961         
42962         // inherit styels from page...?? 
42963         if (this.stylesheets === false) {
42964             
42965             Roo.get(document.head).select('style').each(function(node) {
42966                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
42967             });
42968             
42969             Roo.get(document.head).select('link').each(function(node) { 
42970                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
42971             });
42972             
42973         } else if (!this.stylesheets.length) {
42974                 // simple..
42975                 st = '<style type="text/css">' +
42976                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
42977                    '</style>';
42978         } else { 
42979             st = '<style type="text/css">' +
42980                     this.stylesheets +
42981                 '</style>';
42982         }
42983         
42984         st +=  '<style type="text/css">' +
42985             'IMG { cursor: pointer } ' +
42986         '</style>';
42987
42988         var cls = 'roo-htmleditor-body';
42989         
42990         if(this.bodyCls.length){
42991             cls += ' ' + this.bodyCls;
42992         }
42993         
42994         return '<html><head>' + st  +
42995             //<style type="text/css">' +
42996             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
42997             //'</style>' +
42998             ' </head><body class="' +  cls + '"></body></html>';
42999     },
43000
43001     // private
43002     onRender : function(ct, position)
43003     {
43004         var _t = this;
43005         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43006         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43007         
43008         
43009         this.el.dom.style.border = '0 none';
43010         this.el.dom.setAttribute('tabIndex', -1);
43011         this.el.addClass('x-hidden hide');
43012         
43013         
43014         
43015         if(Roo.isIE){ // fix IE 1px bogus margin
43016             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43017         }
43018        
43019         
43020         this.frameId = Roo.id();
43021         
43022          
43023         
43024         var iframe = this.owner.wrap.createChild({
43025             tag: 'iframe',
43026             cls: 'form-control', // bootstrap..
43027             id: this.frameId,
43028             name: this.frameId,
43029             frameBorder : 'no',
43030             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43031         }, this.el
43032         );
43033         
43034         
43035         this.iframe = iframe.dom;
43036
43037          this.assignDocWin();
43038         
43039         this.doc.designMode = 'on';
43040        
43041         this.doc.open();
43042         this.doc.write(this.getDocMarkup());
43043         this.doc.close();
43044
43045         
43046         var task = { // must defer to wait for browser to be ready
43047             run : function(){
43048                 //console.log("run task?" + this.doc.readyState);
43049                 this.assignDocWin();
43050                 if(this.doc.body || this.doc.readyState == 'complete'){
43051                     try {
43052                         this.doc.designMode="on";
43053                     } catch (e) {
43054                         return;
43055                     }
43056                     Roo.TaskMgr.stop(task);
43057                     this.initEditor.defer(10, this);
43058                 }
43059             },
43060             interval : 10,
43061             duration: 10000,
43062             scope: this
43063         };
43064         Roo.TaskMgr.start(task);
43065
43066     },
43067
43068     // private
43069     onResize : function(w, h)
43070     {
43071          Roo.log('resize: ' +w + ',' + h );
43072         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43073         if(!this.iframe){
43074             return;
43075         }
43076         if(typeof w == 'number'){
43077             
43078             this.iframe.style.width = w + 'px';
43079         }
43080         if(typeof h == 'number'){
43081             
43082             this.iframe.style.height = h + 'px';
43083             if(this.doc){
43084                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43085             }
43086         }
43087         
43088     },
43089
43090     /**
43091      * Toggles the editor between standard and source edit mode.
43092      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43093      */
43094     toggleSourceEdit : function(sourceEditMode){
43095         
43096         this.sourceEditMode = sourceEditMode === true;
43097         
43098         if(this.sourceEditMode){
43099  
43100             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43101             
43102         }else{
43103             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43104             //this.iframe.className = '';
43105             this.deferFocus();
43106         }
43107         //this.setSize(this.owner.wrap.getSize());
43108         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43109     },
43110
43111     
43112   
43113
43114     /**
43115      * Protected method that will not generally be called directly. If you need/want
43116      * custom HTML cleanup, this is the method you should override.
43117      * @param {String} html The HTML to be cleaned
43118      * return {String} The cleaned HTML
43119      */
43120     cleanHtml : function(html){
43121         html = String(html);
43122         if(html.length > 5){
43123             if(Roo.isSafari){ // strip safari nonsense
43124                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43125             }
43126         }
43127         if(html == '&nbsp;'){
43128             html = '';
43129         }
43130         return html;
43131     },
43132
43133     /**
43134      * HTML Editor -> Textarea
43135      * Protected method that will not generally be called directly. Syncs the contents
43136      * of the editor iframe with the textarea.
43137      */
43138     syncValue : function(){
43139         if(this.initialized){
43140             var bd = (this.doc.body || this.doc.documentElement);
43141             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43142             var html = bd.innerHTML;
43143             if(Roo.isSafari){
43144                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43145                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43146                 if(m && m[1]){
43147                     html = '<div style="'+m[0]+'">' + html + '</div>';
43148                 }
43149             }
43150             html = this.cleanHtml(html);
43151             // fix up the special chars.. normaly like back quotes in word...
43152             // however we do not want to do this with chinese..
43153             html = html.replace(/([\x80-\uffff])/g, function (a, b) {
43154                 var cc = b.charCodeAt();
43155                 if (
43156                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43157                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43158                     (cc >= 0xf900 && cc < 0xfb00 )
43159                 ) {
43160                         return b;
43161                 }
43162                 return "&#"+cc+";" 
43163             });
43164             if(this.owner.fireEvent('beforesync', this, html) !== false){
43165                 this.el.dom.value = html;
43166                 this.owner.fireEvent('sync', this, html);
43167             }
43168         }
43169     },
43170
43171     /**
43172      * Protected method that will not generally be called directly. Pushes the value of the textarea
43173      * into the iframe editor.
43174      */
43175     pushValue : function(){
43176         if(this.initialized){
43177             var v = this.el.dom.value.trim();
43178             
43179 //            if(v.length < 1){
43180 //                v = '&#160;';
43181 //            }
43182             
43183             if(this.owner.fireEvent('beforepush', this, v) !== false){
43184                 var d = (this.doc.body || this.doc.documentElement);
43185                 d.innerHTML = v;
43186                 this.cleanUpPaste();
43187                 this.el.dom.value = d.innerHTML;
43188                 this.owner.fireEvent('push', this, v);
43189             }
43190         }
43191     },
43192
43193     // private
43194     deferFocus : function(){
43195         this.focus.defer(10, this);
43196     },
43197
43198     // doc'ed in Field
43199     focus : function(){
43200         if(this.win && !this.sourceEditMode){
43201             this.win.focus();
43202         }else{
43203             this.el.focus();
43204         }
43205     },
43206     
43207     assignDocWin: function()
43208     {
43209         var iframe = this.iframe;
43210         
43211          if(Roo.isIE){
43212             this.doc = iframe.contentWindow.document;
43213             this.win = iframe.contentWindow;
43214         } else {
43215 //            if (!Roo.get(this.frameId)) {
43216 //                return;
43217 //            }
43218 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43219 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43220             
43221             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43222                 return;
43223             }
43224             
43225             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43226             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43227         }
43228     },
43229     
43230     // private
43231     initEditor : function(){
43232         //console.log("INIT EDITOR");
43233         this.assignDocWin();
43234         
43235         
43236         
43237         this.doc.designMode="on";
43238         this.doc.open();
43239         this.doc.write(this.getDocMarkup());
43240         this.doc.close();
43241         
43242         var dbody = (this.doc.body || this.doc.documentElement);
43243         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43244         // this copies styles from the containing element into thsi one..
43245         // not sure why we need all of this..
43246         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43247         
43248         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43249         //ss['background-attachment'] = 'fixed'; // w3c
43250         dbody.bgProperties = 'fixed'; // ie
43251         //Roo.DomHelper.applyStyles(dbody, ss);
43252         Roo.EventManager.on(this.doc, {
43253             //'mousedown': this.onEditorEvent,
43254             'mouseup': this.onEditorEvent,
43255             'dblclick': this.onEditorEvent,
43256             'click': this.onEditorEvent,
43257             'keyup': this.onEditorEvent,
43258             buffer:100,
43259             scope: this
43260         });
43261         if(Roo.isGecko){
43262             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43263         }
43264         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43265             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43266         }
43267         this.initialized = true;
43268
43269         this.owner.fireEvent('initialize', this);
43270         this.pushValue();
43271     },
43272
43273     // private
43274     onDestroy : function(){
43275         
43276         
43277         
43278         if(this.rendered){
43279             
43280             //for (var i =0; i < this.toolbars.length;i++) {
43281             //    // fixme - ask toolbars for heights?
43282             //    this.toolbars[i].onDestroy();
43283            // }
43284             
43285             //this.wrap.dom.innerHTML = '';
43286             //this.wrap.remove();
43287         }
43288     },
43289
43290     // private
43291     onFirstFocus : function(){
43292         
43293         this.assignDocWin();
43294         
43295         
43296         this.activated = true;
43297          
43298     
43299         if(Roo.isGecko){ // prevent silly gecko errors
43300             this.win.focus();
43301             var s = this.win.getSelection();
43302             if(!s.focusNode || s.focusNode.nodeType != 3){
43303                 var r = s.getRangeAt(0);
43304                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43305                 r.collapse(true);
43306                 this.deferFocus();
43307             }
43308             try{
43309                 this.execCmd('useCSS', true);
43310                 this.execCmd('styleWithCSS', false);
43311             }catch(e){}
43312         }
43313         this.owner.fireEvent('activate', this);
43314     },
43315
43316     // private
43317     adjustFont: function(btn){
43318         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43319         //if(Roo.isSafari){ // safari
43320         //    adjust *= 2;
43321        // }
43322         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43323         if(Roo.isSafari){ // safari
43324             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43325             v =  (v < 10) ? 10 : v;
43326             v =  (v > 48) ? 48 : v;
43327             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43328             
43329         }
43330         
43331         
43332         v = Math.max(1, v+adjust);
43333         
43334         this.execCmd('FontSize', v  );
43335     },
43336
43337     onEditorEvent : function(e)
43338     {
43339         this.owner.fireEvent('editorevent', this, e);
43340       //  this.updateToolbar();
43341         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43342     },
43343
43344     insertTag : function(tg)
43345     {
43346         // could be a bit smarter... -> wrap the current selected tRoo..
43347         if (tg.toLowerCase() == 'span' || tg.toLowerCase() == 'code') {
43348             
43349             range = this.createRange(this.getSelection());
43350             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43351             wrappingNode.appendChild(range.extractContents());
43352             range.insertNode(wrappingNode);
43353
43354             return;
43355             
43356             
43357             
43358         }
43359         this.execCmd("formatblock",   tg);
43360         
43361     },
43362     
43363     insertText : function(txt)
43364     {
43365         
43366         
43367         var range = this.createRange();
43368         range.deleteContents();
43369                //alert(Sender.getAttribute('label'));
43370                
43371         range.insertNode(this.doc.createTextNode(txt));
43372     } ,
43373     
43374      
43375
43376     /**
43377      * Executes a Midas editor command on the editor document and performs necessary focus and
43378      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43379      * @param {String} cmd The Midas command
43380      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43381      */
43382     relayCmd : function(cmd, value){
43383         this.win.focus();
43384         this.execCmd(cmd, value);
43385         this.owner.fireEvent('editorevent', this);
43386         //this.updateToolbar();
43387         this.owner.deferFocus();
43388     },
43389
43390     /**
43391      * Executes a Midas editor command directly on the editor document.
43392      * For visual commands, you should use {@link #relayCmd} instead.
43393      * <b>This should only be called after the editor is initialized.</b>
43394      * @param {String} cmd The Midas command
43395      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43396      */
43397     execCmd : function(cmd, value){
43398         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43399         this.syncValue();
43400     },
43401  
43402  
43403    
43404     /**
43405      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43406      * to insert tRoo.
43407      * @param {String} text | dom node.. 
43408      */
43409     insertAtCursor : function(text)
43410     {
43411         
43412         if(!this.activated){
43413             return;
43414         }
43415         /*
43416         if(Roo.isIE){
43417             this.win.focus();
43418             var r = this.doc.selection.createRange();
43419             if(r){
43420                 r.collapse(true);
43421                 r.pasteHTML(text);
43422                 this.syncValue();
43423                 this.deferFocus();
43424             
43425             }
43426             return;
43427         }
43428         */
43429         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43430             this.win.focus();
43431             
43432             
43433             // from jquery ui (MIT licenced)
43434             var range, node;
43435             var win = this.win;
43436             
43437             if (win.getSelection && win.getSelection().getRangeAt) {
43438                 range = win.getSelection().getRangeAt(0);
43439                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43440                 range.insertNode(node);
43441             } else if (win.document.selection && win.document.selection.createRange) {
43442                 // no firefox support
43443                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43444                 win.document.selection.createRange().pasteHTML(txt);
43445             } else {
43446                 // no firefox support
43447                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43448                 this.execCmd('InsertHTML', txt);
43449             } 
43450             
43451             this.syncValue();
43452             
43453             this.deferFocus();
43454         }
43455     },
43456  // private
43457     mozKeyPress : function(e){
43458         if(e.ctrlKey){
43459             var c = e.getCharCode(), cmd;
43460           
43461             if(c > 0){
43462                 c = String.fromCharCode(c).toLowerCase();
43463                 switch(c){
43464                     case 'b':
43465                         cmd = 'bold';
43466                         break;
43467                     case 'i':
43468                         cmd = 'italic';
43469                         break;
43470                     
43471                     case 'u':
43472                         cmd = 'underline';
43473                         break;
43474                     
43475                     case 'v':
43476                         this.cleanUpPaste.defer(100, this);
43477                         return;
43478                         
43479                 }
43480                 if(cmd){
43481                     this.win.focus();
43482                     this.execCmd(cmd);
43483                     this.deferFocus();
43484                     e.preventDefault();
43485                 }
43486                 
43487             }
43488         }
43489     },
43490
43491     // private
43492     fixKeys : function(){ // load time branching for fastest keydown performance
43493         if(Roo.isIE){
43494             return function(e){
43495                 var k = e.getKey(), r;
43496                 if(k == e.TAB){
43497                     e.stopEvent();
43498                     r = this.doc.selection.createRange();
43499                     if(r){
43500                         r.collapse(true);
43501                         r.pasteHTML('&#160;&#160;&#160;&#160;');
43502                         this.deferFocus();
43503                     }
43504                     return;
43505                 }
43506                 
43507                 if(k == e.ENTER){
43508                     r = this.doc.selection.createRange();
43509                     if(r){
43510                         var target = r.parentElement();
43511                         if(!target || target.tagName.toLowerCase() != 'li'){
43512                             e.stopEvent();
43513                             r.pasteHTML('<br />');
43514                             r.collapse(false);
43515                             r.select();
43516                         }
43517                     }
43518                 }
43519                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43520                     this.cleanUpPaste.defer(100, this);
43521                     return;
43522                 }
43523                 
43524                 
43525             };
43526         }else if(Roo.isOpera){
43527             return function(e){
43528                 var k = e.getKey();
43529                 if(k == e.TAB){
43530                     e.stopEvent();
43531                     this.win.focus();
43532                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
43533                     this.deferFocus();
43534                 }
43535                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43536                     this.cleanUpPaste.defer(100, this);
43537                     return;
43538                 }
43539                 
43540             };
43541         }else if(Roo.isSafari){
43542             return function(e){
43543                 var k = e.getKey();
43544                 
43545                 if(k == e.TAB){
43546                     e.stopEvent();
43547                     this.execCmd('InsertText','\t');
43548                     this.deferFocus();
43549                     return;
43550                 }
43551                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
43552                     this.cleanUpPaste.defer(100, this);
43553                     return;
43554                 }
43555                 
43556              };
43557         }
43558     }(),
43559     
43560     getAllAncestors: function()
43561     {
43562         var p = this.getSelectedNode();
43563         var a = [];
43564         if (!p) {
43565             a.push(p); // push blank onto stack..
43566             p = this.getParentElement();
43567         }
43568         
43569         
43570         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
43571             a.push(p);
43572             p = p.parentNode;
43573         }
43574         a.push(this.doc.body);
43575         return a;
43576     },
43577     lastSel : false,
43578     lastSelNode : false,
43579     
43580     
43581     getSelection : function() 
43582     {
43583         this.assignDocWin();
43584         return Roo.isIE ? this.doc.selection : this.win.getSelection();
43585     },
43586     
43587     getSelectedNode: function() 
43588     {
43589         // this may only work on Gecko!!!
43590         
43591         // should we cache this!!!!
43592         
43593         
43594         
43595          
43596         var range = this.createRange(this.getSelection()).cloneRange();
43597         
43598         if (Roo.isIE) {
43599             var parent = range.parentElement();
43600             while (true) {
43601                 var testRange = range.duplicate();
43602                 testRange.moveToElementText(parent);
43603                 if (testRange.inRange(range)) {
43604                     break;
43605                 }
43606                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
43607                     break;
43608                 }
43609                 parent = parent.parentElement;
43610             }
43611             return parent;
43612         }
43613         
43614         // is ancestor a text element.
43615         var ac =  range.commonAncestorContainer;
43616         if (ac.nodeType == 3) {
43617             ac = ac.parentNode;
43618         }
43619         
43620         var ar = ac.childNodes;
43621          
43622         var nodes = [];
43623         var other_nodes = [];
43624         var has_other_nodes = false;
43625         for (var i=0;i<ar.length;i++) {
43626             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
43627                 continue;
43628             }
43629             // fullly contained node.
43630             
43631             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
43632                 nodes.push(ar[i]);
43633                 continue;
43634             }
43635             
43636             // probably selected..
43637             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
43638                 other_nodes.push(ar[i]);
43639                 continue;
43640             }
43641             // outer..
43642             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
43643                 continue;
43644             }
43645             
43646             
43647             has_other_nodes = true;
43648         }
43649         if (!nodes.length && other_nodes.length) {
43650             nodes= other_nodes;
43651         }
43652         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
43653             return false;
43654         }
43655         
43656         return nodes[0];
43657     },
43658     createRange: function(sel)
43659     {
43660         // this has strange effects when using with 
43661         // top toolbar - not sure if it's a great idea.
43662         //this.editor.contentWindow.focus();
43663         if (typeof sel != "undefined") {
43664             try {
43665                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
43666             } catch(e) {
43667                 return this.doc.createRange();
43668             }
43669         } else {
43670             return this.doc.createRange();
43671         }
43672     },
43673     getParentElement: function()
43674     {
43675         
43676         this.assignDocWin();
43677         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
43678         
43679         var range = this.createRange(sel);
43680          
43681         try {
43682             var p = range.commonAncestorContainer;
43683             while (p.nodeType == 3) { // text node
43684                 p = p.parentNode;
43685             }
43686             return p;
43687         } catch (e) {
43688             return null;
43689         }
43690     
43691     },
43692     /***
43693      *
43694      * Range intersection.. the hard stuff...
43695      *  '-1' = before
43696      *  '0' = hits..
43697      *  '1' = after.
43698      *         [ -- selected range --- ]
43699      *   [fail]                        [fail]
43700      *
43701      *    basically..
43702      *      if end is before start or  hits it. fail.
43703      *      if start is after end or hits it fail.
43704      *
43705      *   if either hits (but other is outside. - then it's not 
43706      *   
43707      *    
43708      **/
43709     
43710     
43711     // @see http://www.thismuchiknow.co.uk/?p=64.
43712     rangeIntersectsNode : function(range, node)
43713     {
43714         var nodeRange = node.ownerDocument.createRange();
43715         try {
43716             nodeRange.selectNode(node);
43717         } catch (e) {
43718             nodeRange.selectNodeContents(node);
43719         }
43720     
43721         var rangeStartRange = range.cloneRange();
43722         rangeStartRange.collapse(true);
43723     
43724         var rangeEndRange = range.cloneRange();
43725         rangeEndRange.collapse(false);
43726     
43727         var nodeStartRange = nodeRange.cloneRange();
43728         nodeStartRange.collapse(true);
43729     
43730         var nodeEndRange = nodeRange.cloneRange();
43731         nodeEndRange.collapse(false);
43732     
43733         return rangeStartRange.compareBoundaryPoints(
43734                  Range.START_TO_START, nodeEndRange) == -1 &&
43735                rangeEndRange.compareBoundaryPoints(
43736                  Range.START_TO_START, nodeStartRange) == 1;
43737         
43738          
43739     },
43740     rangeCompareNode : function(range, node)
43741     {
43742         var nodeRange = node.ownerDocument.createRange();
43743         try {
43744             nodeRange.selectNode(node);
43745         } catch (e) {
43746             nodeRange.selectNodeContents(node);
43747         }
43748         
43749         
43750         range.collapse(true);
43751     
43752         nodeRange.collapse(true);
43753      
43754         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
43755         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
43756          
43757         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
43758         
43759         var nodeIsBefore   =  ss == 1;
43760         var nodeIsAfter    = ee == -1;
43761         
43762         if (nodeIsBefore && nodeIsAfter) {
43763             return 0; // outer
43764         }
43765         if (!nodeIsBefore && nodeIsAfter) {
43766             return 1; //right trailed.
43767         }
43768         
43769         if (nodeIsBefore && !nodeIsAfter) {
43770             return 2;  // left trailed.
43771         }
43772         // fully contined.
43773         return 3;
43774     },
43775
43776     // private? - in a new class?
43777     cleanUpPaste :  function()
43778     {
43779         // cleans up the whole document..
43780         Roo.log('cleanuppaste');
43781         
43782         this.cleanUpChildren(this.doc.body);
43783         var clean = this.cleanWordChars(this.doc.body.innerHTML);
43784         if (clean != this.doc.body.innerHTML) {
43785             this.doc.body.innerHTML = clean;
43786         }
43787         
43788     },
43789     
43790     cleanWordChars : function(input) {// change the chars to hex code
43791         var he = Roo.HtmlEditorCore;
43792         
43793         var output = input;
43794         Roo.each(he.swapCodes, function(sw) { 
43795             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
43796             
43797             output = output.replace(swapper, sw[1]);
43798         });
43799         
43800         return output;
43801     },
43802     
43803     
43804     cleanUpChildren : function (n)
43805     {
43806         if (!n.childNodes.length) {
43807             return;
43808         }
43809         for (var i = n.childNodes.length-1; i > -1 ; i--) {
43810            this.cleanUpChild(n.childNodes[i]);
43811         }
43812     },
43813     
43814     
43815         
43816     
43817     cleanUpChild : function (node)
43818     {
43819         var ed = this;
43820         //console.log(node);
43821         if (node.nodeName == "#text") {
43822             // clean up silly Windows -- stuff?
43823             return; 
43824         }
43825         if (node.nodeName == "#comment") {
43826             node.parentNode.removeChild(node);
43827             // clean up silly Windows -- stuff?
43828             return; 
43829         }
43830         var lcname = node.tagName.toLowerCase();
43831         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
43832         // whitelist of tags..
43833         
43834         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
43835             // remove node.
43836             node.parentNode.removeChild(node);
43837             return;
43838             
43839         }
43840         
43841         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
43842         
43843         // remove <a name=....> as rendering on yahoo mailer is borked with this.
43844         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
43845         
43846         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
43847         //    remove_keep_children = true;
43848         //}
43849         
43850         if (remove_keep_children) {
43851             this.cleanUpChildren(node);
43852             // inserts everything just before this node...
43853             while (node.childNodes.length) {
43854                 var cn = node.childNodes[0];
43855                 node.removeChild(cn);
43856                 node.parentNode.insertBefore(cn, node);
43857             }
43858             node.parentNode.removeChild(node);
43859             return;
43860         }
43861         
43862         if (!node.attributes || !node.attributes.length) {
43863             this.cleanUpChildren(node);
43864             return;
43865         }
43866         
43867         function cleanAttr(n,v)
43868         {
43869             
43870             if (v.match(/^\./) || v.match(/^\//)) {
43871                 return;
43872             }
43873             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
43874                 return;
43875             }
43876             if (v.match(/^#/)) {
43877                 return;
43878             }
43879 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
43880             node.removeAttribute(n);
43881             
43882         }
43883         
43884         var cwhite = this.cwhite;
43885         var cblack = this.cblack;
43886             
43887         function cleanStyle(n,v)
43888         {
43889             if (v.match(/expression/)) { //XSS?? should we even bother..
43890                 node.removeAttribute(n);
43891                 return;
43892             }
43893             
43894             var parts = v.split(/;/);
43895             var clean = [];
43896             
43897             Roo.each(parts, function(p) {
43898                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
43899                 if (!p.length) {
43900                     return true;
43901                 }
43902                 var l = p.split(':').shift().replace(/\s+/g,'');
43903                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
43904                 
43905                 if ( cwhite.length && cblack.indexOf(l) > -1) {
43906 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
43907                     //node.removeAttribute(n);
43908                     return true;
43909                 }
43910                 //Roo.log()
43911                 // only allow 'c whitelisted system attributes'
43912                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
43913 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
43914                     //node.removeAttribute(n);
43915                     return true;
43916                 }
43917                 
43918                 
43919                  
43920                 
43921                 clean.push(p);
43922                 return true;
43923             });
43924             if (clean.length) { 
43925                 node.setAttribute(n, clean.join(';'));
43926             } else {
43927                 node.removeAttribute(n);
43928             }
43929             
43930         }
43931         
43932         
43933         for (var i = node.attributes.length-1; i > -1 ; i--) {
43934             var a = node.attributes[i];
43935             //console.log(a);
43936             
43937             if (a.name.toLowerCase().substr(0,2)=='on')  {
43938                 node.removeAttribute(a.name);
43939                 continue;
43940             }
43941             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
43942                 node.removeAttribute(a.name);
43943                 continue;
43944             }
43945             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
43946                 cleanAttr(a.name,a.value); // fixme..
43947                 continue;
43948             }
43949             if (a.name == 'style') {
43950                 cleanStyle(a.name,a.value);
43951                 continue;
43952             }
43953             /// clean up MS crap..
43954             // tecnically this should be a list of valid class'es..
43955             
43956             
43957             if (a.name == 'class') {
43958                 if (a.value.match(/^Mso/)) {
43959                     node.className = '';
43960                 }
43961                 
43962                 if (a.value.match(/^body$/)) {
43963                     node.className = '';
43964                 }
43965                 continue;
43966             }
43967             
43968             // style cleanup!?
43969             // class cleanup?
43970             
43971         }
43972         
43973         
43974         this.cleanUpChildren(node);
43975         
43976         
43977     },
43978     
43979     /**
43980      * Clean up MS wordisms...
43981      */
43982     cleanWord : function(node)
43983     {
43984         
43985         
43986         if (!node) {
43987             this.cleanWord(this.doc.body);
43988             return;
43989         }
43990         if (node.nodeName == "#text") {
43991             // clean up silly Windows -- stuff?
43992             return; 
43993         }
43994         if (node.nodeName == "#comment") {
43995             node.parentNode.removeChild(node);
43996             // clean up silly Windows -- stuff?
43997             return; 
43998         }
43999         
44000         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44001             node.parentNode.removeChild(node);
44002             return;
44003         }
44004         
44005         // remove - but keep children..
44006         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|font)/)) {
44007             while (node.childNodes.length) {
44008                 var cn = node.childNodes[0];
44009                 node.removeChild(cn);
44010                 node.parentNode.insertBefore(cn, node);
44011             }
44012             node.parentNode.removeChild(node);
44013             this.iterateChildren(node, this.cleanWord);
44014             return;
44015         }
44016         // clean styles
44017         if (node.className.length) {
44018             
44019             var cn = node.className.split(/\W+/);
44020             var cna = [];
44021             Roo.each(cn, function(cls) {
44022                 if (cls.match(/Mso[a-zA-Z]+/)) {
44023                     return;
44024                 }
44025                 cna.push(cls);
44026             });
44027             node.className = cna.length ? cna.join(' ') : '';
44028             if (!cna.length) {
44029                 node.removeAttribute("class");
44030             }
44031         }
44032         
44033         if (node.hasAttribute("lang")) {
44034             node.removeAttribute("lang");
44035         }
44036         
44037         if (node.hasAttribute("style")) {
44038             
44039             var styles = node.getAttribute("style").split(";");
44040             var nstyle = [];
44041             Roo.each(styles, function(s) {
44042                 if (!s.match(/:/)) {
44043                     return;
44044                 }
44045                 var kv = s.split(":");
44046                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44047                     return;
44048                 }
44049                 // what ever is left... we allow.
44050                 nstyle.push(s);
44051             });
44052             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44053             if (!nstyle.length) {
44054                 node.removeAttribute('style');
44055             }
44056         }
44057         this.iterateChildren(node, this.cleanWord);
44058         
44059         
44060         
44061     },
44062     /**
44063      * iterateChildren of a Node, calling fn each time, using this as the scole..
44064      * @param {DomNode} node node to iterate children of.
44065      * @param {Function} fn method of this class to call on each item.
44066      */
44067     iterateChildren : function(node, fn)
44068     {
44069         if (!node.childNodes.length) {
44070                 return;
44071         }
44072         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44073            fn.call(this, node.childNodes[i])
44074         }
44075     },
44076     
44077     
44078     /**
44079      * cleanTableWidths.
44080      *
44081      * Quite often pasting from word etc.. results in tables with column and widths.
44082      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44083      *
44084      */
44085     cleanTableWidths : function(node)
44086     {
44087          
44088          
44089         if (!node) {
44090             this.cleanTableWidths(this.doc.body);
44091             return;
44092         }
44093         
44094         // ignore list...
44095         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44096             return; 
44097         }
44098         Roo.log(node.tagName);
44099         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44100             this.iterateChildren(node, this.cleanTableWidths);
44101             return;
44102         }
44103         if (node.hasAttribute('width')) {
44104             node.removeAttribute('width');
44105         }
44106         
44107          
44108         if (node.hasAttribute("style")) {
44109             // pretty basic...
44110             
44111             var styles = node.getAttribute("style").split(";");
44112             var nstyle = [];
44113             Roo.each(styles, function(s) {
44114                 if (!s.match(/:/)) {
44115                     return;
44116                 }
44117                 var kv = s.split(":");
44118                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44119                     return;
44120                 }
44121                 // what ever is left... we allow.
44122                 nstyle.push(s);
44123             });
44124             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44125             if (!nstyle.length) {
44126                 node.removeAttribute('style');
44127             }
44128         }
44129         
44130         this.iterateChildren(node, this.cleanTableWidths);
44131         
44132         
44133     },
44134     
44135     
44136     
44137     
44138     domToHTML : function(currentElement, depth, nopadtext) {
44139         
44140         depth = depth || 0;
44141         nopadtext = nopadtext || false;
44142     
44143         if (!currentElement) {
44144             return this.domToHTML(this.doc.body);
44145         }
44146         
44147         //Roo.log(currentElement);
44148         var j;
44149         var allText = false;
44150         var nodeName = currentElement.nodeName;
44151         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44152         
44153         if  (nodeName == '#text') {
44154             
44155             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44156         }
44157         
44158         
44159         var ret = '';
44160         if (nodeName != 'BODY') {
44161              
44162             var i = 0;
44163             // Prints the node tagName, such as <A>, <IMG>, etc
44164             if (tagName) {
44165                 var attr = [];
44166                 for(i = 0; i < currentElement.attributes.length;i++) {
44167                     // quoting?
44168                     var aname = currentElement.attributes.item(i).name;
44169                     if (!currentElement.attributes.item(i).value.length) {
44170                         continue;
44171                     }
44172                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44173                 }
44174                 
44175                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44176             } 
44177             else {
44178                 
44179                 // eack
44180             }
44181         } else {
44182             tagName = false;
44183         }
44184         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44185             return ret;
44186         }
44187         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44188             nopadtext = true;
44189         }
44190         
44191         
44192         // Traverse the tree
44193         i = 0;
44194         var currentElementChild = currentElement.childNodes.item(i);
44195         var allText = true;
44196         var innerHTML  = '';
44197         lastnode = '';
44198         while (currentElementChild) {
44199             // Formatting code (indent the tree so it looks nice on the screen)
44200             var nopad = nopadtext;
44201             if (lastnode == 'SPAN') {
44202                 nopad  = true;
44203             }
44204             // text
44205             if  (currentElementChild.nodeName == '#text') {
44206                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44207                 toadd = nopadtext ? toadd : toadd.trim();
44208                 if (!nopad && toadd.length > 80) {
44209                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44210                 }
44211                 innerHTML  += toadd;
44212                 
44213                 i++;
44214                 currentElementChild = currentElement.childNodes.item(i);
44215                 lastNode = '';
44216                 continue;
44217             }
44218             allText = false;
44219             
44220             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44221                 
44222             // Recursively traverse the tree structure of the child node
44223             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44224             lastnode = currentElementChild.nodeName;
44225             i++;
44226             currentElementChild=currentElement.childNodes.item(i);
44227         }
44228         
44229         ret += innerHTML;
44230         
44231         if (!allText) {
44232                 // The remaining code is mostly for formatting the tree
44233             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44234         }
44235         
44236         
44237         if (tagName) {
44238             ret+= "</"+tagName+">";
44239         }
44240         return ret;
44241         
44242     },
44243         
44244     applyBlacklists : function()
44245     {
44246         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44247         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44248         
44249         this.white = [];
44250         this.black = [];
44251         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44252             if (b.indexOf(tag) > -1) {
44253                 return;
44254             }
44255             this.white.push(tag);
44256             
44257         }, this);
44258         
44259         Roo.each(w, function(tag) {
44260             if (b.indexOf(tag) > -1) {
44261                 return;
44262             }
44263             if (this.white.indexOf(tag) > -1) {
44264                 return;
44265             }
44266             this.white.push(tag);
44267             
44268         }, this);
44269         
44270         
44271         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44272             if (w.indexOf(tag) > -1) {
44273                 return;
44274             }
44275             this.black.push(tag);
44276             
44277         }, this);
44278         
44279         Roo.each(b, function(tag) {
44280             if (w.indexOf(tag) > -1) {
44281                 return;
44282             }
44283             if (this.black.indexOf(tag) > -1) {
44284                 return;
44285             }
44286             this.black.push(tag);
44287             
44288         }, this);
44289         
44290         
44291         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44292         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44293         
44294         this.cwhite = [];
44295         this.cblack = [];
44296         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44297             if (b.indexOf(tag) > -1) {
44298                 return;
44299             }
44300             this.cwhite.push(tag);
44301             
44302         }, this);
44303         
44304         Roo.each(w, function(tag) {
44305             if (b.indexOf(tag) > -1) {
44306                 return;
44307             }
44308             if (this.cwhite.indexOf(tag) > -1) {
44309                 return;
44310             }
44311             this.cwhite.push(tag);
44312             
44313         }, this);
44314         
44315         
44316         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44317             if (w.indexOf(tag) > -1) {
44318                 return;
44319             }
44320             this.cblack.push(tag);
44321             
44322         }, this);
44323         
44324         Roo.each(b, function(tag) {
44325             if (w.indexOf(tag) > -1) {
44326                 return;
44327             }
44328             if (this.cblack.indexOf(tag) > -1) {
44329                 return;
44330             }
44331             this.cblack.push(tag);
44332             
44333         }, this);
44334     },
44335     
44336     setStylesheets : function(stylesheets)
44337     {
44338         if(typeof(stylesheets) == 'string'){
44339             Roo.get(this.iframe.contentDocument.head).createChild({
44340                 tag : 'link',
44341                 rel : 'stylesheet',
44342                 type : 'text/css',
44343                 href : stylesheets
44344             });
44345             
44346             return;
44347         }
44348         var _this = this;
44349      
44350         Roo.each(stylesheets, function(s) {
44351             if(!s.length){
44352                 return;
44353             }
44354             
44355             Roo.get(_this.iframe.contentDocument.head).createChild({
44356                 tag : 'link',
44357                 rel : 'stylesheet',
44358                 type : 'text/css',
44359                 href : s
44360             });
44361         });
44362
44363         
44364     },
44365     
44366     removeStylesheets : function()
44367     {
44368         var _this = this;
44369         
44370         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44371             s.remove();
44372         });
44373     },
44374     
44375     setStyle : function(style)
44376     {
44377         Roo.get(this.iframe.contentDocument.head).createChild({
44378             tag : 'style',
44379             type : 'text/css',
44380             html : style
44381         });
44382
44383         return;
44384     }
44385     
44386     // hide stuff that is not compatible
44387     /**
44388      * @event blur
44389      * @hide
44390      */
44391     /**
44392      * @event change
44393      * @hide
44394      */
44395     /**
44396      * @event focus
44397      * @hide
44398      */
44399     /**
44400      * @event specialkey
44401      * @hide
44402      */
44403     /**
44404      * @cfg {String} fieldClass @hide
44405      */
44406     /**
44407      * @cfg {String} focusClass @hide
44408      */
44409     /**
44410      * @cfg {String} autoCreate @hide
44411      */
44412     /**
44413      * @cfg {String} inputType @hide
44414      */
44415     /**
44416      * @cfg {String} invalidClass @hide
44417      */
44418     /**
44419      * @cfg {String} invalidText @hide
44420      */
44421     /**
44422      * @cfg {String} msgFx @hide
44423      */
44424     /**
44425      * @cfg {String} validateOnBlur @hide
44426      */
44427 });
44428
44429 Roo.HtmlEditorCore.white = [
44430         'area', 'br', 'img', 'input', 'hr', 'wbr',
44431         
44432        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44433        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44434        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44435        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44436        'table',   'ul',         'xmp', 
44437        
44438        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44439       'thead',   'tr', 
44440      
44441       'dir', 'menu', 'ol', 'ul', 'dl',
44442        
44443       'embed',  'object'
44444 ];
44445
44446
44447 Roo.HtmlEditorCore.black = [
44448     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44449         'applet', // 
44450         'base',   'basefont', 'bgsound', 'blink',  'body', 
44451         'frame',  'frameset', 'head',    'html',   'ilayer', 
44452         'iframe', 'layer',  'link',     'meta',    'object',   
44453         'script', 'style' ,'title',  'xml' // clean later..
44454 ];
44455 Roo.HtmlEditorCore.clean = [
44456     'script', 'style', 'title', 'xml'
44457 ];
44458 Roo.HtmlEditorCore.remove = [
44459     'font'
44460 ];
44461 // attributes..
44462
44463 Roo.HtmlEditorCore.ablack = [
44464     'on'
44465 ];
44466     
44467 Roo.HtmlEditorCore.aclean = [ 
44468     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
44469 ];
44470
44471 // protocols..
44472 Roo.HtmlEditorCore.pwhite= [
44473         'http',  'https',  'mailto'
44474 ];
44475
44476 // white listed style attributes.
44477 Roo.HtmlEditorCore.cwhite= [
44478       //  'text-align', /// default is to allow most things..
44479       
44480          
44481 //        'font-size'//??
44482 ];
44483
44484 // black listed style attributes.
44485 Roo.HtmlEditorCore.cblack= [
44486       //  'font-size' -- this can be set by the project 
44487 ];
44488
44489
44490 Roo.HtmlEditorCore.swapCodes   =[ 
44491     [    8211, "--" ], 
44492     [    8212, "--" ], 
44493     [    8216,  "'" ],  
44494     [    8217, "'" ],  
44495     [    8220, '"' ],  
44496     [    8221, '"' ],  
44497     [    8226, "*" ],  
44498     [    8230, "..." ]
44499 ]; 
44500
44501     //<script type="text/javascript">
44502
44503 /*
44504  * Ext JS Library 1.1.1
44505  * Copyright(c) 2006-2007, Ext JS, LLC.
44506  * Licence LGPL
44507  * 
44508  */
44509  
44510  
44511 Roo.form.HtmlEditor = function(config){
44512     
44513     
44514     
44515     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
44516     
44517     if (!this.toolbars) {
44518         this.toolbars = [];
44519     }
44520     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
44521     
44522     
44523 };
44524
44525 /**
44526  * @class Roo.form.HtmlEditor
44527  * @extends Roo.form.Field
44528  * Provides a lightweight HTML Editor component.
44529  *
44530  * This has been tested on Fireforx / Chrome.. IE may not be so great..
44531  * 
44532  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
44533  * supported by this editor.</b><br/><br/>
44534  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
44535  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
44536  */
44537 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
44538     /**
44539      * @cfg {Boolean} clearUp
44540      */
44541     clearUp : true,
44542       /**
44543      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
44544      */
44545     toolbars : false,
44546    
44547      /**
44548      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
44549      *                        Roo.resizable.
44550      */
44551     resizable : false,
44552      /**
44553      * @cfg {Number} height (in pixels)
44554      */   
44555     height: 300,
44556    /**
44557      * @cfg {Number} width (in pixels)
44558      */   
44559     width: 500,
44560     
44561     /**
44562      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
44563      * 
44564      */
44565     stylesheets: false,
44566     
44567     
44568      /**
44569      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
44570      * 
44571      */
44572     cblack: false,
44573     /**
44574      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
44575      * 
44576      */
44577     cwhite: false,
44578     
44579      /**
44580      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
44581      * 
44582      */
44583     black: false,
44584     /**
44585      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
44586      * 
44587      */
44588     white: false,
44589     
44590     // id of frame..
44591     frameId: false,
44592     
44593     // private properties
44594     validationEvent : false,
44595     deferHeight: true,
44596     initialized : false,
44597     activated : false,
44598     
44599     onFocus : Roo.emptyFn,
44600     iframePad:3,
44601     hideMode:'offsets',
44602     
44603     actionMode : 'container', // defaults to hiding it...
44604     
44605     defaultAutoCreate : { // modified by initCompnoent..
44606         tag: "textarea",
44607         style:"width:500px;height:300px;",
44608         autocomplete: "new-password"
44609     },
44610
44611     // private
44612     initComponent : function(){
44613         this.addEvents({
44614             /**
44615              * @event initialize
44616              * Fires when the editor is fully initialized (including the iframe)
44617              * @param {HtmlEditor} this
44618              */
44619             initialize: true,
44620             /**
44621              * @event activate
44622              * Fires when the editor is first receives the focus. Any insertion must wait
44623              * until after this event.
44624              * @param {HtmlEditor} this
44625              */
44626             activate: true,
44627              /**
44628              * @event beforesync
44629              * Fires before the textarea is updated with content from the editor iframe. Return false
44630              * to cancel the sync.
44631              * @param {HtmlEditor} this
44632              * @param {String} html
44633              */
44634             beforesync: true,
44635              /**
44636              * @event beforepush
44637              * Fires before the iframe editor is updated with content from the textarea. Return false
44638              * to cancel the push.
44639              * @param {HtmlEditor} this
44640              * @param {String} html
44641              */
44642             beforepush: true,
44643              /**
44644              * @event sync
44645              * Fires when the textarea is updated with content from the editor iframe.
44646              * @param {HtmlEditor} this
44647              * @param {String} html
44648              */
44649             sync: true,
44650              /**
44651              * @event push
44652              * Fires when the iframe editor is updated with content from the textarea.
44653              * @param {HtmlEditor} this
44654              * @param {String} html
44655              */
44656             push: true,
44657              /**
44658              * @event editmodechange
44659              * Fires when the editor switches edit modes
44660              * @param {HtmlEditor} this
44661              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
44662              */
44663             editmodechange: true,
44664             /**
44665              * @event editorevent
44666              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
44667              * @param {HtmlEditor} this
44668              */
44669             editorevent: true,
44670             /**
44671              * @event firstfocus
44672              * Fires when on first focus - needed by toolbars..
44673              * @param {HtmlEditor} this
44674              */
44675             firstfocus: true,
44676             /**
44677              * @event autosave
44678              * Auto save the htmlEditor value as a file into Events
44679              * @param {HtmlEditor} this
44680              */
44681             autosave: true,
44682             /**
44683              * @event savedpreview
44684              * preview the saved version of htmlEditor
44685              * @param {HtmlEditor} this
44686              */
44687             savedpreview: true,
44688             
44689             /**
44690             * @event stylesheetsclick
44691             * Fires when press the Sytlesheets button
44692             * @param {Roo.HtmlEditorCore} this
44693             */
44694             stylesheetsclick: true
44695         });
44696         this.defaultAutoCreate =  {
44697             tag: "textarea",
44698             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
44699             autocomplete: "new-password"
44700         };
44701     },
44702
44703     /**
44704      * Protected method that will not generally be called directly. It
44705      * is called when the editor creates its toolbar. Override this method if you need to
44706      * add custom toolbar buttons.
44707      * @param {HtmlEditor} editor
44708      */
44709     createToolbar : function(editor){
44710         Roo.log("create toolbars");
44711         if (!editor.toolbars || !editor.toolbars.length) {
44712             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
44713         }
44714         
44715         for (var i =0 ; i < editor.toolbars.length;i++) {
44716             editor.toolbars[i] = Roo.factory(
44717                     typeof(editor.toolbars[i]) == 'string' ?
44718                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
44719                 Roo.form.HtmlEditor);
44720             editor.toolbars[i].init(editor);
44721         }
44722          
44723         
44724     },
44725
44726      
44727     // private
44728     onRender : function(ct, position)
44729     {
44730         var _t = this;
44731         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
44732         
44733         this.wrap = this.el.wrap({
44734             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
44735         });
44736         
44737         this.editorcore.onRender(ct, position);
44738          
44739         if (this.resizable) {
44740             this.resizeEl = new Roo.Resizable(this.wrap, {
44741                 pinned : true,
44742                 wrap: true,
44743                 dynamic : true,
44744                 minHeight : this.height,
44745                 height: this.height,
44746                 handles : this.resizable,
44747                 width: this.width,
44748                 listeners : {
44749                     resize : function(r, w, h) {
44750                         _t.onResize(w,h); // -something
44751                     }
44752                 }
44753             });
44754             
44755         }
44756         this.createToolbar(this);
44757        
44758         
44759         if(!this.width){
44760             this.setSize(this.wrap.getSize());
44761         }
44762         if (this.resizeEl) {
44763             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
44764             // should trigger onReize..
44765         }
44766         
44767         this.keyNav = new Roo.KeyNav(this.el, {
44768             
44769             "tab" : function(e){
44770                 e.preventDefault();
44771                 
44772                 var value = this.getValue();
44773                 
44774                 var start = this.el.dom.selectionStart;
44775                 var end = this.el.dom.selectionEnd;
44776                 
44777                 if(!e.shiftKey){
44778                     
44779                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
44780                     this.el.dom.setSelectionRange(end + 1, end + 1);
44781                     return;
44782                 }
44783                 
44784                 var f = value.substring(0, start).split("\t");
44785                 
44786                 if(f.pop().length != 0){
44787                     return;
44788                 }
44789                 
44790                 this.setValue(f.join("\t") + value.substring(end));
44791                 this.el.dom.setSelectionRange(start - 1, start - 1);
44792                 
44793             },
44794             
44795             "home" : function(e){
44796                 e.preventDefault();
44797                 
44798                 var curr = this.el.dom.selectionStart;
44799                 var lines = this.getValue().split("\n");
44800                 
44801                 if(!lines.length){
44802                     return;
44803                 }
44804                 
44805                 if(e.ctrlKey){
44806                     this.el.dom.setSelectionRange(0, 0);
44807                     return;
44808                 }
44809                 
44810                 var pos = 0;
44811                 
44812                 for (var i = 0; i < lines.length;i++) {
44813                     pos += lines[i].length;
44814                     
44815                     if(i != 0){
44816                         pos += 1;
44817                     }
44818                     
44819                     if(pos < curr){
44820                         continue;
44821                     }
44822                     
44823                     pos -= lines[i].length;
44824                     
44825                     break;
44826                 }
44827                 
44828                 if(!e.shiftKey){
44829                     this.el.dom.setSelectionRange(pos, pos);
44830                     return;
44831                 }
44832                 
44833                 this.el.dom.selectionStart = pos;
44834                 this.el.dom.selectionEnd = curr;
44835             },
44836             
44837             "end" : function(e){
44838                 e.preventDefault();
44839                 
44840                 var curr = this.el.dom.selectionStart;
44841                 var lines = this.getValue().split("\n");
44842                 
44843                 if(!lines.length){
44844                     return;
44845                 }
44846                 
44847                 if(e.ctrlKey){
44848                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
44849                     return;
44850                 }
44851                 
44852                 var pos = 0;
44853                 
44854                 for (var i = 0; i < lines.length;i++) {
44855                     
44856                     pos += lines[i].length;
44857                     
44858                     if(i != 0){
44859                         pos += 1;
44860                     }
44861                     
44862                     if(pos < curr){
44863                         continue;
44864                     }
44865                     
44866                     break;
44867                 }
44868                 
44869                 if(!e.shiftKey){
44870                     this.el.dom.setSelectionRange(pos, pos);
44871                     return;
44872                 }
44873                 
44874                 this.el.dom.selectionStart = curr;
44875                 this.el.dom.selectionEnd = pos;
44876             },
44877
44878             scope : this,
44879
44880             doRelay : function(foo, bar, hname){
44881                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
44882             },
44883
44884             forceKeyDown: true
44885         });
44886         
44887 //        if(this.autosave && this.w){
44888 //            this.autoSaveFn = setInterval(this.autosave, 1000);
44889 //        }
44890     },
44891
44892     // private
44893     onResize : function(w, h)
44894     {
44895         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
44896         var ew = false;
44897         var eh = false;
44898         
44899         if(this.el ){
44900             if(typeof w == 'number'){
44901                 var aw = w - this.wrap.getFrameWidth('lr');
44902                 this.el.setWidth(this.adjustWidth('textarea', aw));
44903                 ew = aw;
44904             }
44905             if(typeof h == 'number'){
44906                 var tbh = 0;
44907                 for (var i =0; i < this.toolbars.length;i++) {
44908                     // fixme - ask toolbars for heights?
44909                     tbh += this.toolbars[i].tb.el.getHeight();
44910                     if (this.toolbars[i].footer) {
44911                         tbh += this.toolbars[i].footer.el.getHeight();
44912                     }
44913                 }
44914                 
44915                 
44916                 
44917                 
44918                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
44919                 ah -= 5; // knock a few pixes off for look..
44920 //                Roo.log(ah);
44921                 this.el.setHeight(this.adjustWidth('textarea', ah));
44922                 var eh = ah;
44923             }
44924         }
44925         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
44926         this.editorcore.onResize(ew,eh);
44927         
44928     },
44929
44930     /**
44931      * Toggles the editor between standard and source edit mode.
44932      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
44933      */
44934     toggleSourceEdit : function(sourceEditMode)
44935     {
44936         this.editorcore.toggleSourceEdit(sourceEditMode);
44937         
44938         if(this.editorcore.sourceEditMode){
44939             Roo.log('editor - showing textarea');
44940             
44941 //            Roo.log('in');
44942 //            Roo.log(this.syncValue());
44943             this.editorcore.syncValue();
44944             this.el.removeClass('x-hidden');
44945             this.el.dom.removeAttribute('tabIndex');
44946             this.el.focus();
44947             
44948             for (var i = 0; i < this.toolbars.length; i++) {
44949                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
44950                     this.toolbars[i].tb.hide();
44951                     this.toolbars[i].footer.hide();
44952                 }
44953             }
44954             
44955         }else{
44956             Roo.log('editor - hiding textarea');
44957 //            Roo.log('out')
44958 //            Roo.log(this.pushValue()); 
44959             this.editorcore.pushValue();
44960             
44961             this.el.addClass('x-hidden');
44962             this.el.dom.setAttribute('tabIndex', -1);
44963             
44964             for (var i = 0; i < this.toolbars.length; i++) {
44965                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
44966                     this.toolbars[i].tb.show();
44967                     this.toolbars[i].footer.show();
44968                 }
44969             }
44970             
44971             //this.deferFocus();
44972         }
44973         
44974         this.setSize(this.wrap.getSize());
44975         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
44976         
44977         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
44978     },
44979  
44980     // private (for BoxComponent)
44981     adjustSize : Roo.BoxComponent.prototype.adjustSize,
44982
44983     // private (for BoxComponent)
44984     getResizeEl : function(){
44985         return this.wrap;
44986     },
44987
44988     // private (for BoxComponent)
44989     getPositionEl : function(){
44990         return this.wrap;
44991     },
44992
44993     // private
44994     initEvents : function(){
44995         this.originalValue = this.getValue();
44996     },
44997
44998     /**
44999      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45000      * @method
45001      */
45002     markInvalid : Roo.emptyFn,
45003     /**
45004      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45005      * @method
45006      */
45007     clearInvalid : Roo.emptyFn,
45008
45009     setValue : function(v){
45010         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45011         this.editorcore.pushValue();
45012     },
45013
45014      
45015     // private
45016     deferFocus : function(){
45017         this.focus.defer(10, this);
45018     },
45019
45020     // doc'ed in Field
45021     focus : function(){
45022         this.editorcore.focus();
45023         
45024     },
45025       
45026
45027     // private
45028     onDestroy : function(){
45029         
45030         
45031         
45032         if(this.rendered){
45033             
45034             for (var i =0; i < this.toolbars.length;i++) {
45035                 // fixme - ask toolbars for heights?
45036                 this.toolbars[i].onDestroy();
45037             }
45038             
45039             this.wrap.dom.innerHTML = '';
45040             this.wrap.remove();
45041         }
45042     },
45043
45044     // private
45045     onFirstFocus : function(){
45046         //Roo.log("onFirstFocus");
45047         this.editorcore.onFirstFocus();
45048          for (var i =0; i < this.toolbars.length;i++) {
45049             this.toolbars[i].onFirstFocus();
45050         }
45051         
45052     },
45053     
45054     // private
45055     syncValue : function()
45056     {
45057         this.editorcore.syncValue();
45058     },
45059     
45060     pushValue : function()
45061     {
45062         this.editorcore.pushValue();
45063     },
45064     
45065     setStylesheets : function(stylesheets)
45066     {
45067         this.editorcore.setStylesheets(stylesheets);
45068     },
45069     
45070     removeStylesheets : function()
45071     {
45072         this.editorcore.removeStylesheets();
45073     }
45074      
45075     
45076     // hide stuff that is not compatible
45077     /**
45078      * @event blur
45079      * @hide
45080      */
45081     /**
45082      * @event change
45083      * @hide
45084      */
45085     /**
45086      * @event focus
45087      * @hide
45088      */
45089     /**
45090      * @event specialkey
45091      * @hide
45092      */
45093     /**
45094      * @cfg {String} fieldClass @hide
45095      */
45096     /**
45097      * @cfg {String} focusClass @hide
45098      */
45099     /**
45100      * @cfg {String} autoCreate @hide
45101      */
45102     /**
45103      * @cfg {String} inputType @hide
45104      */
45105     /**
45106      * @cfg {String} invalidClass @hide
45107      */
45108     /**
45109      * @cfg {String} invalidText @hide
45110      */
45111     /**
45112      * @cfg {String} msgFx @hide
45113      */
45114     /**
45115      * @cfg {String} validateOnBlur @hide
45116      */
45117 });
45118  
45119     // <script type="text/javascript">
45120 /*
45121  * Based on
45122  * Ext JS Library 1.1.1
45123  * Copyright(c) 2006-2007, Ext JS, LLC.
45124  *  
45125  
45126  */
45127
45128 /**
45129  * @class Roo.form.HtmlEditorToolbar1
45130  * Basic Toolbar
45131  * 
45132  * Usage:
45133  *
45134  new Roo.form.HtmlEditor({
45135     ....
45136     toolbars : [
45137         new Roo.form.HtmlEditorToolbar1({
45138             disable : { fonts: 1 , format: 1, ..., ... , ...],
45139             btns : [ .... ]
45140         })
45141     }
45142      
45143  * 
45144  * @cfg {Object} disable List of elements to disable..
45145  * @cfg {Array} btns List of additional buttons.
45146  * 
45147  * 
45148  * NEEDS Extra CSS? 
45149  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45150  */
45151  
45152 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45153 {
45154     
45155     Roo.apply(this, config);
45156     
45157     // default disabled, based on 'good practice'..
45158     this.disable = this.disable || {};
45159     Roo.applyIf(this.disable, {
45160         fontSize : true,
45161         colors : true,
45162         specialElements : true
45163     });
45164     
45165     
45166     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45167     // dont call parent... till later.
45168 }
45169
45170 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45171     
45172     tb: false,
45173     
45174     rendered: false,
45175     
45176     editor : false,
45177     editorcore : false,
45178     /**
45179      * @cfg {Object} disable  List of toolbar elements to disable
45180          
45181      */
45182     disable : false,
45183     
45184     
45185      /**
45186      * @cfg {String} createLinkText The default text for the create link prompt
45187      */
45188     createLinkText : 'Please enter the URL for the link:',
45189     /**
45190      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45191      */
45192     defaultLinkValue : 'http:/'+'/',
45193    
45194     
45195       /**
45196      * @cfg {Array} fontFamilies An array of available font families
45197      */
45198     fontFamilies : [
45199         'Arial',
45200         'Courier New',
45201         'Tahoma',
45202         'Times New Roman',
45203         'Verdana'
45204     ],
45205     
45206     specialChars : [
45207            "&#169;",
45208           "&#174;",     
45209           "&#8482;",    
45210           "&#163;" ,    
45211          // "&#8212;",    
45212           "&#8230;",    
45213           "&#247;" ,    
45214         //  "&#225;" ,     ?? a acute?
45215            "&#8364;"    , //Euro
45216        //   "&#8220;"    ,
45217         //  "&#8221;"    ,
45218         //  "&#8226;"    ,
45219           "&#176;"  //   , // degrees
45220
45221          // "&#233;"     , // e ecute
45222          // "&#250;"     , // u ecute?
45223     ],
45224     
45225     specialElements : [
45226         {
45227             text: "Insert Table",
45228             xtype: 'MenuItem',
45229             xns : Roo.Menu,
45230             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45231                 
45232         },
45233         {    
45234             text: "Insert Image",
45235             xtype: 'MenuItem',
45236             xns : Roo.Menu,
45237             ihtml : '<img src="about:blank"/>'
45238             
45239         }
45240         
45241          
45242     ],
45243     
45244     
45245     inputElements : [ 
45246             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45247             "input:submit", "input:button", "select", "textarea", "label" ],
45248     formats : [
45249         ["p"] ,  
45250         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45251         ["pre"],[ "code"], 
45252         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45253         ['div'],['span']
45254     ],
45255     
45256     cleanStyles : [
45257         "font-size"
45258     ],
45259      /**
45260      * @cfg {String} defaultFont default font to use.
45261      */
45262     defaultFont: 'tahoma',
45263    
45264     fontSelect : false,
45265     
45266     
45267     formatCombo : false,
45268     
45269     init : function(editor)
45270     {
45271         this.editor = editor;
45272         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45273         var editorcore = this.editorcore;
45274         
45275         var _t = this;
45276         
45277         var fid = editorcore.frameId;
45278         var etb = this;
45279         function btn(id, toggle, handler){
45280             var xid = fid + '-'+ id ;
45281             return {
45282                 id : xid,
45283                 cmd : id,
45284                 cls : 'x-btn-icon x-edit-'+id,
45285                 enableToggle:toggle !== false,
45286                 scope: _t, // was editor...
45287                 handler:handler||_t.relayBtnCmd,
45288                 clickEvent:'mousedown',
45289                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45290                 tabIndex:-1
45291             };
45292         }
45293         
45294         
45295         
45296         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45297         this.tb = tb;
45298          // stop form submits
45299         tb.el.on('click', function(e){
45300             e.preventDefault(); // what does this do?
45301         });
45302
45303         if(!this.disable.font) { // && !Roo.isSafari){
45304             /* why no safari for fonts 
45305             editor.fontSelect = tb.el.createChild({
45306                 tag:'select',
45307                 tabIndex: -1,
45308                 cls:'x-font-select',
45309                 html: this.createFontOptions()
45310             });
45311             
45312             editor.fontSelect.on('change', function(){
45313                 var font = editor.fontSelect.dom.value;
45314                 editor.relayCmd('fontname', font);
45315                 editor.deferFocus();
45316             }, editor);
45317             
45318             tb.add(
45319                 editor.fontSelect.dom,
45320                 '-'
45321             );
45322             */
45323             
45324         };
45325         if(!this.disable.formats){
45326             this.formatCombo = new Roo.form.ComboBox({
45327                 store: new Roo.data.SimpleStore({
45328                     id : 'tag',
45329                     fields: ['tag'],
45330                     data : this.formats // from states.js
45331                 }),
45332                 blockFocus : true,
45333                 name : '',
45334                 //autoCreate : {tag: "div",  size: "20"},
45335                 displayField:'tag',
45336                 typeAhead: false,
45337                 mode: 'local',
45338                 editable : false,
45339                 triggerAction: 'all',
45340                 emptyText:'Add tag',
45341                 selectOnFocus:true,
45342                 width:135,
45343                 listeners : {
45344                     'select': function(c, r, i) {
45345                         editorcore.insertTag(r.get('tag'));
45346                         editor.focus();
45347                     }
45348                 }
45349
45350             });
45351             tb.addField(this.formatCombo);
45352             
45353         }
45354         
45355         if(!this.disable.format){
45356             tb.add(
45357                 btn('bold'),
45358                 btn('italic'),
45359                 btn('underline'),
45360                 btn('strikethrough')
45361             );
45362         };
45363         if(!this.disable.fontSize){
45364             tb.add(
45365                 '-',
45366                 
45367                 
45368                 btn('increasefontsize', false, editorcore.adjustFont),
45369                 btn('decreasefontsize', false, editorcore.adjustFont)
45370             );
45371         };
45372         
45373         
45374         if(!this.disable.colors){
45375             tb.add(
45376                 '-', {
45377                     id:editorcore.frameId +'-forecolor',
45378                     cls:'x-btn-icon x-edit-forecolor',
45379                     clickEvent:'mousedown',
45380                     tooltip: this.buttonTips['forecolor'] || undefined,
45381                     tabIndex:-1,
45382                     menu : new Roo.menu.ColorMenu({
45383                         allowReselect: true,
45384                         focus: Roo.emptyFn,
45385                         value:'000000',
45386                         plain:true,
45387                         selectHandler: function(cp, color){
45388                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45389                             editor.deferFocus();
45390                         },
45391                         scope: editorcore,
45392                         clickEvent:'mousedown'
45393                     })
45394                 }, {
45395                     id:editorcore.frameId +'backcolor',
45396                     cls:'x-btn-icon x-edit-backcolor',
45397                     clickEvent:'mousedown',
45398                     tooltip: this.buttonTips['backcolor'] || undefined,
45399                     tabIndex:-1,
45400                     menu : new Roo.menu.ColorMenu({
45401                         focus: Roo.emptyFn,
45402                         value:'FFFFFF',
45403                         plain:true,
45404                         allowReselect: true,
45405                         selectHandler: function(cp, color){
45406                             if(Roo.isGecko){
45407                                 editorcore.execCmd('useCSS', false);
45408                                 editorcore.execCmd('hilitecolor', color);
45409                                 editorcore.execCmd('useCSS', true);
45410                                 editor.deferFocus();
45411                             }else{
45412                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45413                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45414                                 editor.deferFocus();
45415                             }
45416                         },
45417                         scope:editorcore,
45418                         clickEvent:'mousedown'
45419                     })
45420                 }
45421             );
45422         };
45423         // now add all the items...
45424         
45425
45426         if(!this.disable.alignments){
45427             tb.add(
45428                 '-',
45429                 btn('justifyleft'),
45430                 btn('justifycenter'),
45431                 btn('justifyright')
45432             );
45433         };
45434
45435         //if(!Roo.isSafari){
45436             if(!this.disable.links){
45437                 tb.add(
45438                     '-',
45439                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45440                 );
45441             };
45442
45443             if(!this.disable.lists){
45444                 tb.add(
45445                     '-',
45446                     btn('insertorderedlist'),
45447                     btn('insertunorderedlist')
45448                 );
45449             }
45450             if(!this.disable.sourceEdit){
45451                 tb.add(
45452                     '-',
45453                     btn('sourceedit', true, function(btn){
45454                         this.toggleSourceEdit(btn.pressed);
45455                     })
45456                 );
45457             }
45458         //}
45459         
45460         var smenu = { };
45461         // special menu.. - needs to be tidied up..
45462         if (!this.disable.special) {
45463             smenu = {
45464                 text: "&#169;",
45465                 cls: 'x-edit-none',
45466                 
45467                 menu : {
45468                     items : []
45469                 }
45470             };
45471             for (var i =0; i < this.specialChars.length; i++) {
45472                 smenu.menu.items.push({
45473                     
45474                     html: this.specialChars[i],
45475                     handler: function(a,b) {
45476                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
45477                         //editor.insertAtCursor(a.html);
45478                         
45479                     },
45480                     tabIndex:-1
45481                 });
45482             }
45483             
45484             
45485             tb.add(smenu);
45486             
45487             
45488         }
45489         
45490         var cmenu = { };
45491         if (!this.disable.cleanStyles) {
45492             cmenu = {
45493                 cls: 'x-btn-icon x-btn-clear',
45494                 
45495                 menu : {
45496                     items : []
45497                 }
45498             };
45499             for (var i =0; i < this.cleanStyles.length; i++) {
45500                 cmenu.menu.items.push({
45501                     actiontype : this.cleanStyles[i],
45502                     html: 'Remove ' + this.cleanStyles[i],
45503                     handler: function(a,b) {
45504 //                        Roo.log(a);
45505 //                        Roo.log(b);
45506                         var c = Roo.get(editorcore.doc.body);
45507                         c.select('[style]').each(function(s) {
45508                             s.dom.style.removeProperty(a.actiontype);
45509                         });
45510                         editorcore.syncValue();
45511                     },
45512                     tabIndex:-1
45513                 });
45514             }
45515              cmenu.menu.items.push({
45516                 actiontype : 'tablewidths',
45517                 html: 'Remove Table Widths',
45518                 handler: function(a,b) {
45519                     editorcore.cleanTableWidths();
45520                     editorcore.syncValue();
45521                 },
45522                 tabIndex:-1
45523             });
45524             cmenu.menu.items.push({
45525                 actiontype : 'word',
45526                 html: 'Remove MS Word Formating',
45527                 handler: function(a,b) {
45528                     editorcore.cleanWord();
45529                     editorcore.syncValue();
45530                 },
45531                 tabIndex:-1
45532             });
45533             
45534             cmenu.menu.items.push({
45535                 actiontype : 'all',
45536                 html: 'Remove All Styles',
45537                 handler: function(a,b) {
45538                     
45539                     var c = Roo.get(editorcore.doc.body);
45540                     c.select('[style]').each(function(s) {
45541                         s.dom.removeAttribute('style');
45542                     });
45543                     editorcore.syncValue();
45544                 },
45545                 tabIndex:-1
45546             });
45547             
45548             cmenu.menu.items.push({
45549                 actiontype : 'all',
45550                 html: 'Remove All CSS Classes',
45551                 handler: function(a,b) {
45552                     
45553                     var c = Roo.get(editorcore.doc.body);
45554                     c.select('[class]').each(function(s) {
45555                         s.dom.className = '';
45556                     });
45557                     editorcore.syncValue();
45558                 },
45559                 tabIndex:-1
45560             });
45561             
45562              cmenu.menu.items.push({
45563                 actiontype : 'tidy',
45564                 html: 'Tidy HTML Source',
45565                 handler: function(a,b) {
45566                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
45567                     editorcore.syncValue();
45568                 },
45569                 tabIndex:-1
45570             });
45571             
45572             
45573             tb.add(cmenu);
45574         }
45575          
45576         if (!this.disable.specialElements) {
45577             var semenu = {
45578                 text: "Other;",
45579                 cls: 'x-edit-none',
45580                 menu : {
45581                     items : []
45582                 }
45583             };
45584             for (var i =0; i < this.specialElements.length; i++) {
45585                 semenu.menu.items.push(
45586                     Roo.apply({ 
45587                         handler: function(a,b) {
45588                             editor.insertAtCursor(this.ihtml);
45589                         }
45590                     }, this.specialElements[i])
45591                 );
45592                     
45593             }
45594             
45595             tb.add(semenu);
45596             
45597             
45598         }
45599          
45600         
45601         if (this.btns) {
45602             for(var i =0; i< this.btns.length;i++) {
45603                 var b = Roo.factory(this.btns[i],Roo.form);
45604                 b.cls =  'x-edit-none';
45605                 
45606                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
45607                     b.cls += ' x-init-enable';
45608                 }
45609                 
45610                 b.scope = editorcore;
45611                 tb.add(b);
45612             }
45613         
45614         }
45615         
45616         
45617         
45618         // disable everything...
45619         
45620         this.tb.items.each(function(item){
45621             
45622            if(
45623                 item.id != editorcore.frameId+ '-sourceedit' && 
45624                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
45625             ){
45626                 
45627                 item.disable();
45628             }
45629         });
45630         this.rendered = true;
45631         
45632         // the all the btns;
45633         editor.on('editorevent', this.updateToolbar, this);
45634         // other toolbars need to implement this..
45635         //editor.on('editmodechange', this.updateToolbar, this);
45636     },
45637     
45638     
45639     relayBtnCmd : function(btn) {
45640         this.editorcore.relayCmd(btn.cmd);
45641     },
45642     // private used internally
45643     createLink : function(){
45644         Roo.log("create link?");
45645         var url = prompt(this.createLinkText, this.defaultLinkValue);
45646         if(url && url != 'http:/'+'/'){
45647             this.editorcore.relayCmd('createlink', url);
45648         }
45649     },
45650
45651     
45652     /**
45653      * Protected method that will not generally be called directly. It triggers
45654      * a toolbar update by reading the markup state of the current selection in the editor.
45655      */
45656     updateToolbar: function(){
45657
45658         if(!this.editorcore.activated){
45659             this.editor.onFirstFocus();
45660             return;
45661         }
45662
45663         var btns = this.tb.items.map, 
45664             doc = this.editorcore.doc,
45665             frameId = this.editorcore.frameId;
45666
45667         if(!this.disable.font && !Roo.isSafari){
45668             /*
45669             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
45670             if(name != this.fontSelect.dom.value){
45671                 this.fontSelect.dom.value = name;
45672             }
45673             */
45674         }
45675         if(!this.disable.format){
45676             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
45677             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
45678             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
45679             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
45680         }
45681         if(!this.disable.alignments){
45682             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
45683             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
45684             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
45685         }
45686         if(!Roo.isSafari && !this.disable.lists){
45687             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
45688             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
45689         }
45690         
45691         var ans = this.editorcore.getAllAncestors();
45692         if (this.formatCombo) {
45693             
45694             
45695             var store = this.formatCombo.store;
45696             this.formatCombo.setValue("");
45697             for (var i =0; i < ans.length;i++) {
45698                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
45699                     // select it..
45700                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
45701                     break;
45702                 }
45703             }
45704         }
45705         
45706         
45707         
45708         // hides menus... - so this cant be on a menu...
45709         Roo.menu.MenuMgr.hideAll();
45710
45711         //this.editorsyncValue();
45712     },
45713    
45714     
45715     createFontOptions : function(){
45716         var buf = [], fs = this.fontFamilies, ff, lc;
45717         
45718         
45719         
45720         for(var i = 0, len = fs.length; i< len; i++){
45721             ff = fs[i];
45722             lc = ff.toLowerCase();
45723             buf.push(
45724                 '<option value="',lc,'" style="font-family:',ff,';"',
45725                     (this.defaultFont == lc ? ' selected="true">' : '>'),
45726                     ff,
45727                 '</option>'
45728             );
45729         }
45730         return buf.join('');
45731     },
45732     
45733     toggleSourceEdit : function(sourceEditMode){
45734         
45735         Roo.log("toolbar toogle");
45736         if(sourceEditMode === undefined){
45737             sourceEditMode = !this.sourceEditMode;
45738         }
45739         this.sourceEditMode = sourceEditMode === true;
45740         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
45741         // just toggle the button?
45742         if(btn.pressed !== this.sourceEditMode){
45743             btn.toggle(this.sourceEditMode);
45744             return;
45745         }
45746         
45747         if(sourceEditMode){
45748             Roo.log("disabling buttons");
45749             this.tb.items.each(function(item){
45750                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
45751                     item.disable();
45752                 }
45753             });
45754           
45755         }else{
45756             Roo.log("enabling buttons");
45757             if(this.editorcore.initialized){
45758                 this.tb.items.each(function(item){
45759                     item.enable();
45760                 });
45761             }
45762             
45763         }
45764         Roo.log("calling toggole on editor");
45765         // tell the editor that it's been pressed..
45766         this.editor.toggleSourceEdit(sourceEditMode);
45767        
45768     },
45769      /**
45770      * Object collection of toolbar tooltips for the buttons in the editor. The key
45771      * is the command id associated with that button and the value is a valid QuickTips object.
45772      * For example:
45773 <pre><code>
45774 {
45775     bold : {
45776         title: 'Bold (Ctrl+B)',
45777         text: 'Make the selected text bold.',
45778         cls: 'x-html-editor-tip'
45779     },
45780     italic : {
45781         title: 'Italic (Ctrl+I)',
45782         text: 'Make the selected text italic.',
45783         cls: 'x-html-editor-tip'
45784     },
45785     ...
45786 </code></pre>
45787     * @type Object
45788      */
45789     buttonTips : {
45790         bold : {
45791             title: 'Bold (Ctrl+B)',
45792             text: 'Make the selected text bold.',
45793             cls: 'x-html-editor-tip'
45794         },
45795         italic : {
45796             title: 'Italic (Ctrl+I)',
45797             text: 'Make the selected text italic.',
45798             cls: 'x-html-editor-tip'
45799         },
45800         underline : {
45801             title: 'Underline (Ctrl+U)',
45802             text: 'Underline the selected text.',
45803             cls: 'x-html-editor-tip'
45804         },
45805         strikethrough : {
45806             title: 'Strikethrough',
45807             text: 'Strikethrough the selected text.',
45808             cls: 'x-html-editor-tip'
45809         },
45810         increasefontsize : {
45811             title: 'Grow Text',
45812             text: 'Increase the font size.',
45813             cls: 'x-html-editor-tip'
45814         },
45815         decreasefontsize : {
45816             title: 'Shrink Text',
45817             text: 'Decrease the font size.',
45818             cls: 'x-html-editor-tip'
45819         },
45820         backcolor : {
45821             title: 'Text Highlight Color',
45822             text: 'Change the background color of the selected text.',
45823             cls: 'x-html-editor-tip'
45824         },
45825         forecolor : {
45826             title: 'Font Color',
45827             text: 'Change the color of the selected text.',
45828             cls: 'x-html-editor-tip'
45829         },
45830         justifyleft : {
45831             title: 'Align Text Left',
45832             text: 'Align text to the left.',
45833             cls: 'x-html-editor-tip'
45834         },
45835         justifycenter : {
45836             title: 'Center Text',
45837             text: 'Center text in the editor.',
45838             cls: 'x-html-editor-tip'
45839         },
45840         justifyright : {
45841             title: 'Align Text Right',
45842             text: 'Align text to the right.',
45843             cls: 'x-html-editor-tip'
45844         },
45845         insertunorderedlist : {
45846             title: 'Bullet List',
45847             text: 'Start a bulleted list.',
45848             cls: 'x-html-editor-tip'
45849         },
45850         insertorderedlist : {
45851             title: 'Numbered List',
45852             text: 'Start a numbered list.',
45853             cls: 'x-html-editor-tip'
45854         },
45855         createlink : {
45856             title: 'Hyperlink',
45857             text: 'Make the selected text a hyperlink.',
45858             cls: 'x-html-editor-tip'
45859         },
45860         sourceedit : {
45861             title: 'Source Edit',
45862             text: 'Switch to source editing mode.',
45863             cls: 'x-html-editor-tip'
45864         }
45865     },
45866     // private
45867     onDestroy : function(){
45868         if(this.rendered){
45869             
45870             this.tb.items.each(function(item){
45871                 if(item.menu){
45872                     item.menu.removeAll();
45873                     if(item.menu.el){
45874                         item.menu.el.destroy();
45875                     }
45876                 }
45877                 item.destroy();
45878             });
45879              
45880         }
45881     },
45882     onFirstFocus: function() {
45883         this.tb.items.each(function(item){
45884            item.enable();
45885         });
45886     }
45887 });
45888
45889
45890
45891
45892 // <script type="text/javascript">
45893 /*
45894  * Based on
45895  * Ext JS Library 1.1.1
45896  * Copyright(c) 2006-2007, Ext JS, LLC.
45897  *  
45898  
45899  */
45900
45901  
45902 /**
45903  * @class Roo.form.HtmlEditor.ToolbarContext
45904  * Context Toolbar
45905  * 
45906  * Usage:
45907  *
45908  new Roo.form.HtmlEditor({
45909     ....
45910     toolbars : [
45911         { xtype: 'ToolbarStandard', styles : {} }
45912         { xtype: 'ToolbarContext', disable : {} }
45913     ]
45914 })
45915
45916      
45917  * 
45918  * @config : {Object} disable List of elements to disable.. (not done yet.)
45919  * @config : {Object} styles  Map of styles available.
45920  * 
45921  */
45922
45923 Roo.form.HtmlEditor.ToolbarContext = function(config)
45924 {
45925     
45926     Roo.apply(this, config);
45927     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45928     // dont call parent... till later.
45929     this.styles = this.styles || {};
45930 }
45931
45932  
45933
45934 Roo.form.HtmlEditor.ToolbarContext.types = {
45935     'IMG' : {
45936         width : {
45937             title: "Width",
45938             width: 40
45939         },
45940         height:  {
45941             title: "Height",
45942             width: 40
45943         },
45944         align: {
45945             title: "Align",
45946             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
45947             width : 80
45948             
45949         },
45950         border: {
45951             title: "Border",
45952             width: 40
45953         },
45954         alt: {
45955             title: "Alt",
45956             width: 120
45957         },
45958         src : {
45959             title: "Src",
45960             width: 220
45961         }
45962         
45963     },
45964     'A' : {
45965         name : {
45966             title: "Name",
45967             width: 50
45968         },
45969         target:  {
45970             title: "Target",
45971             width: 120
45972         },
45973         href:  {
45974             title: "Href",
45975             width: 220
45976         } // border?
45977         
45978     },
45979     'TABLE' : {
45980         rows : {
45981             title: "Rows",
45982             width: 20
45983         },
45984         cols : {
45985             title: "Cols",
45986             width: 20
45987         },
45988         width : {
45989             title: "Width",
45990             width: 40
45991         },
45992         height : {
45993             title: "Height",
45994             width: 40
45995         },
45996         border : {
45997             title: "Border",
45998             width: 20
45999         }
46000     },
46001     'TD' : {
46002         width : {
46003             title: "Width",
46004             width: 40
46005         },
46006         height : {
46007             title: "Height",
46008             width: 40
46009         },   
46010         align: {
46011             title: "Align",
46012             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46013             width: 80
46014         },
46015         valign: {
46016             title: "Valign",
46017             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46018             width: 80
46019         },
46020         colspan: {
46021             title: "Colspan",
46022             width: 20
46023             
46024         },
46025          'font-family'  : {
46026             title : "Font",
46027             style : 'fontFamily',
46028             displayField: 'display',
46029             optname : 'font-family',
46030             width: 140
46031         }
46032     },
46033     'INPUT' : {
46034         name : {
46035             title: "name",
46036             width: 120
46037         },
46038         value : {
46039             title: "Value",
46040             width: 120
46041         },
46042         width : {
46043             title: "Width",
46044             width: 40
46045         }
46046     },
46047     'LABEL' : {
46048         'for' : {
46049             title: "For",
46050             width: 120
46051         }
46052     },
46053     'TEXTAREA' : {
46054           name : {
46055             title: "name",
46056             width: 120
46057         },
46058         rows : {
46059             title: "Rows",
46060             width: 20
46061         },
46062         cols : {
46063             title: "Cols",
46064             width: 20
46065         }
46066     },
46067     'SELECT' : {
46068         name : {
46069             title: "name",
46070             width: 120
46071         },
46072         selectoptions : {
46073             title: "Options",
46074             width: 200
46075         }
46076     },
46077     
46078     // should we really allow this??
46079     // should this just be 
46080     'BODY' : {
46081         title : {
46082             title: "Title",
46083             width: 200,
46084             disabled : true
46085         }
46086     },
46087     'SPAN' : {
46088         'font-family'  : {
46089             title : "Font",
46090             style : 'fontFamily',
46091             displayField: 'display',
46092             optname : 'font-family',
46093             width: 140
46094         }
46095     },
46096     'DIV' : {
46097         'font-family'  : {
46098             title : "Font",
46099             style : 'fontFamily',
46100             displayField: 'display',
46101             optname : 'font-family',
46102             width: 140
46103         }
46104     },
46105      'P' : {
46106         'font-family'  : {
46107             title : "Font",
46108             style : 'fontFamily',
46109             displayField: 'display',
46110             optname : 'font-family',
46111             width: 140
46112         }
46113     },
46114     
46115     '*' : {
46116         // empty..
46117     }
46118
46119 };
46120
46121 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46122 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46123
46124 Roo.form.HtmlEditor.ToolbarContext.options = {
46125         'font-family'  : [ 
46126                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46127                 [ 'Courier New', 'Courier New'],
46128                 [ 'Tahoma', 'Tahoma'],
46129                 [ 'Times New Roman,serif', 'Times'],
46130                 [ 'Verdana','Verdana' ]
46131         ]
46132 };
46133
46134 // fixme - these need to be configurable..
46135  
46136
46137 //Roo.form.HtmlEditor.ToolbarContext.types
46138
46139
46140 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46141     
46142     tb: false,
46143     
46144     rendered: false,
46145     
46146     editor : false,
46147     editorcore : false,
46148     /**
46149      * @cfg {Object} disable  List of toolbar elements to disable
46150          
46151      */
46152     disable : false,
46153     /**
46154      * @cfg {Object} styles List of styles 
46155      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46156      *
46157      * These must be defined in the page, so they get rendered correctly..
46158      * .headline { }
46159      * TD.underline { }
46160      * 
46161      */
46162     styles : false,
46163     
46164     options: false,
46165     
46166     toolbars : false,
46167     
46168     init : function(editor)
46169     {
46170         this.editor = editor;
46171         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46172         var editorcore = this.editorcore;
46173         
46174         var fid = editorcore.frameId;
46175         var etb = this;
46176         function btn(id, toggle, handler){
46177             var xid = fid + '-'+ id ;
46178             return {
46179                 id : xid,
46180                 cmd : id,
46181                 cls : 'x-btn-icon x-edit-'+id,
46182                 enableToggle:toggle !== false,
46183                 scope: editorcore, // was editor...
46184                 handler:handler||editorcore.relayBtnCmd,
46185                 clickEvent:'mousedown',
46186                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46187                 tabIndex:-1
46188             };
46189         }
46190         // create a new element.
46191         var wdiv = editor.wrap.createChild({
46192                 tag: 'div'
46193             }, editor.wrap.dom.firstChild.nextSibling, true);
46194         
46195         // can we do this more than once??
46196         
46197          // stop form submits
46198       
46199  
46200         // disable everything...
46201         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46202         this.toolbars = {};
46203            
46204         for (var i in  ty) {
46205           
46206             this.toolbars[i] = this.buildToolbar(ty[i],i);
46207         }
46208         this.tb = this.toolbars.BODY;
46209         this.tb.el.show();
46210         this.buildFooter();
46211         this.footer.show();
46212         editor.on('hide', function( ) { this.footer.hide() }, this);
46213         editor.on('show', function( ) { this.footer.show() }, this);
46214         
46215          
46216         this.rendered = true;
46217         
46218         // the all the btns;
46219         editor.on('editorevent', this.updateToolbar, this);
46220         // other toolbars need to implement this..
46221         //editor.on('editmodechange', this.updateToolbar, this);
46222     },
46223     
46224     
46225     
46226     /**
46227      * Protected method that will not generally be called directly. It triggers
46228      * a toolbar update by reading the markup state of the current selection in the editor.
46229      *
46230      * Note you can force an update by calling on('editorevent', scope, false)
46231      */
46232     updateToolbar: function(editor,ev,sel){
46233
46234         //Roo.log(ev);
46235         // capture mouse up - this is handy for selecting images..
46236         // perhaps should go somewhere else...
46237         if(!this.editorcore.activated){
46238              this.editor.onFirstFocus();
46239             return;
46240         }
46241         
46242         
46243         
46244         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46245         // selectNode - might want to handle IE?
46246         if (ev &&
46247             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46248             ev.target && ev.target.tagName == 'IMG') {
46249             // they have click on an image...
46250             // let's see if we can change the selection...
46251             sel = ev.target;
46252          
46253               var nodeRange = sel.ownerDocument.createRange();
46254             try {
46255                 nodeRange.selectNode(sel);
46256             } catch (e) {
46257                 nodeRange.selectNodeContents(sel);
46258             }
46259             //nodeRange.collapse(true);
46260             var s = this.editorcore.win.getSelection();
46261             s.removeAllRanges();
46262             s.addRange(nodeRange);
46263         }  
46264         
46265       
46266         var updateFooter = sel ? false : true;
46267         
46268         
46269         var ans = this.editorcore.getAllAncestors();
46270         
46271         // pick
46272         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46273         
46274         if (!sel) { 
46275             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46276             sel = sel ? sel : this.editorcore.doc.body;
46277             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46278             
46279         }
46280         // pick a menu that exists..
46281         var tn = sel.tagName.toUpperCase();
46282         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46283         
46284         tn = sel.tagName.toUpperCase();
46285         
46286         var lastSel = this.tb.selectedNode;
46287         
46288         this.tb.selectedNode = sel;
46289         
46290         // if current menu does not match..
46291         
46292         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46293                 
46294             this.tb.el.hide();
46295             ///console.log("show: " + tn);
46296             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46297             this.tb.el.show();
46298             // update name
46299             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46300             
46301             
46302             // update attributes
46303             if (this.tb.fields) {
46304                 this.tb.fields.each(function(e) {
46305                     if (e.stylename) {
46306                         e.setValue(sel.style[e.stylename]);
46307                         return;
46308                     } 
46309                    e.setValue(sel.getAttribute(e.attrname));
46310                 });
46311             }
46312             
46313             var hasStyles = false;
46314             for(var i in this.styles) {
46315                 hasStyles = true;
46316                 break;
46317             }
46318             
46319             // update styles
46320             if (hasStyles) { 
46321                 var st = this.tb.fields.item(0);
46322                 
46323                 st.store.removeAll();
46324                
46325                 
46326                 var cn = sel.className.split(/\s+/);
46327                 
46328                 var avs = [];
46329                 if (this.styles['*']) {
46330                     
46331                     Roo.each(this.styles['*'], function(v) {
46332                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46333                     });
46334                 }
46335                 if (this.styles[tn]) { 
46336                     Roo.each(this.styles[tn], function(v) {
46337                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46338                     });
46339                 }
46340                 
46341                 st.store.loadData(avs);
46342                 st.collapse();
46343                 st.setValue(cn);
46344             }
46345             // flag our selected Node.
46346             this.tb.selectedNode = sel;
46347            
46348            
46349             Roo.menu.MenuMgr.hideAll();
46350
46351         }
46352         
46353         if (!updateFooter) {
46354             //this.footDisp.dom.innerHTML = ''; 
46355             return;
46356         }
46357         // update the footer
46358         //
46359         var html = '';
46360         
46361         this.footerEls = ans.reverse();
46362         Roo.each(this.footerEls, function(a,i) {
46363             if (!a) { return; }
46364             html += html.length ? ' &gt; '  :  '';
46365             
46366             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46367             
46368         });
46369        
46370         // 
46371         var sz = this.footDisp.up('td').getSize();
46372         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46373         this.footDisp.dom.style.marginLeft = '5px';
46374         
46375         this.footDisp.dom.style.overflow = 'hidden';
46376         
46377         this.footDisp.dom.innerHTML = html;
46378             
46379         //this.editorsyncValue();
46380     },
46381      
46382     
46383    
46384        
46385     // private
46386     onDestroy : function(){
46387         if(this.rendered){
46388             
46389             this.tb.items.each(function(item){
46390                 if(item.menu){
46391                     item.menu.removeAll();
46392                     if(item.menu.el){
46393                         item.menu.el.destroy();
46394                     }
46395                 }
46396                 item.destroy();
46397             });
46398              
46399         }
46400     },
46401     onFirstFocus: function() {
46402         // need to do this for all the toolbars..
46403         this.tb.items.each(function(item){
46404            item.enable();
46405         });
46406     },
46407     buildToolbar: function(tlist, nm)
46408     {
46409         var editor = this.editor;
46410         var editorcore = this.editorcore;
46411          // create a new element.
46412         var wdiv = editor.wrap.createChild({
46413                 tag: 'div'
46414             }, editor.wrap.dom.firstChild.nextSibling, true);
46415         
46416        
46417         var tb = new Roo.Toolbar(wdiv);
46418         // add the name..
46419         
46420         tb.add(nm+ ":&nbsp;");
46421         
46422         var styles = [];
46423         for(var i in this.styles) {
46424             styles.push(i);
46425         }
46426         
46427         // styles...
46428         if (styles && styles.length) {
46429             
46430             // this needs a multi-select checkbox...
46431             tb.addField( new Roo.form.ComboBox({
46432                 store: new Roo.data.SimpleStore({
46433                     id : 'val',
46434                     fields: ['val', 'selected'],
46435                     data : [] 
46436                 }),
46437                 name : '-roo-edit-className',
46438                 attrname : 'className',
46439                 displayField: 'val',
46440                 typeAhead: false,
46441                 mode: 'local',
46442                 editable : false,
46443                 triggerAction: 'all',
46444                 emptyText:'Select Style',
46445                 selectOnFocus:true,
46446                 width: 130,
46447                 listeners : {
46448                     'select': function(c, r, i) {
46449                         // initial support only for on class per el..
46450                         tb.selectedNode.className =  r ? r.get('val') : '';
46451                         editorcore.syncValue();
46452                     }
46453                 }
46454     
46455             }));
46456         }
46457         
46458         var tbc = Roo.form.HtmlEditor.ToolbarContext;
46459         var tbops = tbc.options;
46460         
46461         for (var i in tlist) {
46462             
46463             var item = tlist[i];
46464             tb.add(item.title + ":&nbsp;");
46465             
46466             
46467             //optname == used so you can configure the options available..
46468             var opts = item.opts ? item.opts : false;
46469             if (item.optname) {
46470                 opts = tbops[item.optname];
46471            
46472             }
46473             
46474             if (opts) {
46475                 // opts == pulldown..
46476                 tb.addField( new Roo.form.ComboBox({
46477                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
46478                         id : 'val',
46479                         fields: ['val', 'display'],
46480                         data : opts  
46481                     }),
46482                     name : '-roo-edit-' + i,
46483                     attrname : i,
46484                     stylename : item.style ? item.style : false,
46485                     displayField: item.displayField ? item.displayField : 'val',
46486                     valueField :  'val',
46487                     typeAhead: false,
46488                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
46489                     editable : false,
46490                     triggerAction: 'all',
46491                     emptyText:'Select',
46492                     selectOnFocus:true,
46493                     width: item.width ? item.width  : 130,
46494                     listeners : {
46495                         'select': function(c, r, i) {
46496                             if (c.stylename) {
46497                                 tb.selectedNode.style[c.stylename] =  r.get('val');
46498                                 return;
46499                             }
46500                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
46501                         }
46502                     }
46503
46504                 }));
46505                 continue;
46506                     
46507                  
46508                 
46509                 tb.addField( new Roo.form.TextField({
46510                     name: i,
46511                     width: 100,
46512                     //allowBlank:false,
46513                     value: ''
46514                 }));
46515                 continue;
46516             }
46517             tb.addField( new Roo.form.TextField({
46518                 name: '-roo-edit-' + i,
46519                 attrname : i,
46520                 
46521                 width: item.width,
46522                 //allowBlank:true,
46523                 value: '',
46524                 listeners: {
46525                     'change' : function(f, nv, ov) {
46526                         tb.selectedNode.setAttribute(f.attrname, nv);
46527                         editorcore.syncValue();
46528                     }
46529                 }
46530             }));
46531              
46532         }
46533         
46534         var _this = this;
46535         
46536         if(nm == 'BODY'){
46537             tb.addSeparator();
46538         
46539             tb.addButton( {
46540                 text: 'Stylesheets',
46541
46542                 listeners : {
46543                     click : function ()
46544                     {
46545                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
46546                     }
46547                 }
46548             });
46549         }
46550         
46551         tb.addFill();
46552         tb.addButton( {
46553             text: 'Remove Tag',
46554     
46555             listeners : {
46556                 click : function ()
46557                 {
46558                     // remove
46559                     // undo does not work.
46560                      
46561                     var sn = tb.selectedNode;
46562                     
46563                     var pn = sn.parentNode;
46564                     
46565                     var stn =  sn.childNodes[0];
46566                     var en = sn.childNodes[sn.childNodes.length - 1 ];
46567                     while (sn.childNodes.length) {
46568                         var node = sn.childNodes[0];
46569                         sn.removeChild(node);
46570                         //Roo.log(node);
46571                         pn.insertBefore(node, sn);
46572                         
46573                     }
46574                     pn.removeChild(sn);
46575                     var range = editorcore.createRange();
46576         
46577                     range.setStart(stn,0);
46578                     range.setEnd(en,0); //????
46579                     //range.selectNode(sel);
46580                     
46581                     
46582                     var selection = editorcore.getSelection();
46583                     selection.removeAllRanges();
46584                     selection.addRange(range);
46585                     
46586                     
46587                     
46588                     //_this.updateToolbar(null, null, pn);
46589                     _this.updateToolbar(null, null, null);
46590                     _this.footDisp.dom.innerHTML = ''; 
46591                 }
46592             }
46593             
46594                     
46595                 
46596             
46597         });
46598         
46599         
46600         tb.el.on('click', function(e){
46601             e.preventDefault(); // what does this do?
46602         });
46603         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
46604         tb.el.hide();
46605         tb.name = nm;
46606         // dont need to disable them... as they will get hidden
46607         return tb;
46608          
46609         
46610     },
46611     buildFooter : function()
46612     {
46613         
46614         var fel = this.editor.wrap.createChild();
46615         this.footer = new Roo.Toolbar(fel);
46616         // toolbar has scrolly on left / right?
46617         var footDisp= new Roo.Toolbar.Fill();
46618         var _t = this;
46619         this.footer.add(
46620             {
46621                 text : '&lt;',
46622                 xtype: 'Button',
46623                 handler : function() {
46624                     _t.footDisp.scrollTo('left',0,true)
46625                 }
46626             }
46627         );
46628         this.footer.add( footDisp );
46629         this.footer.add( 
46630             {
46631                 text : '&gt;',
46632                 xtype: 'Button',
46633                 handler : function() {
46634                     // no animation..
46635                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
46636                 }
46637             }
46638         );
46639         var fel = Roo.get(footDisp.el);
46640         fel.addClass('x-editor-context');
46641         this.footDispWrap = fel; 
46642         this.footDispWrap.overflow  = 'hidden';
46643         
46644         this.footDisp = fel.createChild();
46645         this.footDispWrap.on('click', this.onContextClick, this)
46646         
46647         
46648     },
46649     onContextClick : function (ev,dom)
46650     {
46651         ev.preventDefault();
46652         var  cn = dom.className;
46653         //Roo.log(cn);
46654         if (!cn.match(/x-ed-loc-/)) {
46655             return;
46656         }
46657         var n = cn.split('-').pop();
46658         var ans = this.footerEls;
46659         var sel = ans[n];
46660         
46661          // pick
46662         var range = this.editorcore.createRange();
46663         
46664         range.selectNodeContents(sel);
46665         //range.selectNode(sel);
46666         
46667         
46668         var selection = this.editorcore.getSelection();
46669         selection.removeAllRanges();
46670         selection.addRange(range);
46671         
46672         
46673         
46674         this.updateToolbar(null, null, sel);
46675         
46676         
46677     }
46678     
46679     
46680     
46681     
46682     
46683 });
46684
46685
46686
46687
46688
46689 /*
46690  * Based on:
46691  * Ext JS Library 1.1.1
46692  * Copyright(c) 2006-2007, Ext JS, LLC.
46693  *
46694  * Originally Released Under LGPL - original licence link has changed is not relivant.
46695  *
46696  * Fork - LGPL
46697  * <script type="text/javascript">
46698  */
46699  
46700 /**
46701  * @class Roo.form.BasicForm
46702  * @extends Roo.util.Observable
46703  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
46704  * @constructor
46705  * @param {String/HTMLElement/Roo.Element} el The form element or its id
46706  * @param {Object} config Configuration options
46707  */
46708 Roo.form.BasicForm = function(el, config){
46709     this.allItems = [];
46710     this.childForms = [];
46711     Roo.apply(this, config);
46712     /*
46713      * The Roo.form.Field items in this form.
46714      * @type MixedCollection
46715      */
46716      
46717      
46718     this.items = new Roo.util.MixedCollection(false, function(o){
46719         return o.id || (o.id = Roo.id());
46720     });
46721     this.addEvents({
46722         /**
46723          * @event beforeaction
46724          * Fires before any action is performed. Return false to cancel the action.
46725          * @param {Form} this
46726          * @param {Action} action The action to be performed
46727          */
46728         beforeaction: true,
46729         /**
46730          * @event actionfailed
46731          * Fires when an action fails.
46732          * @param {Form} this
46733          * @param {Action} action The action that failed
46734          */
46735         actionfailed : true,
46736         /**
46737          * @event actioncomplete
46738          * Fires when an action is completed.
46739          * @param {Form} this
46740          * @param {Action} action The action that completed
46741          */
46742         actioncomplete : true
46743     });
46744     if(el){
46745         this.initEl(el);
46746     }
46747     Roo.form.BasicForm.superclass.constructor.call(this);
46748 };
46749
46750 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
46751     /**
46752      * @cfg {String} method
46753      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
46754      */
46755     /**
46756      * @cfg {DataReader} reader
46757      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
46758      * This is optional as there is built-in support for processing JSON.
46759      */
46760     /**
46761      * @cfg {DataReader} errorReader
46762      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
46763      * This is completely optional as there is built-in support for processing JSON.
46764      */
46765     /**
46766      * @cfg {String} url
46767      * The URL to use for form actions if one isn't supplied in the action options.
46768      */
46769     /**
46770      * @cfg {Boolean} fileUpload
46771      * Set to true if this form is a file upload.
46772      */
46773      
46774     /**
46775      * @cfg {Object} baseParams
46776      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
46777      */
46778      /**
46779      
46780     /**
46781      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
46782      */
46783     timeout: 30,
46784
46785     // private
46786     activeAction : null,
46787
46788     /**
46789      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
46790      * or setValues() data instead of when the form was first created.
46791      */
46792     trackResetOnLoad : false,
46793     
46794     
46795     /**
46796      * childForms - used for multi-tab forms
46797      * @type {Array}
46798      */
46799     childForms : false,
46800     
46801     /**
46802      * allItems - full list of fields.
46803      * @type {Array}
46804      */
46805     allItems : false,
46806     
46807     /**
46808      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
46809      * element by passing it or its id or mask the form itself by passing in true.
46810      * @type Mixed
46811      */
46812     waitMsgTarget : false,
46813     
46814     /**
46815      * @type Boolean
46816      */
46817     disableMask : false,
46818
46819     // private
46820     initEl : function(el){
46821         this.el = Roo.get(el);
46822         this.id = this.el.id || Roo.id();
46823         this.el.on('submit', this.onSubmit, this);
46824         this.el.addClass('x-form');
46825     },
46826
46827     // private
46828     onSubmit : function(e){
46829         e.stopEvent();
46830     },
46831
46832     /**
46833      * Returns true if client-side validation on the form is successful.
46834      * @return Boolean
46835      */
46836     isValid : function(){
46837         var valid = true;
46838         this.items.each(function(f){
46839            if(!f.validate()){
46840                valid = false;
46841            }
46842         });
46843         return valid;
46844     },
46845
46846     /**
46847      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
46848      * @return Boolean
46849      */
46850     isDirty : function(){
46851         var dirty = false;
46852         this.items.each(function(f){
46853            if(f.isDirty()){
46854                dirty = true;
46855                return false;
46856            }
46857         });
46858         return dirty;
46859     },
46860     
46861     /**
46862      * Returns true if any fields in this form have changed since their original load. (New version)
46863      * @return Boolean
46864      */
46865     
46866     hasChanged : function()
46867     {
46868         var dirty = false;
46869         this.items.each(function(f){
46870            if(f.hasChanged()){
46871                dirty = true;
46872                return false;
46873            }
46874         });
46875         return dirty;
46876         
46877     },
46878     /**
46879      * Resets all hasChanged to 'false' -
46880      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
46881      * So hasChanged storage is only to be used for this purpose
46882      * @return Boolean
46883      */
46884     resetHasChanged : function()
46885     {
46886         this.items.each(function(f){
46887            f.resetHasChanged();
46888         });
46889         
46890     },
46891     
46892     
46893     /**
46894      * Performs a predefined action (submit or load) or custom actions you define on this form.
46895      * @param {String} actionName The name of the action type
46896      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
46897      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
46898      * accept other config options):
46899      * <pre>
46900 Property          Type             Description
46901 ----------------  ---------------  ----------------------------------------------------------------------------------
46902 url               String           The url for the action (defaults to the form's url)
46903 method            String           The form method to use (defaults to the form's method, or POST if not defined)
46904 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
46905 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
46906                                    validate the form on the client (defaults to false)
46907      * </pre>
46908      * @return {BasicForm} this
46909      */
46910     doAction : function(action, options){
46911         if(typeof action == 'string'){
46912             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
46913         }
46914         if(this.fireEvent('beforeaction', this, action) !== false){
46915             this.beforeAction(action);
46916             action.run.defer(100, action);
46917         }
46918         return this;
46919     },
46920
46921     /**
46922      * Shortcut to do a submit action.
46923      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
46924      * @return {BasicForm} this
46925      */
46926     submit : function(options){
46927         this.doAction('submit', options);
46928         return this;
46929     },
46930
46931     /**
46932      * Shortcut to do a load action.
46933      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
46934      * @return {BasicForm} this
46935      */
46936     load : function(options){
46937         this.doAction('load', options);
46938         return this;
46939     },
46940
46941     /**
46942      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
46943      * @param {Record} record The record to edit
46944      * @return {BasicForm} this
46945      */
46946     updateRecord : function(record){
46947         record.beginEdit();
46948         var fs = record.fields;
46949         fs.each(function(f){
46950             var field = this.findField(f.name);
46951             if(field){
46952                 record.set(f.name, field.getValue());
46953             }
46954         }, this);
46955         record.endEdit();
46956         return this;
46957     },
46958
46959     /**
46960      * Loads an Roo.data.Record into this form.
46961      * @param {Record} record The record to load
46962      * @return {BasicForm} this
46963      */
46964     loadRecord : function(record){
46965         this.setValues(record.data);
46966         return this;
46967     },
46968
46969     // private
46970     beforeAction : function(action){
46971         var o = action.options;
46972         
46973         if(!this.disableMask) {
46974             if(this.waitMsgTarget === true){
46975                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
46976             }else if(this.waitMsgTarget){
46977                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
46978                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
46979             }else {
46980                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
46981             }
46982         }
46983         
46984          
46985     },
46986
46987     // private
46988     afterAction : function(action, success){
46989         this.activeAction = null;
46990         var o = action.options;
46991         
46992         if(!this.disableMask) {
46993             if(this.waitMsgTarget === true){
46994                 this.el.unmask();
46995             }else if(this.waitMsgTarget){
46996                 this.waitMsgTarget.unmask();
46997             }else{
46998                 Roo.MessageBox.updateProgress(1);
46999                 Roo.MessageBox.hide();
47000             }
47001         }
47002         
47003         if(success){
47004             if(o.reset){
47005                 this.reset();
47006             }
47007             Roo.callback(o.success, o.scope, [this, action]);
47008             this.fireEvent('actioncomplete', this, action);
47009             
47010         }else{
47011             
47012             // failure condition..
47013             // we have a scenario where updates need confirming.
47014             // eg. if a locking scenario exists..
47015             // we look for { errors : { needs_confirm : true }} in the response.
47016             if (
47017                 (typeof(action.result) != 'undefined')  &&
47018                 (typeof(action.result.errors) != 'undefined')  &&
47019                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47020            ){
47021                 var _t = this;
47022                 Roo.MessageBox.confirm(
47023                     "Change requires confirmation",
47024                     action.result.errorMsg,
47025                     function(r) {
47026                         if (r != 'yes') {
47027                             return;
47028                         }
47029                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47030                     }
47031                     
47032                 );
47033                 
47034                 
47035                 
47036                 return;
47037             }
47038             
47039             Roo.callback(o.failure, o.scope, [this, action]);
47040             // show an error message if no failed handler is set..
47041             if (!this.hasListener('actionfailed')) {
47042                 Roo.MessageBox.alert("Error",
47043                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47044                         action.result.errorMsg :
47045                         "Saving Failed, please check your entries or try again"
47046                 );
47047             }
47048             
47049             this.fireEvent('actionfailed', this, action);
47050         }
47051         
47052     },
47053
47054     /**
47055      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47056      * @param {String} id The value to search for
47057      * @return Field
47058      */
47059     findField : function(id){
47060         var field = this.items.get(id);
47061         if(!field){
47062             this.items.each(function(f){
47063                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47064                     field = f;
47065                     return false;
47066                 }
47067             });
47068         }
47069         return field || null;
47070     },
47071
47072     /**
47073      * Add a secondary form to this one, 
47074      * Used to provide tabbed forms. One form is primary, with hidden values 
47075      * which mirror the elements from the other forms.
47076      * 
47077      * @param {Roo.form.Form} form to add.
47078      * 
47079      */
47080     addForm : function(form)
47081     {
47082        
47083         if (this.childForms.indexOf(form) > -1) {
47084             // already added..
47085             return;
47086         }
47087         this.childForms.push(form);
47088         var n = '';
47089         Roo.each(form.allItems, function (fe) {
47090             
47091             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47092             if (this.findField(n)) { // already added..
47093                 return;
47094             }
47095             var add = new Roo.form.Hidden({
47096                 name : n
47097             });
47098             add.render(this.el);
47099             
47100             this.add( add );
47101         }, this);
47102         
47103     },
47104     /**
47105      * Mark fields in this form invalid in bulk.
47106      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47107      * @return {BasicForm} this
47108      */
47109     markInvalid : function(errors){
47110         if(errors instanceof Array){
47111             for(var i = 0, len = errors.length; i < len; i++){
47112                 var fieldError = errors[i];
47113                 var f = this.findField(fieldError.id);
47114                 if(f){
47115                     f.markInvalid(fieldError.msg);
47116                 }
47117             }
47118         }else{
47119             var field, id;
47120             for(id in errors){
47121                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47122                     field.markInvalid(errors[id]);
47123                 }
47124             }
47125         }
47126         Roo.each(this.childForms || [], function (f) {
47127             f.markInvalid(errors);
47128         });
47129         
47130         return this;
47131     },
47132
47133     /**
47134      * Set values for fields in this form in bulk.
47135      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47136      * @return {BasicForm} this
47137      */
47138     setValues : function(values){
47139         if(values instanceof Array){ // array of objects
47140             for(var i = 0, len = values.length; i < len; i++){
47141                 var v = values[i];
47142                 var f = this.findField(v.id);
47143                 if(f){
47144                     f.setValue(v.value);
47145                     if(this.trackResetOnLoad){
47146                         f.originalValue = f.getValue();
47147                     }
47148                 }
47149             }
47150         }else{ // object hash
47151             var field, id;
47152             for(id in values){
47153                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47154                     
47155                     if (field.setFromData && 
47156                         field.valueField && 
47157                         field.displayField &&
47158                         // combos' with local stores can 
47159                         // be queried via setValue()
47160                         // to set their value..
47161                         (field.store && !field.store.isLocal)
47162                         ) {
47163                         // it's a combo
47164                         var sd = { };
47165                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47166                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47167                         field.setFromData(sd);
47168                         
47169                     } else {
47170                         field.setValue(values[id]);
47171                     }
47172                     
47173                     
47174                     if(this.trackResetOnLoad){
47175                         field.originalValue = field.getValue();
47176                     }
47177                 }
47178             }
47179         }
47180         this.resetHasChanged();
47181         
47182         
47183         Roo.each(this.childForms || [], function (f) {
47184             f.setValues(values);
47185             f.resetHasChanged();
47186         });
47187                 
47188         return this;
47189     },
47190
47191     /**
47192      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47193      * they are returned as an array.
47194      * @param {Boolean} asString
47195      * @return {Object}
47196      */
47197     getValues : function(asString){
47198         if (this.childForms) {
47199             // copy values from the child forms
47200             Roo.each(this.childForms, function (f) {
47201                 this.setValues(f.getValues());
47202             }, this);
47203         }
47204         
47205         
47206         
47207         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47208         if(asString === true){
47209             return fs;
47210         }
47211         return Roo.urlDecode(fs);
47212     },
47213     
47214     /**
47215      * Returns the fields in this form as an object with key/value pairs. 
47216      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47217      * @return {Object}
47218      */
47219     getFieldValues : function(with_hidden)
47220     {
47221         if (this.childForms) {
47222             // copy values from the child forms
47223             // should this call getFieldValues - probably not as we do not currently copy
47224             // hidden fields when we generate..
47225             Roo.each(this.childForms, function (f) {
47226                 this.setValues(f.getValues());
47227             }, this);
47228         }
47229         
47230         var ret = {};
47231         this.items.each(function(f){
47232             if (!f.getName()) {
47233                 return;
47234             }
47235             var v = f.getValue();
47236             if (f.inputType =='radio') {
47237                 if (typeof(ret[f.getName()]) == 'undefined') {
47238                     ret[f.getName()] = ''; // empty..
47239                 }
47240                 
47241                 if (!f.el.dom.checked) {
47242                     return;
47243                     
47244                 }
47245                 v = f.el.dom.value;
47246                 
47247             }
47248             
47249             // not sure if this supported any more..
47250             if ((typeof(v) == 'object') && f.getRawValue) {
47251                 v = f.getRawValue() ; // dates..
47252             }
47253             // combo boxes where name != hiddenName...
47254             if (f.name != f.getName()) {
47255                 ret[f.name] = f.getRawValue();
47256             }
47257             ret[f.getName()] = v;
47258         });
47259         
47260         return ret;
47261     },
47262
47263     /**
47264      * Clears all invalid messages in this form.
47265      * @return {BasicForm} this
47266      */
47267     clearInvalid : function(){
47268         this.items.each(function(f){
47269            f.clearInvalid();
47270         });
47271         
47272         Roo.each(this.childForms || [], function (f) {
47273             f.clearInvalid();
47274         });
47275         
47276         
47277         return this;
47278     },
47279
47280     /**
47281      * Resets this form.
47282      * @return {BasicForm} this
47283      */
47284     reset : function(){
47285         this.items.each(function(f){
47286             f.reset();
47287         });
47288         
47289         Roo.each(this.childForms || [], function (f) {
47290             f.reset();
47291         });
47292         this.resetHasChanged();
47293         
47294         return this;
47295     },
47296
47297     /**
47298      * Add Roo.form components to this form.
47299      * @param {Field} field1
47300      * @param {Field} field2 (optional)
47301      * @param {Field} etc (optional)
47302      * @return {BasicForm} this
47303      */
47304     add : function(){
47305         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47306         return this;
47307     },
47308
47309
47310     /**
47311      * Removes a field from the items collection (does NOT remove its markup).
47312      * @param {Field} field
47313      * @return {BasicForm} this
47314      */
47315     remove : function(field){
47316         this.items.remove(field);
47317         return this;
47318     },
47319
47320     /**
47321      * Looks at the fields in this form, checks them for an id attribute,
47322      * and calls applyTo on the existing dom element with that id.
47323      * @return {BasicForm} this
47324      */
47325     render : function(){
47326         this.items.each(function(f){
47327             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47328                 f.applyTo(f.id);
47329             }
47330         });
47331         return this;
47332     },
47333
47334     /**
47335      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47336      * @param {Object} values
47337      * @return {BasicForm} this
47338      */
47339     applyToFields : function(o){
47340         this.items.each(function(f){
47341            Roo.apply(f, o);
47342         });
47343         return this;
47344     },
47345
47346     /**
47347      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47348      * @param {Object} values
47349      * @return {BasicForm} this
47350      */
47351     applyIfToFields : function(o){
47352         this.items.each(function(f){
47353            Roo.applyIf(f, o);
47354         });
47355         return this;
47356     }
47357 });
47358
47359 // back compat
47360 Roo.BasicForm = Roo.form.BasicForm;/*
47361  * Based on:
47362  * Ext JS Library 1.1.1
47363  * Copyright(c) 2006-2007, Ext JS, LLC.
47364  *
47365  * Originally Released Under LGPL - original licence link has changed is not relivant.
47366  *
47367  * Fork - LGPL
47368  * <script type="text/javascript">
47369  */
47370
47371 /**
47372  * @class Roo.form.Form
47373  * @extends Roo.form.BasicForm
47374  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
47375  * @constructor
47376  * @param {Object} config Configuration options
47377  */
47378 Roo.form.Form = function(config){
47379     var xitems =  [];
47380     if (config.items) {
47381         xitems = config.items;
47382         delete config.items;
47383     }
47384    
47385     
47386     Roo.form.Form.superclass.constructor.call(this, null, config);
47387     this.url = this.url || this.action;
47388     if(!this.root){
47389         this.root = new Roo.form.Layout(Roo.applyIf({
47390             id: Roo.id()
47391         }, config));
47392     }
47393     this.active = this.root;
47394     /**
47395      * Array of all the buttons that have been added to this form via {@link addButton}
47396      * @type Array
47397      */
47398     this.buttons = [];
47399     this.allItems = [];
47400     this.addEvents({
47401         /**
47402          * @event clientvalidation
47403          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
47404          * @param {Form} this
47405          * @param {Boolean} valid true if the form has passed client-side validation
47406          */
47407         clientvalidation: true,
47408         /**
47409          * @event rendered
47410          * Fires when the form is rendered
47411          * @param {Roo.form.Form} form
47412          */
47413         rendered : true
47414     });
47415     
47416     if (this.progressUrl) {
47417             // push a hidden field onto the list of fields..
47418             this.addxtype( {
47419                     xns: Roo.form, 
47420                     xtype : 'Hidden', 
47421                     name : 'UPLOAD_IDENTIFIER' 
47422             });
47423         }
47424         
47425     
47426     Roo.each(xitems, this.addxtype, this);
47427     
47428     
47429     
47430 };
47431
47432 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
47433     /**
47434      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
47435      */
47436     /**
47437      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
47438      */
47439     /**
47440      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
47441      */
47442     buttonAlign:'center',
47443
47444     /**
47445      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
47446      */
47447     minButtonWidth:75,
47448
47449     /**
47450      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
47451      * This property cascades to child containers if not set.
47452      */
47453     labelAlign:'left',
47454
47455     /**
47456      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
47457      * fires a looping event with that state. This is required to bind buttons to the valid
47458      * state using the config value formBind:true on the button.
47459      */
47460     monitorValid : false,
47461
47462     /**
47463      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
47464      */
47465     monitorPoll : 200,
47466     
47467     /**
47468      * @cfg {String} progressUrl - Url to return progress data 
47469      */
47470     
47471     progressUrl : false,
47472     /**
47473      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
47474      * sending a formdata with extra parameters - eg uploaded elements.
47475      */
47476     
47477     formData : false,
47478     
47479     /**
47480      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
47481      * fields are added and the column is closed. If no fields are passed the column remains open
47482      * until end() is called.
47483      * @param {Object} config The config to pass to the column
47484      * @param {Field} field1 (optional)
47485      * @param {Field} field2 (optional)
47486      * @param {Field} etc (optional)
47487      * @return Column The column container object
47488      */
47489     column : function(c){
47490         var col = new Roo.form.Column(c);
47491         this.start(col);
47492         if(arguments.length > 1){ // duplicate code required because of Opera
47493             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47494             this.end();
47495         }
47496         return col;
47497     },
47498
47499     /**
47500      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
47501      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
47502      * until end() is called.
47503      * @param {Object} config The config to pass to the fieldset
47504      * @param {Field} field1 (optional)
47505      * @param {Field} field2 (optional)
47506      * @param {Field} etc (optional)
47507      * @return FieldSet The fieldset container object
47508      */
47509     fieldset : function(c){
47510         var fs = new Roo.form.FieldSet(c);
47511         this.start(fs);
47512         if(arguments.length > 1){ // duplicate code required because of Opera
47513             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47514             this.end();
47515         }
47516         return fs;
47517     },
47518
47519     /**
47520      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
47521      * fields are added and the container is closed. If no fields are passed the container remains open
47522      * until end() is called.
47523      * @param {Object} config The config to pass to the Layout
47524      * @param {Field} field1 (optional)
47525      * @param {Field} field2 (optional)
47526      * @param {Field} etc (optional)
47527      * @return Layout The container object
47528      */
47529     container : function(c){
47530         var l = new Roo.form.Layout(c);
47531         this.start(l);
47532         if(arguments.length > 1){ // duplicate code required because of Opera
47533             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
47534             this.end();
47535         }
47536         return l;
47537     },
47538
47539     /**
47540      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
47541      * @param {Object} container A Roo.form.Layout or subclass of Layout
47542      * @return {Form} this
47543      */
47544     start : function(c){
47545         // cascade label info
47546         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
47547         this.active.stack.push(c);
47548         c.ownerCt = this.active;
47549         this.active = c;
47550         return this;
47551     },
47552
47553     /**
47554      * Closes the current open container
47555      * @return {Form} this
47556      */
47557     end : function(){
47558         if(this.active == this.root){
47559             return this;
47560         }
47561         this.active = this.active.ownerCt;
47562         return this;
47563     },
47564
47565     /**
47566      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
47567      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
47568      * as the label of the field.
47569      * @param {Field} field1
47570      * @param {Field} field2 (optional)
47571      * @param {Field} etc. (optional)
47572      * @return {Form} this
47573      */
47574     add : function(){
47575         this.active.stack.push.apply(this.active.stack, arguments);
47576         this.allItems.push.apply(this.allItems,arguments);
47577         var r = [];
47578         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
47579             if(a[i].isFormField){
47580                 r.push(a[i]);
47581             }
47582         }
47583         if(r.length > 0){
47584             Roo.form.Form.superclass.add.apply(this, r);
47585         }
47586         return this;
47587     },
47588     
47589
47590     
47591     
47592     
47593      /**
47594      * Find any element that has been added to a form, using it's ID or name
47595      * This can include framesets, columns etc. along with regular fields..
47596      * @param {String} id - id or name to find.
47597      
47598      * @return {Element} e - or false if nothing found.
47599      */
47600     findbyId : function(id)
47601     {
47602         var ret = false;
47603         if (!id) {
47604             return ret;
47605         }
47606         Roo.each(this.allItems, function(f){
47607             if (f.id == id || f.name == id ){
47608                 ret = f;
47609                 return false;
47610             }
47611         });
47612         return ret;
47613     },
47614
47615     
47616     
47617     /**
47618      * Render this form into the passed container. This should only be called once!
47619      * @param {String/HTMLElement/Element} container The element this component should be rendered into
47620      * @return {Form} this
47621      */
47622     render : function(ct)
47623     {
47624         
47625         
47626         
47627         ct = Roo.get(ct);
47628         var o = this.autoCreate || {
47629             tag: 'form',
47630             method : this.method || 'POST',
47631             id : this.id || Roo.id()
47632         };
47633         this.initEl(ct.createChild(o));
47634
47635         this.root.render(this.el);
47636         
47637        
47638              
47639         this.items.each(function(f){
47640             f.render('x-form-el-'+f.id);
47641         });
47642
47643         if(this.buttons.length > 0){
47644             // tables are required to maintain order and for correct IE layout
47645             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
47646                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
47647                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
47648             }}, null, true);
47649             var tr = tb.getElementsByTagName('tr')[0];
47650             for(var i = 0, len = this.buttons.length; i < len; i++) {
47651                 var b = this.buttons[i];
47652                 var td = document.createElement('td');
47653                 td.className = 'x-form-btn-td';
47654                 b.render(tr.appendChild(td));
47655             }
47656         }
47657         if(this.monitorValid){ // initialize after render
47658             this.startMonitoring();
47659         }
47660         this.fireEvent('rendered', this);
47661         return this;
47662     },
47663
47664     /**
47665      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
47666      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
47667      * object or a valid Roo.DomHelper element config
47668      * @param {Function} handler The function called when the button is clicked
47669      * @param {Object} scope (optional) The scope of the handler function
47670      * @return {Roo.Button}
47671      */
47672     addButton : function(config, handler, scope){
47673         var bc = {
47674             handler: handler,
47675             scope: scope,
47676             minWidth: this.minButtonWidth,
47677             hideParent:true
47678         };
47679         if(typeof config == "string"){
47680             bc.text = config;
47681         }else{
47682             Roo.apply(bc, config);
47683         }
47684         var btn = new Roo.Button(null, bc);
47685         this.buttons.push(btn);
47686         return btn;
47687     },
47688
47689      /**
47690      * Adds a series of form elements (using the xtype property as the factory method.
47691      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
47692      * @param {Object} config 
47693      */
47694     
47695     addxtype : function()
47696     {
47697         var ar = Array.prototype.slice.call(arguments, 0);
47698         var ret = false;
47699         for(var i = 0; i < ar.length; i++) {
47700             if (!ar[i]) {
47701                 continue; // skip -- if this happends something invalid got sent, we 
47702                 // should ignore it, as basically that interface element will not show up
47703                 // and that should be pretty obvious!!
47704             }
47705             
47706             if (Roo.form[ar[i].xtype]) {
47707                 ar[i].form = this;
47708                 var fe = Roo.factory(ar[i], Roo.form);
47709                 if (!ret) {
47710                     ret = fe;
47711                 }
47712                 fe.form = this;
47713                 if (fe.store) {
47714                     fe.store.form = this;
47715                 }
47716                 if (fe.isLayout) {  
47717                          
47718                     this.start(fe);
47719                     this.allItems.push(fe);
47720                     if (fe.items && fe.addxtype) {
47721                         fe.addxtype.apply(fe, fe.items);
47722                         delete fe.items;
47723                     }
47724                      this.end();
47725                     continue;
47726                 }
47727                 
47728                 
47729                  
47730                 this.add(fe);
47731               //  console.log('adding ' + ar[i].xtype);
47732             }
47733             if (ar[i].xtype == 'Button') {  
47734                 //console.log('adding button');
47735                 //console.log(ar[i]);
47736                 this.addButton(ar[i]);
47737                 this.allItems.push(fe);
47738                 continue;
47739             }
47740             
47741             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
47742                 alert('end is not supported on xtype any more, use items');
47743             //    this.end();
47744             //    //console.log('adding end');
47745             }
47746             
47747         }
47748         return ret;
47749     },
47750     
47751     /**
47752      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
47753      * option "monitorValid"
47754      */
47755     startMonitoring : function(){
47756         if(!this.bound){
47757             this.bound = true;
47758             Roo.TaskMgr.start({
47759                 run : this.bindHandler,
47760                 interval : this.monitorPoll || 200,
47761                 scope: this
47762             });
47763         }
47764     },
47765
47766     /**
47767      * Stops monitoring of the valid state of this form
47768      */
47769     stopMonitoring : function(){
47770         this.bound = false;
47771     },
47772
47773     // private
47774     bindHandler : function(){
47775         if(!this.bound){
47776             return false; // stops binding
47777         }
47778         var valid = true;
47779         this.items.each(function(f){
47780             if(!f.isValid(true)){
47781                 valid = false;
47782                 return false;
47783             }
47784         });
47785         for(var i = 0, len = this.buttons.length; i < len; i++){
47786             var btn = this.buttons[i];
47787             if(btn.formBind === true && btn.disabled === valid){
47788                 btn.setDisabled(!valid);
47789             }
47790         }
47791         this.fireEvent('clientvalidation', this, valid);
47792     }
47793     
47794     
47795     
47796     
47797     
47798     
47799     
47800     
47801 });
47802
47803
47804 // back compat
47805 Roo.Form = Roo.form.Form;
47806 /*
47807  * Based on:
47808  * Ext JS Library 1.1.1
47809  * Copyright(c) 2006-2007, Ext JS, LLC.
47810  *
47811  * Originally Released Under LGPL - original licence link has changed is not relivant.
47812  *
47813  * Fork - LGPL
47814  * <script type="text/javascript">
47815  */
47816
47817 // as we use this in bootstrap.
47818 Roo.namespace('Roo.form');
47819  /**
47820  * @class Roo.form.Action
47821  * Internal Class used to handle form actions
47822  * @constructor
47823  * @param {Roo.form.BasicForm} el The form element or its id
47824  * @param {Object} config Configuration options
47825  */
47826
47827  
47828  
47829 // define the action interface
47830 Roo.form.Action = function(form, options){
47831     this.form = form;
47832     this.options = options || {};
47833 };
47834 /**
47835  * Client Validation Failed
47836  * @const 
47837  */
47838 Roo.form.Action.CLIENT_INVALID = 'client';
47839 /**
47840  * Server Validation Failed
47841  * @const 
47842  */
47843 Roo.form.Action.SERVER_INVALID = 'server';
47844  /**
47845  * Connect to Server Failed
47846  * @const 
47847  */
47848 Roo.form.Action.CONNECT_FAILURE = 'connect';
47849 /**
47850  * Reading Data from Server Failed
47851  * @const 
47852  */
47853 Roo.form.Action.LOAD_FAILURE = 'load';
47854
47855 Roo.form.Action.prototype = {
47856     type : 'default',
47857     failureType : undefined,
47858     response : undefined,
47859     result : undefined,
47860
47861     // interface method
47862     run : function(options){
47863
47864     },
47865
47866     // interface method
47867     success : function(response){
47868
47869     },
47870
47871     // interface method
47872     handleResponse : function(response){
47873
47874     },
47875
47876     // default connection failure
47877     failure : function(response){
47878         
47879         this.response = response;
47880         this.failureType = Roo.form.Action.CONNECT_FAILURE;
47881         this.form.afterAction(this, false);
47882     },
47883
47884     processResponse : function(response){
47885         this.response = response;
47886         if(!response.responseText){
47887             return true;
47888         }
47889         this.result = this.handleResponse(response);
47890         return this.result;
47891     },
47892
47893     // utility functions used internally
47894     getUrl : function(appendParams){
47895         var url = this.options.url || this.form.url || this.form.el.dom.action;
47896         if(appendParams){
47897             var p = this.getParams();
47898             if(p){
47899                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
47900             }
47901         }
47902         return url;
47903     },
47904
47905     getMethod : function(){
47906         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
47907     },
47908
47909     getParams : function(){
47910         var bp = this.form.baseParams;
47911         var p = this.options.params;
47912         if(p){
47913             if(typeof p == "object"){
47914                 p = Roo.urlEncode(Roo.applyIf(p, bp));
47915             }else if(typeof p == 'string' && bp){
47916                 p += '&' + Roo.urlEncode(bp);
47917             }
47918         }else if(bp){
47919             p = Roo.urlEncode(bp);
47920         }
47921         return p;
47922     },
47923
47924     createCallback : function(){
47925         return {
47926             success: this.success,
47927             failure: this.failure,
47928             scope: this,
47929             timeout: (this.form.timeout*1000),
47930             upload: this.form.fileUpload ? this.success : undefined
47931         };
47932     }
47933 };
47934
47935 Roo.form.Action.Submit = function(form, options){
47936     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
47937 };
47938
47939 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
47940     type : 'submit',
47941
47942     haveProgress : false,
47943     uploadComplete : false,
47944     
47945     // uploadProgress indicator.
47946     uploadProgress : function()
47947     {
47948         if (!this.form.progressUrl) {
47949             return;
47950         }
47951         
47952         if (!this.haveProgress) {
47953             Roo.MessageBox.progress("Uploading", "Uploading");
47954         }
47955         if (this.uploadComplete) {
47956            Roo.MessageBox.hide();
47957            return;
47958         }
47959         
47960         this.haveProgress = true;
47961    
47962         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
47963         
47964         var c = new Roo.data.Connection();
47965         c.request({
47966             url : this.form.progressUrl,
47967             params: {
47968                 id : uid
47969             },
47970             method: 'GET',
47971             success : function(req){
47972                //console.log(data);
47973                 var rdata = false;
47974                 var edata;
47975                 try  {
47976                    rdata = Roo.decode(req.responseText)
47977                 } catch (e) {
47978                     Roo.log("Invalid data from server..");
47979                     Roo.log(edata);
47980                     return;
47981                 }
47982                 if (!rdata || !rdata.success) {
47983                     Roo.log(rdata);
47984                     Roo.MessageBox.alert(Roo.encode(rdata));
47985                     return;
47986                 }
47987                 var data = rdata.data;
47988                 
47989                 if (this.uploadComplete) {
47990                    Roo.MessageBox.hide();
47991                    return;
47992                 }
47993                    
47994                 if (data){
47995                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
47996                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
47997                     );
47998                 }
47999                 this.uploadProgress.defer(2000,this);
48000             },
48001        
48002             failure: function(data) {
48003                 Roo.log('progress url failed ');
48004                 Roo.log(data);
48005             },
48006             scope : this
48007         });
48008            
48009     },
48010     
48011     
48012     run : function()
48013     {
48014         // run get Values on the form, so it syncs any secondary forms.
48015         this.form.getValues();
48016         
48017         var o = this.options;
48018         var method = this.getMethod();
48019         var isPost = method == 'POST';
48020         if(o.clientValidation === false || this.form.isValid()){
48021             
48022             if (this.form.progressUrl) {
48023                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48024                     (new Date() * 1) + '' + Math.random());
48025                     
48026             } 
48027             
48028             
48029             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48030                 form:this.form.el.dom,
48031                 url:this.getUrl(!isPost),
48032                 method: method,
48033                 params:isPost ? this.getParams() : null,
48034                 isUpload: this.form.fileUpload,
48035                 formData : this.form.formData
48036             }));
48037             
48038             this.uploadProgress();
48039
48040         }else if (o.clientValidation !== false){ // client validation failed
48041             this.failureType = Roo.form.Action.CLIENT_INVALID;
48042             this.form.afterAction(this, false);
48043         }
48044     },
48045
48046     success : function(response)
48047     {
48048         this.uploadComplete= true;
48049         if (this.haveProgress) {
48050             Roo.MessageBox.hide();
48051         }
48052         
48053         
48054         var result = this.processResponse(response);
48055         if(result === true || result.success){
48056             this.form.afterAction(this, true);
48057             return;
48058         }
48059         if(result.errors){
48060             this.form.markInvalid(result.errors);
48061             this.failureType = Roo.form.Action.SERVER_INVALID;
48062         }
48063         this.form.afterAction(this, false);
48064     },
48065     failure : function(response)
48066     {
48067         this.uploadComplete= true;
48068         if (this.haveProgress) {
48069             Roo.MessageBox.hide();
48070         }
48071         
48072         this.response = response;
48073         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48074         this.form.afterAction(this, false);
48075     },
48076     
48077     handleResponse : function(response){
48078         if(this.form.errorReader){
48079             var rs = this.form.errorReader.read(response);
48080             var errors = [];
48081             if(rs.records){
48082                 for(var i = 0, len = rs.records.length; i < len; i++) {
48083                     var r = rs.records[i];
48084                     errors[i] = r.data;
48085                 }
48086             }
48087             if(errors.length < 1){
48088                 errors = null;
48089             }
48090             return {
48091                 success : rs.success,
48092                 errors : errors
48093             };
48094         }
48095         var ret = false;
48096         try {
48097             ret = Roo.decode(response.responseText);
48098         } catch (e) {
48099             ret = {
48100                 success: false,
48101                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48102                 errors : []
48103             };
48104         }
48105         return ret;
48106         
48107     }
48108 });
48109
48110
48111 Roo.form.Action.Load = function(form, options){
48112     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48113     this.reader = this.form.reader;
48114 };
48115
48116 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48117     type : 'load',
48118
48119     run : function(){
48120         
48121         Roo.Ajax.request(Roo.apply(
48122                 this.createCallback(), {
48123                     method:this.getMethod(),
48124                     url:this.getUrl(false),
48125                     params:this.getParams()
48126         }));
48127     },
48128
48129     success : function(response){
48130         
48131         var result = this.processResponse(response);
48132         if(result === true || !result.success || !result.data){
48133             this.failureType = Roo.form.Action.LOAD_FAILURE;
48134             this.form.afterAction(this, false);
48135             return;
48136         }
48137         this.form.clearInvalid();
48138         this.form.setValues(result.data);
48139         this.form.afterAction(this, true);
48140     },
48141
48142     handleResponse : function(response){
48143         if(this.form.reader){
48144             var rs = this.form.reader.read(response);
48145             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48146             return {
48147                 success : rs.success,
48148                 data : data
48149             };
48150         }
48151         return Roo.decode(response.responseText);
48152     }
48153 });
48154
48155 Roo.form.Action.ACTION_TYPES = {
48156     'load' : Roo.form.Action.Load,
48157     'submit' : Roo.form.Action.Submit
48158 };/*
48159  * Based on:
48160  * Ext JS Library 1.1.1
48161  * Copyright(c) 2006-2007, Ext JS, LLC.
48162  *
48163  * Originally Released Under LGPL - original licence link has changed is not relivant.
48164  *
48165  * Fork - LGPL
48166  * <script type="text/javascript">
48167  */
48168  
48169 /**
48170  * @class Roo.form.Layout
48171  * @extends Roo.Component
48172  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48173  * @constructor
48174  * @param {Object} config Configuration options
48175  */
48176 Roo.form.Layout = function(config){
48177     var xitems = [];
48178     if (config.items) {
48179         xitems = config.items;
48180         delete config.items;
48181     }
48182     Roo.form.Layout.superclass.constructor.call(this, config);
48183     this.stack = [];
48184     Roo.each(xitems, this.addxtype, this);
48185      
48186 };
48187
48188 Roo.extend(Roo.form.Layout, Roo.Component, {
48189     /**
48190      * @cfg {String/Object} autoCreate
48191      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48192      */
48193     /**
48194      * @cfg {String/Object/Function} style
48195      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48196      * a function which returns such a specification.
48197      */
48198     /**
48199      * @cfg {String} labelAlign
48200      * Valid values are "left," "top" and "right" (defaults to "left")
48201      */
48202     /**
48203      * @cfg {Number} labelWidth
48204      * Fixed width in pixels of all field labels (defaults to undefined)
48205      */
48206     /**
48207      * @cfg {Boolean} clear
48208      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48209      */
48210     clear : true,
48211     /**
48212      * @cfg {String} labelSeparator
48213      * The separator to use after field labels (defaults to ':')
48214      */
48215     labelSeparator : ':',
48216     /**
48217      * @cfg {Boolean} hideLabels
48218      * True to suppress the display of field labels in this layout (defaults to false)
48219      */
48220     hideLabels : false,
48221
48222     // private
48223     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48224     
48225     isLayout : true,
48226     
48227     // private
48228     onRender : function(ct, position){
48229         if(this.el){ // from markup
48230             this.el = Roo.get(this.el);
48231         }else {  // generate
48232             var cfg = this.getAutoCreate();
48233             this.el = ct.createChild(cfg, position);
48234         }
48235         if(this.style){
48236             this.el.applyStyles(this.style);
48237         }
48238         if(this.labelAlign){
48239             this.el.addClass('x-form-label-'+this.labelAlign);
48240         }
48241         if(this.hideLabels){
48242             this.labelStyle = "display:none";
48243             this.elementStyle = "padding-left:0;";
48244         }else{
48245             if(typeof this.labelWidth == 'number'){
48246                 this.labelStyle = "width:"+this.labelWidth+"px;";
48247                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48248             }
48249             if(this.labelAlign == 'top'){
48250                 this.labelStyle = "width:auto;";
48251                 this.elementStyle = "padding-left:0;";
48252             }
48253         }
48254         var stack = this.stack;
48255         var slen = stack.length;
48256         if(slen > 0){
48257             if(!this.fieldTpl){
48258                 var t = new Roo.Template(
48259                     '<div class="x-form-item {5}">',
48260                         '<label for="{0}" style="{2}">{1}{4}</label>',
48261                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48262                         '</div>',
48263                     '</div><div class="x-form-clear-left"></div>'
48264                 );
48265                 t.disableFormats = true;
48266                 t.compile();
48267                 Roo.form.Layout.prototype.fieldTpl = t;
48268             }
48269             for(var i = 0; i < slen; i++) {
48270                 if(stack[i].isFormField){
48271                     this.renderField(stack[i]);
48272                 }else{
48273                     this.renderComponent(stack[i]);
48274                 }
48275             }
48276         }
48277         if(this.clear){
48278             this.el.createChild({cls:'x-form-clear'});
48279         }
48280     },
48281
48282     // private
48283     renderField : function(f){
48284         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
48285                f.id, //0
48286                f.fieldLabel, //1
48287                f.labelStyle||this.labelStyle||'', //2
48288                this.elementStyle||'', //3
48289                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
48290                f.itemCls||this.itemCls||''  //5
48291        ], true).getPrevSibling());
48292     },
48293
48294     // private
48295     renderComponent : function(c){
48296         c.render(c.isLayout ? this.el : this.el.createChild());    
48297     },
48298     /**
48299      * Adds a object form elements (using the xtype property as the factory method.)
48300      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
48301      * @param {Object} config 
48302      */
48303     addxtype : function(o)
48304     {
48305         // create the lement.
48306         o.form = this.form;
48307         var fe = Roo.factory(o, Roo.form);
48308         this.form.allItems.push(fe);
48309         this.stack.push(fe);
48310         
48311         if (fe.isFormField) {
48312             this.form.items.add(fe);
48313         }
48314          
48315         return fe;
48316     }
48317 });
48318
48319 /**
48320  * @class Roo.form.Column
48321  * @extends Roo.form.Layout
48322  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
48323  * @constructor
48324  * @param {Object} config Configuration options
48325  */
48326 Roo.form.Column = function(config){
48327     Roo.form.Column.superclass.constructor.call(this, config);
48328 };
48329
48330 Roo.extend(Roo.form.Column, Roo.form.Layout, {
48331     /**
48332      * @cfg {Number/String} width
48333      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48334      */
48335     /**
48336      * @cfg {String/Object} autoCreate
48337      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
48338      */
48339
48340     // private
48341     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
48342
48343     // private
48344     onRender : function(ct, position){
48345         Roo.form.Column.superclass.onRender.call(this, ct, position);
48346         if(this.width){
48347             this.el.setWidth(this.width);
48348         }
48349     }
48350 });
48351
48352
48353 /**
48354  * @class Roo.form.Row
48355  * @extends Roo.form.Layout
48356  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
48357  * @constructor
48358  * @param {Object} config Configuration options
48359  */
48360
48361  
48362 Roo.form.Row = function(config){
48363     Roo.form.Row.superclass.constructor.call(this, config);
48364 };
48365  
48366 Roo.extend(Roo.form.Row, Roo.form.Layout, {
48367       /**
48368      * @cfg {Number/String} width
48369      * The fixed width of the column in pixels or CSS value (defaults to "auto")
48370      */
48371     /**
48372      * @cfg {Number/String} height
48373      * The fixed height of the column in pixels or CSS value (defaults to "auto")
48374      */
48375     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
48376     
48377     padWidth : 20,
48378     // private
48379     onRender : function(ct, position){
48380         //console.log('row render');
48381         if(!this.rowTpl){
48382             var t = new Roo.Template(
48383                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
48384                     '<label for="{0}" style="{2}">{1}{4}</label>',
48385                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48386                     '</div>',
48387                 '</div>'
48388             );
48389             t.disableFormats = true;
48390             t.compile();
48391             Roo.form.Layout.prototype.rowTpl = t;
48392         }
48393         this.fieldTpl = this.rowTpl;
48394         
48395         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
48396         var labelWidth = 100;
48397         
48398         if ((this.labelAlign != 'top')) {
48399             if (typeof this.labelWidth == 'number') {
48400                 labelWidth = this.labelWidth
48401             }
48402             this.padWidth =  20 + labelWidth;
48403             
48404         }
48405         
48406         Roo.form.Column.superclass.onRender.call(this, ct, position);
48407         if(this.width){
48408             this.el.setWidth(this.width);
48409         }
48410         if(this.height){
48411             this.el.setHeight(this.height);
48412         }
48413     },
48414     
48415     // private
48416     renderField : function(f){
48417         f.fieldEl = this.fieldTpl.append(this.el, [
48418                f.id, f.fieldLabel,
48419                f.labelStyle||this.labelStyle||'',
48420                this.elementStyle||'',
48421                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
48422                f.itemCls||this.itemCls||'',
48423                f.width ? f.width + this.padWidth : 160 + this.padWidth
48424        ],true);
48425     }
48426 });
48427  
48428
48429 /**
48430  * @class Roo.form.FieldSet
48431  * @extends Roo.form.Layout
48432  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
48433  * @constructor
48434  * @param {Object} config Configuration options
48435  */
48436 Roo.form.FieldSet = function(config){
48437     Roo.form.FieldSet.superclass.constructor.call(this, config);
48438 };
48439
48440 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
48441     /**
48442      * @cfg {String} legend
48443      * The text to display as the legend for the FieldSet (defaults to '')
48444      */
48445     /**
48446      * @cfg {String/Object} autoCreate
48447      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
48448      */
48449
48450     // private
48451     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
48452
48453     // private
48454     onRender : function(ct, position){
48455         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
48456         if(this.legend){
48457             this.setLegend(this.legend);
48458         }
48459     },
48460
48461     // private
48462     setLegend : function(text){
48463         if(this.rendered){
48464             this.el.child('legend').update(text);
48465         }
48466     }
48467 });/*
48468  * Based on:
48469  * Ext JS Library 1.1.1
48470  * Copyright(c) 2006-2007, Ext JS, LLC.
48471  *
48472  * Originally Released Under LGPL - original licence link has changed is not relivant.
48473  *
48474  * Fork - LGPL
48475  * <script type="text/javascript">
48476  */
48477 /**
48478  * @class Roo.form.VTypes
48479  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
48480  * @singleton
48481  */
48482 Roo.form.VTypes = function(){
48483     // closure these in so they are only created once.
48484     var alpha = /^[a-zA-Z_]+$/;
48485     var alphanum = /^[a-zA-Z0-9_]+$/;
48486     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
48487     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
48488
48489     // All these messages and functions are configurable
48490     return {
48491         /**
48492          * The function used to validate email addresses
48493          * @param {String} value The email address
48494          */
48495         'email' : function(v){
48496             return email.test(v);
48497         },
48498         /**
48499          * The error text to display when the email validation function returns false
48500          * @type String
48501          */
48502         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
48503         /**
48504          * The keystroke filter mask to be applied on email input
48505          * @type RegExp
48506          */
48507         'emailMask' : /[a-z0-9_\.\-@]/i,
48508
48509         /**
48510          * The function used to validate URLs
48511          * @param {String} value The URL
48512          */
48513         'url' : function(v){
48514             return url.test(v);
48515         },
48516         /**
48517          * The error text to display when the url validation function returns false
48518          * @type String
48519          */
48520         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
48521         
48522         /**
48523          * The function used to validate alpha values
48524          * @param {String} value The value
48525          */
48526         'alpha' : function(v){
48527             return alpha.test(v);
48528         },
48529         /**
48530          * The error text to display when the alpha validation function returns false
48531          * @type String
48532          */
48533         'alphaText' : 'This field should only contain letters and _',
48534         /**
48535          * The keystroke filter mask to be applied on alpha input
48536          * @type RegExp
48537          */
48538         'alphaMask' : /[a-z_]/i,
48539
48540         /**
48541          * The function used to validate alphanumeric values
48542          * @param {String} value The value
48543          */
48544         'alphanum' : function(v){
48545             return alphanum.test(v);
48546         },
48547         /**
48548          * The error text to display when the alphanumeric validation function returns false
48549          * @type String
48550          */
48551         'alphanumText' : 'This field should only contain letters, numbers and _',
48552         /**
48553          * The keystroke filter mask to be applied on alphanumeric input
48554          * @type RegExp
48555          */
48556         'alphanumMask' : /[a-z0-9_]/i
48557     };
48558 }();//<script type="text/javascript">
48559
48560 /**
48561  * @class Roo.form.FCKeditor
48562  * @extends Roo.form.TextArea
48563  * Wrapper around the FCKEditor http://www.fckeditor.net
48564  * @constructor
48565  * Creates a new FCKeditor
48566  * @param {Object} config Configuration options
48567  */
48568 Roo.form.FCKeditor = function(config){
48569     Roo.form.FCKeditor.superclass.constructor.call(this, config);
48570     this.addEvents({
48571          /**
48572          * @event editorinit
48573          * Fired when the editor is initialized - you can add extra handlers here..
48574          * @param {FCKeditor} this
48575          * @param {Object} the FCK object.
48576          */
48577         editorinit : true
48578     });
48579     
48580     
48581 };
48582 Roo.form.FCKeditor.editors = { };
48583 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
48584 {
48585     //defaultAutoCreate : {
48586     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
48587     //},
48588     // private
48589     /**
48590      * @cfg {Object} fck options - see fck manual for details.
48591      */
48592     fckconfig : false,
48593     
48594     /**
48595      * @cfg {Object} fck toolbar set (Basic or Default)
48596      */
48597     toolbarSet : 'Basic',
48598     /**
48599      * @cfg {Object} fck BasePath
48600      */ 
48601     basePath : '/fckeditor/',
48602     
48603     
48604     frame : false,
48605     
48606     value : '',
48607     
48608    
48609     onRender : function(ct, position)
48610     {
48611         if(!this.el){
48612             this.defaultAutoCreate = {
48613                 tag: "textarea",
48614                 style:"width:300px;height:60px;",
48615                 autocomplete: "new-password"
48616             };
48617         }
48618         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
48619         /*
48620         if(this.grow){
48621             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
48622             if(this.preventScrollbars){
48623                 this.el.setStyle("overflow", "hidden");
48624             }
48625             this.el.setHeight(this.growMin);
48626         }
48627         */
48628         //console.log('onrender' + this.getId() );
48629         Roo.form.FCKeditor.editors[this.getId()] = this;
48630          
48631
48632         this.replaceTextarea() ;
48633         
48634     },
48635     
48636     getEditor : function() {
48637         return this.fckEditor;
48638     },
48639     /**
48640      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
48641      * @param {Mixed} value The value to set
48642      */
48643     
48644     
48645     setValue : function(value)
48646     {
48647         //console.log('setValue: ' + value);
48648         
48649         if(typeof(value) == 'undefined') { // not sure why this is happending...
48650             return;
48651         }
48652         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
48653         
48654         //if(!this.el || !this.getEditor()) {
48655         //    this.value = value;
48656             //this.setValue.defer(100,this,[value]);    
48657         //    return;
48658         //} 
48659         
48660         if(!this.getEditor()) {
48661             return;
48662         }
48663         
48664         this.getEditor().SetData(value);
48665         
48666         //
48667
48668     },
48669
48670     /**
48671      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
48672      * @return {Mixed} value The field value
48673      */
48674     getValue : function()
48675     {
48676         
48677         if (this.frame && this.frame.dom.style.display == 'none') {
48678             return Roo.form.FCKeditor.superclass.getValue.call(this);
48679         }
48680         
48681         if(!this.el || !this.getEditor()) {
48682            
48683            // this.getValue.defer(100,this); 
48684             return this.value;
48685         }
48686        
48687         
48688         var value=this.getEditor().GetData();
48689         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
48690         return Roo.form.FCKeditor.superclass.getValue.call(this);
48691         
48692
48693     },
48694
48695     /**
48696      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
48697      * @return {Mixed} value The field value
48698      */
48699     getRawValue : function()
48700     {
48701         if (this.frame && this.frame.dom.style.display == 'none') {
48702             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
48703         }
48704         
48705         if(!this.el || !this.getEditor()) {
48706             //this.getRawValue.defer(100,this); 
48707             return this.value;
48708             return;
48709         }
48710         
48711         
48712         
48713         var value=this.getEditor().GetData();
48714         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
48715         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
48716          
48717     },
48718     
48719     setSize : function(w,h) {
48720         
48721         
48722         
48723         //if (this.frame && this.frame.dom.style.display == 'none') {
48724         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
48725         //    return;
48726         //}
48727         //if(!this.el || !this.getEditor()) {
48728         //    this.setSize.defer(100,this, [w,h]); 
48729         //    return;
48730         //}
48731         
48732         
48733         
48734         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
48735         
48736         this.frame.dom.setAttribute('width', w);
48737         this.frame.dom.setAttribute('height', h);
48738         this.frame.setSize(w,h);
48739         
48740     },
48741     
48742     toggleSourceEdit : function(value) {
48743         
48744       
48745          
48746         this.el.dom.style.display = value ? '' : 'none';
48747         this.frame.dom.style.display = value ?  'none' : '';
48748         
48749     },
48750     
48751     
48752     focus: function(tag)
48753     {
48754         if (this.frame.dom.style.display == 'none') {
48755             return Roo.form.FCKeditor.superclass.focus.call(this);
48756         }
48757         if(!this.el || !this.getEditor()) {
48758             this.focus.defer(100,this, [tag]); 
48759             return;
48760         }
48761         
48762         
48763         
48764         
48765         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
48766         this.getEditor().Focus();
48767         if (tgs.length) {
48768             if (!this.getEditor().Selection.GetSelection()) {
48769                 this.focus.defer(100,this, [tag]); 
48770                 return;
48771             }
48772             
48773             
48774             var r = this.getEditor().EditorDocument.createRange();
48775             r.setStart(tgs[0],0);
48776             r.setEnd(tgs[0],0);
48777             this.getEditor().Selection.GetSelection().removeAllRanges();
48778             this.getEditor().Selection.GetSelection().addRange(r);
48779             this.getEditor().Focus();
48780         }
48781         
48782     },
48783     
48784     
48785     
48786     replaceTextarea : function()
48787     {
48788         if ( document.getElementById( this.getId() + '___Frame' ) ) {
48789             return ;
48790         }
48791         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
48792         //{
48793             // We must check the elements firstly using the Id and then the name.
48794         var oTextarea = document.getElementById( this.getId() );
48795         
48796         var colElementsByName = document.getElementsByName( this.getId() ) ;
48797          
48798         oTextarea.style.display = 'none' ;
48799
48800         if ( oTextarea.tabIndex ) {            
48801             this.TabIndex = oTextarea.tabIndex ;
48802         }
48803         
48804         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
48805         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
48806         this.frame = Roo.get(this.getId() + '___Frame')
48807     },
48808     
48809     _getConfigHtml : function()
48810     {
48811         var sConfig = '' ;
48812
48813         for ( var o in this.fckconfig ) {
48814             sConfig += sConfig.length > 0  ? '&amp;' : '';
48815             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
48816         }
48817
48818         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
48819     },
48820     
48821     
48822     _getIFrameHtml : function()
48823     {
48824         var sFile = 'fckeditor.html' ;
48825         /* no idea what this is about..
48826         try
48827         {
48828             if ( (/fcksource=true/i).test( window.top.location.search ) )
48829                 sFile = 'fckeditor.original.html' ;
48830         }
48831         catch (e) { 
48832         */
48833
48834         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
48835         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
48836         
48837         
48838         var html = '<iframe id="' + this.getId() +
48839             '___Frame" src="' + sLink +
48840             '" width="' + this.width +
48841             '" height="' + this.height + '"' +
48842             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
48843             ' frameborder="0" scrolling="no"></iframe>' ;
48844
48845         return html ;
48846     },
48847     
48848     _insertHtmlBefore : function( html, element )
48849     {
48850         if ( element.insertAdjacentHTML )       {
48851             // IE
48852             element.insertAdjacentHTML( 'beforeBegin', html ) ;
48853         } else { // Gecko
48854             var oRange = document.createRange() ;
48855             oRange.setStartBefore( element ) ;
48856             var oFragment = oRange.createContextualFragment( html );
48857             element.parentNode.insertBefore( oFragment, element ) ;
48858         }
48859     }
48860     
48861     
48862   
48863     
48864     
48865     
48866     
48867
48868 });
48869
48870 //Roo.reg('fckeditor', Roo.form.FCKeditor);
48871
48872 function FCKeditor_OnComplete(editorInstance){
48873     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
48874     f.fckEditor = editorInstance;
48875     //console.log("loaded");
48876     f.fireEvent('editorinit', f, editorInstance);
48877
48878   
48879
48880  
48881
48882
48883
48884
48885
48886
48887
48888
48889
48890
48891
48892
48893
48894
48895
48896 //<script type="text/javascript">
48897 /**
48898  * @class Roo.form.GridField
48899  * @extends Roo.form.Field
48900  * Embed a grid (or editable grid into a form)
48901  * STATUS ALPHA
48902  * 
48903  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
48904  * it needs 
48905  * xgrid.store = Roo.data.Store
48906  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
48907  * xgrid.store.reader = Roo.data.JsonReader 
48908  * 
48909  * 
48910  * @constructor
48911  * Creates a new GridField
48912  * @param {Object} config Configuration options
48913  */
48914 Roo.form.GridField = function(config){
48915     Roo.form.GridField.superclass.constructor.call(this, config);
48916      
48917 };
48918
48919 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
48920     /**
48921      * @cfg {Number} width  - used to restrict width of grid..
48922      */
48923     width : 100,
48924     /**
48925      * @cfg {Number} height - used to restrict height of grid..
48926      */
48927     height : 50,
48928      /**
48929      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
48930          * 
48931          *}
48932      */
48933     xgrid : false, 
48934     /**
48935      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
48936      * {tag: "input", type: "checkbox", autocomplete: "off"})
48937      */
48938    // defaultAutoCreate : { tag: 'div' },
48939     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
48940     /**
48941      * @cfg {String} addTitle Text to include for adding a title.
48942      */
48943     addTitle : false,
48944     //
48945     onResize : function(){
48946         Roo.form.Field.superclass.onResize.apply(this, arguments);
48947     },
48948
48949     initEvents : function(){
48950         // Roo.form.Checkbox.superclass.initEvents.call(this);
48951         // has no events...
48952        
48953     },
48954
48955
48956     getResizeEl : function(){
48957         return this.wrap;
48958     },
48959
48960     getPositionEl : function(){
48961         return this.wrap;
48962     },
48963
48964     // private
48965     onRender : function(ct, position){
48966         
48967         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
48968         var style = this.style;
48969         delete this.style;
48970         
48971         Roo.form.GridField.superclass.onRender.call(this, ct, position);
48972         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
48973         this.viewEl = this.wrap.createChild({ tag: 'div' });
48974         if (style) {
48975             this.viewEl.applyStyles(style);
48976         }
48977         if (this.width) {
48978             this.viewEl.setWidth(this.width);
48979         }
48980         if (this.height) {
48981             this.viewEl.setHeight(this.height);
48982         }
48983         //if(this.inputValue !== undefined){
48984         //this.setValue(this.value);
48985         
48986         
48987         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
48988         
48989         
48990         this.grid.render();
48991         this.grid.getDataSource().on('remove', this.refreshValue, this);
48992         this.grid.getDataSource().on('update', this.refreshValue, this);
48993         this.grid.on('afteredit', this.refreshValue, this);
48994  
48995     },
48996      
48997     
48998     /**
48999      * Sets the value of the item. 
49000      * @param {String} either an object  or a string..
49001      */
49002     setValue : function(v){
49003         //this.value = v;
49004         v = v || []; // empty set..
49005         // this does not seem smart - it really only affects memoryproxy grids..
49006         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49007             var ds = this.grid.getDataSource();
49008             // assumes a json reader..
49009             var data = {}
49010             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49011             ds.loadData( data);
49012         }
49013         // clear selection so it does not get stale.
49014         if (this.grid.sm) { 
49015             this.grid.sm.clearSelections();
49016         }
49017         
49018         Roo.form.GridField.superclass.setValue.call(this, v);
49019         this.refreshValue();
49020         // should load data in the grid really....
49021     },
49022     
49023     // private
49024     refreshValue: function() {
49025          var val = [];
49026         this.grid.getDataSource().each(function(r) {
49027             val.push(r.data);
49028         });
49029         this.el.dom.value = Roo.encode(val);
49030     }
49031     
49032      
49033     
49034     
49035 });/*
49036  * Based on:
49037  * Ext JS Library 1.1.1
49038  * Copyright(c) 2006-2007, Ext JS, LLC.
49039  *
49040  * Originally Released Under LGPL - original licence link has changed is not relivant.
49041  *
49042  * Fork - LGPL
49043  * <script type="text/javascript">
49044  */
49045 /**
49046  * @class Roo.form.DisplayField
49047  * @extends Roo.form.Field
49048  * A generic Field to display non-editable data.
49049  * @cfg {Boolean} closable (true|false) default false
49050  * @constructor
49051  * Creates a new Display Field item.
49052  * @param {Object} config Configuration options
49053  */
49054 Roo.form.DisplayField = function(config){
49055     Roo.form.DisplayField.superclass.constructor.call(this, config);
49056     
49057     this.addEvents({
49058         /**
49059          * @event close
49060          * Fires after the click the close btn
49061              * @param {Roo.form.DisplayField} this
49062              */
49063         close : true
49064     });
49065 };
49066
49067 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49068     inputType:      'hidden',
49069     allowBlank:     true,
49070     readOnly:         true,
49071     
49072  
49073     /**
49074      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49075      */
49076     focusClass : undefined,
49077     /**
49078      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49079      */
49080     fieldClass: 'x-form-field',
49081     
49082      /**
49083      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49084      */
49085     valueRenderer: undefined,
49086     
49087     width: 100,
49088     /**
49089      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49090      * {tag: "input", type: "checkbox", autocomplete: "off"})
49091      */
49092      
49093  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49094  
49095     closable : false,
49096     
49097     onResize : function(){
49098         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49099         
49100     },
49101
49102     initEvents : function(){
49103         // Roo.form.Checkbox.superclass.initEvents.call(this);
49104         // has no events...
49105         
49106         if(this.closable){
49107             this.closeEl.on('click', this.onClose, this);
49108         }
49109        
49110     },
49111
49112
49113     getResizeEl : function(){
49114         return this.wrap;
49115     },
49116
49117     getPositionEl : function(){
49118         return this.wrap;
49119     },
49120
49121     // private
49122     onRender : function(ct, position){
49123         
49124         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49125         //if(this.inputValue !== undefined){
49126         this.wrap = this.el.wrap();
49127         
49128         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49129         
49130         if(this.closable){
49131             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49132         }
49133         
49134         if (this.bodyStyle) {
49135             this.viewEl.applyStyles(this.bodyStyle);
49136         }
49137         //this.viewEl.setStyle('padding', '2px');
49138         
49139         this.setValue(this.value);
49140         
49141     },
49142 /*
49143     // private
49144     initValue : Roo.emptyFn,
49145
49146   */
49147
49148         // private
49149     onClick : function(){
49150         
49151     },
49152
49153     /**
49154      * Sets the checked state of the checkbox.
49155      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49156      */
49157     setValue : function(v){
49158         this.value = v;
49159         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49160         // this might be called before we have a dom element..
49161         if (!this.viewEl) {
49162             return;
49163         }
49164         this.viewEl.dom.innerHTML = html;
49165         Roo.form.DisplayField.superclass.setValue.call(this, v);
49166
49167     },
49168     
49169     onClose : function(e)
49170     {
49171         e.preventDefault();
49172         
49173         this.fireEvent('close', this);
49174     }
49175 });/*
49176  * 
49177  * Licence- LGPL
49178  * 
49179  */
49180
49181 /**
49182  * @class Roo.form.DayPicker
49183  * @extends Roo.form.Field
49184  * A Day picker show [M] [T] [W] ....
49185  * @constructor
49186  * Creates a new Day Picker
49187  * @param {Object} config Configuration options
49188  */
49189 Roo.form.DayPicker= function(config){
49190     Roo.form.DayPicker.superclass.constructor.call(this, config);
49191      
49192 };
49193
49194 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49195     /**
49196      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49197      */
49198     focusClass : undefined,
49199     /**
49200      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49201      */
49202     fieldClass: "x-form-field",
49203    
49204     /**
49205      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49206      * {tag: "input", type: "checkbox", autocomplete: "off"})
49207      */
49208     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49209     
49210    
49211     actionMode : 'viewEl', 
49212     //
49213     // private
49214  
49215     inputType : 'hidden',
49216     
49217      
49218     inputElement: false, // real input element?
49219     basedOn: false, // ????
49220     
49221     isFormField: true, // not sure where this is needed!!!!
49222
49223     onResize : function(){
49224         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49225         if(!this.boxLabel){
49226             this.el.alignTo(this.wrap, 'c-c');
49227         }
49228     },
49229
49230     initEvents : function(){
49231         Roo.form.Checkbox.superclass.initEvents.call(this);
49232         this.el.on("click", this.onClick,  this);
49233         this.el.on("change", this.onClick,  this);
49234     },
49235
49236
49237     getResizeEl : function(){
49238         return this.wrap;
49239     },
49240
49241     getPositionEl : function(){
49242         return this.wrap;
49243     },
49244
49245     
49246     // private
49247     onRender : function(ct, position){
49248         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49249        
49250         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49251         
49252         var r1 = '<table><tr>';
49253         var r2 = '<tr class="x-form-daypick-icons">';
49254         for (var i=0; i < 7; i++) {
49255             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49256             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49257         }
49258         
49259         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49260         viewEl.select('img').on('click', this.onClick, this);
49261         this.viewEl = viewEl;   
49262         
49263         
49264         // this will not work on Chrome!!!
49265         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49266         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49267         
49268         
49269           
49270
49271     },
49272
49273     // private
49274     initValue : Roo.emptyFn,
49275
49276     /**
49277      * Returns the checked state of the checkbox.
49278      * @return {Boolean} True if checked, else false
49279      */
49280     getValue : function(){
49281         return this.el.dom.value;
49282         
49283     },
49284
49285         // private
49286     onClick : function(e){ 
49287         //this.setChecked(!this.checked);
49288         Roo.get(e.target).toggleClass('x-menu-item-checked');
49289         this.refreshValue();
49290         //if(this.el.dom.checked != this.checked){
49291         //    this.setValue(this.el.dom.checked);
49292        // }
49293     },
49294     
49295     // private
49296     refreshValue : function()
49297     {
49298         var val = '';
49299         this.viewEl.select('img',true).each(function(e,i,n)  {
49300             val += e.is(".x-menu-item-checked") ? String(n) : '';
49301         });
49302         this.setValue(val, true);
49303     },
49304
49305     /**
49306      * Sets the checked state of the checkbox.
49307      * On is always based on a string comparison between inputValue and the param.
49308      * @param {Boolean/String} value - the value to set 
49309      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
49310      */
49311     setValue : function(v,suppressEvent){
49312         if (!this.el.dom) {
49313             return;
49314         }
49315         var old = this.el.dom.value ;
49316         this.el.dom.value = v;
49317         if (suppressEvent) {
49318             return ;
49319         }
49320          
49321         // update display..
49322         this.viewEl.select('img',true).each(function(e,i,n)  {
49323             
49324             var on = e.is(".x-menu-item-checked");
49325             var newv = v.indexOf(String(n)) > -1;
49326             if (on != newv) {
49327                 e.toggleClass('x-menu-item-checked');
49328             }
49329             
49330         });
49331         
49332         
49333         this.fireEvent('change', this, v, old);
49334         
49335         
49336     },
49337    
49338     // handle setting of hidden value by some other method!!?!?
49339     setFromHidden: function()
49340     {
49341         if(!this.el){
49342             return;
49343         }
49344         //console.log("SET FROM HIDDEN");
49345         //alert('setFrom hidden');
49346         this.setValue(this.el.dom.value);
49347     },
49348     
49349     onDestroy : function()
49350     {
49351         if(this.viewEl){
49352             Roo.get(this.viewEl).remove();
49353         }
49354          
49355         Roo.form.DayPicker.superclass.onDestroy.call(this);
49356     }
49357
49358 });/*
49359  * RooJS Library 1.1.1
49360  * Copyright(c) 2008-2011  Alan Knowles
49361  *
49362  * License - LGPL
49363  */
49364  
49365
49366 /**
49367  * @class Roo.form.ComboCheck
49368  * @extends Roo.form.ComboBox
49369  * A combobox for multiple select items.
49370  *
49371  * FIXME - could do with a reset button..
49372  * 
49373  * @constructor
49374  * Create a new ComboCheck
49375  * @param {Object} config Configuration options
49376  */
49377 Roo.form.ComboCheck = function(config){
49378     Roo.form.ComboCheck.superclass.constructor.call(this, config);
49379     // should verify some data...
49380     // like
49381     // hiddenName = required..
49382     // displayField = required
49383     // valudField == required
49384     var req= [ 'hiddenName', 'displayField', 'valueField' ];
49385     var _t = this;
49386     Roo.each(req, function(e) {
49387         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
49388             throw "Roo.form.ComboCheck : missing value for: " + e;
49389         }
49390     });
49391     
49392     
49393 };
49394
49395 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
49396      
49397      
49398     editable : false,
49399      
49400     selectedClass: 'x-menu-item-checked', 
49401     
49402     // private
49403     onRender : function(ct, position){
49404         var _t = this;
49405         
49406         
49407         
49408         if(!this.tpl){
49409             var cls = 'x-combo-list';
49410
49411             
49412             this.tpl =  new Roo.Template({
49413                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
49414                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
49415                    '<span>{' + this.displayField + '}</span>' +
49416                     '</div>' 
49417                 
49418             });
49419         }
49420  
49421         
49422         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
49423         this.view.singleSelect = false;
49424         this.view.multiSelect = true;
49425         this.view.toggleSelect = true;
49426         this.pageTb.add(new Roo.Toolbar.Fill(), {
49427             
49428             text: 'Done',
49429             handler: function()
49430             {
49431                 _t.collapse();
49432             }
49433         });
49434     },
49435     
49436     onViewOver : function(e, t){
49437         // do nothing...
49438         return;
49439         
49440     },
49441     
49442     onViewClick : function(doFocus,index){
49443         return;
49444         
49445     },
49446     select: function () {
49447         //Roo.log("SELECT CALLED");
49448     },
49449      
49450     selectByValue : function(xv, scrollIntoView){
49451         var ar = this.getValueArray();
49452         var sels = [];
49453         
49454         Roo.each(ar, function(v) {
49455             if(v === undefined || v === null){
49456                 return;
49457             }
49458             var r = this.findRecord(this.valueField, v);
49459             if(r){
49460                 sels.push(this.store.indexOf(r))
49461                 
49462             }
49463         },this);
49464         this.view.select(sels);
49465         return false;
49466     },
49467     
49468     
49469     
49470     onSelect : function(record, index){
49471        // Roo.log("onselect Called");
49472        // this is only called by the clear button now..
49473         this.view.clearSelections();
49474         this.setValue('[]');
49475         if (this.value != this.valueBefore) {
49476             this.fireEvent('change', this, this.value, this.valueBefore);
49477             this.valueBefore = this.value;
49478         }
49479     },
49480     getValueArray : function()
49481     {
49482         var ar = [] ;
49483         
49484         try {
49485             //Roo.log(this.value);
49486             if (typeof(this.value) == 'undefined') {
49487                 return [];
49488             }
49489             var ar = Roo.decode(this.value);
49490             return  ar instanceof Array ? ar : []; //?? valid?
49491             
49492         } catch(e) {
49493             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
49494             return [];
49495         }
49496          
49497     },
49498     expand : function ()
49499     {
49500         
49501         Roo.form.ComboCheck.superclass.expand.call(this);
49502         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
49503         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
49504         
49505
49506     },
49507     
49508     collapse : function(){
49509         Roo.form.ComboCheck.superclass.collapse.call(this);
49510         var sl = this.view.getSelectedIndexes();
49511         var st = this.store;
49512         var nv = [];
49513         var tv = [];
49514         var r;
49515         Roo.each(sl, function(i) {
49516             r = st.getAt(i);
49517             nv.push(r.get(this.valueField));
49518         },this);
49519         this.setValue(Roo.encode(nv));
49520         if (this.value != this.valueBefore) {
49521
49522             this.fireEvent('change', this, this.value, this.valueBefore);
49523             this.valueBefore = this.value;
49524         }
49525         
49526     },
49527     
49528     setValue : function(v){
49529         // Roo.log(v);
49530         this.value = v;
49531         
49532         var vals = this.getValueArray();
49533         var tv = [];
49534         Roo.each(vals, function(k) {
49535             var r = this.findRecord(this.valueField, k);
49536             if(r){
49537                 tv.push(r.data[this.displayField]);
49538             }else if(this.valueNotFoundText !== undefined){
49539                 tv.push( this.valueNotFoundText );
49540             }
49541         },this);
49542        // Roo.log(tv);
49543         
49544         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
49545         this.hiddenField.value = v;
49546         this.value = v;
49547     }
49548     
49549 });/*
49550  * Based on:
49551  * Ext JS Library 1.1.1
49552  * Copyright(c) 2006-2007, Ext JS, LLC.
49553  *
49554  * Originally Released Under LGPL - original licence link has changed is not relivant.
49555  *
49556  * Fork - LGPL
49557  * <script type="text/javascript">
49558  */
49559  
49560 /**
49561  * @class Roo.form.Signature
49562  * @extends Roo.form.Field
49563  * Signature field.  
49564  * @constructor
49565  * 
49566  * @param {Object} config Configuration options
49567  */
49568
49569 Roo.form.Signature = function(config){
49570     Roo.form.Signature.superclass.constructor.call(this, config);
49571     
49572     this.addEvents({// not in used??
49573          /**
49574          * @event confirm
49575          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
49576              * @param {Roo.form.Signature} combo This combo box
49577              */
49578         'confirm' : true,
49579         /**
49580          * @event reset
49581          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
49582              * @param {Roo.form.ComboBox} combo This combo box
49583              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
49584              */
49585         'reset' : true
49586     });
49587 };
49588
49589 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
49590     /**
49591      * @cfg {Object} labels Label to use when rendering a form.
49592      * defaults to 
49593      * labels : { 
49594      *      clear : "Clear",
49595      *      confirm : "Confirm"
49596      *  }
49597      */
49598     labels : { 
49599         clear : "Clear",
49600         confirm : "Confirm"
49601     },
49602     /**
49603      * @cfg {Number} width The signature panel width (defaults to 300)
49604      */
49605     width: 300,
49606     /**
49607      * @cfg {Number} height The signature panel height (defaults to 100)
49608      */
49609     height : 100,
49610     /**
49611      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
49612      */
49613     allowBlank : false,
49614     
49615     //private
49616     // {Object} signPanel The signature SVG panel element (defaults to {})
49617     signPanel : {},
49618     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
49619     isMouseDown : false,
49620     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
49621     isConfirmed : false,
49622     // {String} signatureTmp SVG mapping string (defaults to empty string)
49623     signatureTmp : '',
49624     
49625     
49626     defaultAutoCreate : { // modified by initCompnoent..
49627         tag: "input",
49628         type:"hidden"
49629     },
49630
49631     // private
49632     onRender : function(ct, position){
49633         
49634         Roo.form.Signature.superclass.onRender.call(this, ct, position);
49635         
49636         this.wrap = this.el.wrap({
49637             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
49638         });
49639         
49640         this.createToolbar(this);
49641         this.signPanel = this.wrap.createChild({
49642                 tag: 'div',
49643                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
49644             }, this.el
49645         );
49646             
49647         this.svgID = Roo.id();
49648         this.svgEl = this.signPanel.createChild({
49649               xmlns : 'http://www.w3.org/2000/svg',
49650               tag : 'svg',
49651               id : this.svgID + "-svg",
49652               width: this.width,
49653               height: this.height,
49654               viewBox: '0 0 '+this.width+' '+this.height,
49655               cn : [
49656                 {
49657                     tag: "rect",
49658                     id: this.svgID + "-svg-r",
49659                     width: this.width,
49660                     height: this.height,
49661                     fill: "#ffa"
49662                 },
49663                 {
49664                     tag: "line",
49665                     id: this.svgID + "-svg-l",
49666                     x1: "0", // start
49667                     y1: (this.height*0.8), // start set the line in 80% of height
49668                     x2: this.width, // end
49669                     y2: (this.height*0.8), // end set the line in 80% of height
49670                     'stroke': "#666",
49671                     'stroke-width': "1",
49672                     'stroke-dasharray': "3",
49673                     'shape-rendering': "crispEdges",
49674                     'pointer-events': "none"
49675                 },
49676                 {
49677                     tag: "path",
49678                     id: this.svgID + "-svg-p",
49679                     'stroke': "navy",
49680                     'stroke-width': "3",
49681                     'fill': "none",
49682                     'pointer-events': 'none'
49683                 }
49684               ]
49685         });
49686         this.createSVG();
49687         this.svgBox = this.svgEl.dom.getScreenCTM();
49688     },
49689     createSVG : function(){ 
49690         var svg = this.signPanel;
49691         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
49692         var t = this;
49693
49694         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
49695         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
49696         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
49697         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
49698         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
49699         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
49700         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
49701         
49702     },
49703     isTouchEvent : function(e){
49704         return e.type.match(/^touch/);
49705     },
49706     getCoords : function (e) {
49707         var pt    = this.svgEl.dom.createSVGPoint();
49708         pt.x = e.clientX; 
49709         pt.y = e.clientY;
49710         if (this.isTouchEvent(e)) {
49711             pt.x =  e.targetTouches[0].clientX;
49712             pt.y = e.targetTouches[0].clientY;
49713         }
49714         var a = this.svgEl.dom.getScreenCTM();
49715         var b = a.inverse();
49716         var mx = pt.matrixTransform(b);
49717         return mx.x + ',' + mx.y;
49718     },
49719     //mouse event headler 
49720     down : function (e) {
49721         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
49722         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
49723         
49724         this.isMouseDown = true;
49725         
49726         e.preventDefault();
49727     },
49728     move : function (e) {
49729         if (this.isMouseDown) {
49730             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
49731             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
49732         }
49733         
49734         e.preventDefault();
49735     },
49736     up : function (e) {
49737         this.isMouseDown = false;
49738         var sp = this.signatureTmp.split(' ');
49739         
49740         if(sp.length > 1){
49741             if(!sp[sp.length-2].match(/^L/)){
49742                 sp.pop();
49743                 sp.pop();
49744                 sp.push("");
49745                 this.signatureTmp = sp.join(" ");
49746             }
49747         }
49748         if(this.getValue() != this.signatureTmp){
49749             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49750             this.isConfirmed = false;
49751         }
49752         e.preventDefault();
49753     },
49754     
49755     /**
49756      * Protected method that will not generally be called directly. It
49757      * is called when the editor creates its toolbar. Override this method if you need to
49758      * add custom toolbar buttons.
49759      * @param {HtmlEditor} editor
49760      */
49761     createToolbar : function(editor){
49762          function btn(id, toggle, handler){
49763             var xid = fid + '-'+ id ;
49764             return {
49765                 id : xid,
49766                 cmd : id,
49767                 cls : 'x-btn-icon x-edit-'+id,
49768                 enableToggle:toggle !== false,
49769                 scope: editor, // was editor...
49770                 handler:handler||editor.relayBtnCmd,
49771                 clickEvent:'mousedown',
49772                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
49773                 tabIndex:-1
49774             };
49775         }
49776         
49777         
49778         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
49779         this.tb = tb;
49780         this.tb.add(
49781            {
49782                 cls : ' x-signature-btn x-signature-'+id,
49783                 scope: editor, // was editor...
49784                 handler: this.reset,
49785                 clickEvent:'mousedown',
49786                 text: this.labels.clear
49787             },
49788             {
49789                  xtype : 'Fill',
49790                  xns: Roo.Toolbar
49791             }, 
49792             {
49793                 cls : '  x-signature-btn x-signature-'+id,
49794                 scope: editor, // was editor...
49795                 handler: this.confirmHandler,
49796                 clickEvent:'mousedown',
49797                 text: this.labels.confirm
49798             }
49799         );
49800     
49801     },
49802     //public
49803     /**
49804      * when user is clicked confirm then show this image.....
49805      * 
49806      * @return {String} Image Data URI
49807      */
49808     getImageDataURI : function(){
49809         var svg = this.svgEl.dom.parentNode.innerHTML;
49810         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
49811         return src; 
49812     },
49813     /**
49814      * 
49815      * @return {Boolean} this.isConfirmed
49816      */
49817     getConfirmed : function(){
49818         return this.isConfirmed;
49819     },
49820     /**
49821      * 
49822      * @return {Number} this.width
49823      */
49824     getWidth : function(){
49825         return this.width;
49826     },
49827     /**
49828      * 
49829      * @return {Number} this.height
49830      */
49831     getHeight : function(){
49832         return this.height;
49833     },
49834     // private
49835     getSignature : function(){
49836         return this.signatureTmp;
49837     },
49838     // private
49839     reset : function(){
49840         this.signatureTmp = '';
49841         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49842         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
49843         this.isConfirmed = false;
49844         Roo.form.Signature.superclass.reset.call(this);
49845     },
49846     setSignature : function(s){
49847         this.signatureTmp = s;
49848         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
49849         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
49850         this.setValue(s);
49851         this.isConfirmed = false;
49852         Roo.form.Signature.superclass.reset.call(this);
49853     }, 
49854     test : function(){
49855 //        Roo.log(this.signPanel.dom.contentWindow.up())
49856     },
49857     //private
49858     setConfirmed : function(){
49859         
49860         
49861         
49862 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
49863     },
49864     // private
49865     confirmHandler : function(){
49866         if(!this.getSignature()){
49867             return;
49868         }
49869         
49870         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
49871         this.setValue(this.getSignature());
49872         this.isConfirmed = true;
49873         
49874         this.fireEvent('confirm', this);
49875     },
49876     // private
49877     // Subclasses should provide the validation implementation by overriding this
49878     validateValue : function(value){
49879         if(this.allowBlank){
49880             return true;
49881         }
49882         
49883         if(this.isConfirmed){
49884             return true;
49885         }
49886         return false;
49887     }
49888 });/*
49889  * Based on:
49890  * Ext JS Library 1.1.1
49891  * Copyright(c) 2006-2007, Ext JS, LLC.
49892  *
49893  * Originally Released Under LGPL - original licence link has changed is not relivant.
49894  *
49895  * Fork - LGPL
49896  * <script type="text/javascript">
49897  */
49898  
49899
49900 /**
49901  * @class Roo.form.ComboBox
49902  * @extends Roo.form.TriggerField
49903  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
49904  * @constructor
49905  * Create a new ComboBox.
49906  * @param {Object} config Configuration options
49907  */
49908 Roo.form.Select = function(config){
49909     Roo.form.Select.superclass.constructor.call(this, config);
49910      
49911 };
49912
49913 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
49914     /**
49915      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
49916      */
49917     /**
49918      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
49919      * rendering into an Roo.Editor, defaults to false)
49920      */
49921     /**
49922      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
49923      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
49924      */
49925     /**
49926      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
49927      */
49928     /**
49929      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
49930      * the dropdown list (defaults to undefined, with no header element)
49931      */
49932
49933      /**
49934      * @cfg {String/Roo.Template} tpl The template to use to render the output
49935      */
49936      
49937     // private
49938     defaultAutoCreate : {tag: "select"  },
49939     /**
49940      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
49941      */
49942     listWidth: undefined,
49943     /**
49944      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
49945      * mode = 'remote' or 'text' if mode = 'local')
49946      */
49947     displayField: undefined,
49948     /**
49949      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
49950      * mode = 'remote' or 'value' if mode = 'local'). 
49951      * Note: use of a valueField requires the user make a selection
49952      * in order for a value to be mapped.
49953      */
49954     valueField: undefined,
49955     
49956     
49957     /**
49958      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
49959      * field's data value (defaults to the underlying DOM element's name)
49960      */
49961     hiddenName: undefined,
49962     /**
49963      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
49964      */
49965     listClass: '',
49966     /**
49967      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
49968      */
49969     selectedClass: 'x-combo-selected',
49970     /**
49971      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
49972      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
49973      * which displays a downward arrow icon).
49974      */
49975     triggerClass : 'x-form-arrow-trigger',
49976     /**
49977      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
49978      */
49979     shadow:'sides',
49980     /**
49981      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
49982      * anchor positions (defaults to 'tl-bl')
49983      */
49984     listAlign: 'tl-bl?',
49985     /**
49986      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
49987      */
49988     maxHeight: 300,
49989     /**
49990      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
49991      * query specified by the allQuery config option (defaults to 'query')
49992      */
49993     triggerAction: 'query',
49994     /**
49995      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
49996      * (defaults to 4, does not apply if editable = false)
49997      */
49998     minChars : 4,
49999     /**
50000      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50001      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50002      */
50003     typeAhead: false,
50004     /**
50005      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50006      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50007      */
50008     queryDelay: 500,
50009     /**
50010      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50011      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50012      */
50013     pageSize: 0,
50014     /**
50015      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50016      * when editable = true (defaults to false)
50017      */
50018     selectOnFocus:false,
50019     /**
50020      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50021      */
50022     queryParam: 'query',
50023     /**
50024      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50025      * when mode = 'remote' (defaults to 'Loading...')
50026      */
50027     loadingText: 'Loading...',
50028     /**
50029      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50030      */
50031     resizable: false,
50032     /**
50033      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50034      */
50035     handleHeight : 8,
50036     /**
50037      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50038      * traditional select (defaults to true)
50039      */
50040     editable: true,
50041     /**
50042      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50043      */
50044     allQuery: '',
50045     /**
50046      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50047      */
50048     mode: 'remote',
50049     /**
50050      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50051      * listWidth has a higher value)
50052      */
50053     minListWidth : 70,
50054     /**
50055      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50056      * allow the user to set arbitrary text into the field (defaults to false)
50057      */
50058     forceSelection:false,
50059     /**
50060      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50061      * if typeAhead = true (defaults to 250)
50062      */
50063     typeAheadDelay : 250,
50064     /**
50065      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50066      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50067      */
50068     valueNotFoundText : undefined,
50069     
50070     /**
50071      * @cfg {String} defaultValue The value displayed after loading the store.
50072      */
50073     defaultValue: '',
50074     
50075     /**
50076      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50077      */
50078     blockFocus : false,
50079     
50080     /**
50081      * @cfg {Boolean} disableClear Disable showing of clear button.
50082      */
50083     disableClear : false,
50084     /**
50085      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50086      */
50087     alwaysQuery : false,
50088     
50089     //private
50090     addicon : false,
50091     editicon: false,
50092     
50093     // element that contains real text value.. (when hidden is used..)
50094      
50095     // private
50096     onRender : function(ct, position){
50097         Roo.form.Field.prototype.onRender.call(this, ct, position);
50098         
50099         if(this.store){
50100             this.store.on('beforeload', this.onBeforeLoad, this);
50101             this.store.on('load', this.onLoad, this);
50102             this.store.on('loadexception', this.onLoadException, this);
50103             this.store.load({});
50104         }
50105         
50106         
50107         
50108     },
50109
50110     // private
50111     initEvents : function(){
50112         //Roo.form.ComboBox.superclass.initEvents.call(this);
50113  
50114     },
50115
50116     onDestroy : function(){
50117        
50118         if(this.store){
50119             this.store.un('beforeload', this.onBeforeLoad, this);
50120             this.store.un('load', this.onLoad, this);
50121             this.store.un('loadexception', this.onLoadException, this);
50122         }
50123         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50124     },
50125
50126     // private
50127     fireKey : function(e){
50128         if(e.isNavKeyPress() && !this.list.isVisible()){
50129             this.fireEvent("specialkey", this, e);
50130         }
50131     },
50132
50133     // private
50134     onResize: function(w, h){
50135         
50136         return; 
50137     
50138         
50139     },
50140
50141     /**
50142      * Allow or prevent the user from directly editing the field text.  If false is passed,
50143      * the user will only be able to select from the items defined in the dropdown list.  This method
50144      * is the runtime equivalent of setting the 'editable' config option at config time.
50145      * @param {Boolean} value True to allow the user to directly edit the field text
50146      */
50147     setEditable : function(value){
50148          
50149     },
50150
50151     // private
50152     onBeforeLoad : function(){
50153         
50154         Roo.log("Select before load");
50155         return;
50156     
50157         this.innerList.update(this.loadingText ?
50158                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50159         //this.restrictHeight();
50160         this.selectedIndex = -1;
50161     },
50162
50163     // private
50164     onLoad : function(){
50165
50166     
50167         var dom = this.el.dom;
50168         dom.innerHTML = '';
50169          var od = dom.ownerDocument;
50170          
50171         if (this.emptyText) {
50172             var op = od.createElement('option');
50173             op.setAttribute('value', '');
50174             op.innerHTML = String.format('{0}', this.emptyText);
50175             dom.appendChild(op);
50176         }
50177         if(this.store.getCount() > 0){
50178            
50179             var vf = this.valueField;
50180             var df = this.displayField;
50181             this.store.data.each(function(r) {
50182                 // which colmsn to use... testing - cdoe / title..
50183                 var op = od.createElement('option');
50184                 op.setAttribute('value', r.data[vf]);
50185                 op.innerHTML = String.format('{0}', r.data[df]);
50186                 dom.appendChild(op);
50187             });
50188             if (typeof(this.defaultValue != 'undefined')) {
50189                 this.setValue(this.defaultValue);
50190             }
50191             
50192              
50193         }else{
50194             //this.onEmptyResults();
50195         }
50196         //this.el.focus();
50197     },
50198     // private
50199     onLoadException : function()
50200     {
50201         dom.innerHTML = '';
50202             
50203         Roo.log("Select on load exception");
50204         return;
50205     
50206         this.collapse();
50207         Roo.log(this.store.reader.jsonData);
50208         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50209             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50210         }
50211         
50212         
50213     },
50214     // private
50215     onTypeAhead : function(){
50216          
50217     },
50218
50219     // private
50220     onSelect : function(record, index){
50221         Roo.log('on select?');
50222         return;
50223         if(this.fireEvent('beforeselect', this, record, index) !== false){
50224             this.setFromData(index > -1 ? record.data : false);
50225             this.collapse();
50226             this.fireEvent('select', this, record, index);
50227         }
50228     },
50229
50230     /**
50231      * Returns the currently selected field value or empty string if no value is set.
50232      * @return {String} value The selected value
50233      */
50234     getValue : function(){
50235         var dom = this.el.dom;
50236         this.value = dom.options[dom.selectedIndex].value;
50237         return this.value;
50238         
50239     },
50240
50241     /**
50242      * Clears any text/value currently set in the field
50243      */
50244     clearValue : function(){
50245         this.value = '';
50246         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50247         
50248     },
50249
50250     /**
50251      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50252      * will be displayed in the field.  If the value does not match the data value of an existing item,
50253      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50254      * Otherwise the field will be blank (although the value will still be set).
50255      * @param {String} value The value to match
50256      */
50257     setValue : function(v){
50258         var d = this.el.dom;
50259         for (var i =0; i < d.options.length;i++) {
50260             if (v == d.options[i].value) {
50261                 d.selectedIndex = i;
50262                 this.value = v;
50263                 return;
50264             }
50265         }
50266         this.clearValue();
50267     },
50268     /**
50269      * @property {Object} the last set data for the element
50270      */
50271     
50272     lastData : false,
50273     /**
50274      * Sets the value of the field based on a object which is related to the record format for the store.
50275      * @param {Object} value the value to set as. or false on reset?
50276      */
50277     setFromData : function(o){
50278         Roo.log('setfrom data?');
50279          
50280         
50281         
50282     },
50283     // private
50284     reset : function(){
50285         this.clearValue();
50286     },
50287     // private
50288     findRecord : function(prop, value){
50289         
50290         return false;
50291     
50292         var record;
50293         if(this.store.getCount() > 0){
50294             this.store.each(function(r){
50295                 if(r.data[prop] == value){
50296                     record = r;
50297                     return false;
50298                 }
50299                 return true;
50300             });
50301         }
50302         return record;
50303     },
50304     
50305     getName: function()
50306     {
50307         // returns hidden if it's set..
50308         if (!this.rendered) {return ''};
50309         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
50310         
50311     },
50312      
50313
50314     
50315
50316     // private
50317     onEmptyResults : function(){
50318         Roo.log('empty results');
50319         //this.collapse();
50320     },
50321
50322     /**
50323      * Returns true if the dropdown list is expanded, else false.
50324      */
50325     isExpanded : function(){
50326         return false;
50327     },
50328
50329     /**
50330      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
50331      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50332      * @param {String} value The data value of the item to select
50333      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50334      * selected item if it is not currently in view (defaults to true)
50335      * @return {Boolean} True if the value matched an item in the list, else false
50336      */
50337     selectByValue : function(v, scrollIntoView){
50338         Roo.log('select By Value');
50339         return false;
50340     
50341         if(v !== undefined && v !== null){
50342             var r = this.findRecord(this.valueField || this.displayField, v);
50343             if(r){
50344                 this.select(this.store.indexOf(r), scrollIntoView);
50345                 return true;
50346             }
50347         }
50348         return false;
50349     },
50350
50351     /**
50352      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
50353      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
50354      * @param {Number} index The zero-based index of the list item to select
50355      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
50356      * selected item if it is not currently in view (defaults to true)
50357      */
50358     select : function(index, scrollIntoView){
50359         Roo.log('select ');
50360         return  ;
50361         
50362         this.selectedIndex = index;
50363         this.view.select(index);
50364         if(scrollIntoView !== false){
50365             var el = this.view.getNode(index);
50366             if(el){
50367                 this.innerList.scrollChildIntoView(el, false);
50368             }
50369         }
50370     },
50371
50372       
50373
50374     // private
50375     validateBlur : function(){
50376         
50377         return;
50378         
50379     },
50380
50381     // private
50382     initQuery : function(){
50383         this.doQuery(this.getRawValue());
50384     },
50385
50386     // private
50387     doForce : function(){
50388         if(this.el.dom.value.length > 0){
50389             this.el.dom.value =
50390                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
50391              
50392         }
50393     },
50394
50395     /**
50396      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
50397      * query allowing the query action to be canceled if needed.
50398      * @param {String} query The SQL query to execute
50399      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
50400      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
50401      * saved in the current store (defaults to false)
50402      */
50403     doQuery : function(q, forceAll){
50404         
50405         Roo.log('doQuery?');
50406         if(q === undefined || q === null){
50407             q = '';
50408         }
50409         var qe = {
50410             query: q,
50411             forceAll: forceAll,
50412             combo: this,
50413             cancel:false
50414         };
50415         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
50416             return false;
50417         }
50418         q = qe.query;
50419         forceAll = qe.forceAll;
50420         if(forceAll === true || (q.length >= this.minChars)){
50421             if(this.lastQuery != q || this.alwaysQuery){
50422                 this.lastQuery = q;
50423                 if(this.mode == 'local'){
50424                     this.selectedIndex = -1;
50425                     if(forceAll){
50426                         this.store.clearFilter();
50427                     }else{
50428                         this.store.filter(this.displayField, q);
50429                     }
50430                     this.onLoad();
50431                 }else{
50432                     this.store.baseParams[this.queryParam] = q;
50433                     this.store.load({
50434                         params: this.getParams(q)
50435                     });
50436                     this.expand();
50437                 }
50438             }else{
50439                 this.selectedIndex = -1;
50440                 this.onLoad();   
50441             }
50442         }
50443     },
50444
50445     // private
50446     getParams : function(q){
50447         var p = {};
50448         //p[this.queryParam] = q;
50449         if(this.pageSize){
50450             p.start = 0;
50451             p.limit = this.pageSize;
50452         }
50453         return p;
50454     },
50455
50456     /**
50457      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
50458      */
50459     collapse : function(){
50460         
50461     },
50462
50463     // private
50464     collapseIf : function(e){
50465         
50466     },
50467
50468     /**
50469      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
50470      */
50471     expand : function(){
50472         
50473     } ,
50474
50475     // private
50476      
50477
50478     /** 
50479     * @cfg {Boolean} grow 
50480     * @hide 
50481     */
50482     /** 
50483     * @cfg {Number} growMin 
50484     * @hide 
50485     */
50486     /** 
50487     * @cfg {Number} growMax 
50488     * @hide 
50489     */
50490     /**
50491      * @hide
50492      * @method autoSize
50493      */
50494     
50495     setWidth : function()
50496     {
50497         
50498     },
50499     getResizeEl : function(){
50500         return this.el;
50501     }
50502 });//<script type="text/javasscript">
50503  
50504
50505 /**
50506  * @class Roo.DDView
50507  * A DnD enabled version of Roo.View.
50508  * @param {Element/String} container The Element in which to create the View.
50509  * @param {String} tpl The template string used to create the markup for each element of the View
50510  * @param {Object} config The configuration properties. These include all the config options of
50511  * {@link Roo.View} plus some specific to this class.<br>
50512  * <p>
50513  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
50514  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
50515  * <p>
50516  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
50517 .x-view-drag-insert-above {
50518         border-top:1px dotted #3366cc;
50519 }
50520 .x-view-drag-insert-below {
50521         border-bottom:1px dotted #3366cc;
50522 }
50523 </code></pre>
50524  * 
50525  */
50526  
50527 Roo.DDView = function(container, tpl, config) {
50528     Roo.DDView.superclass.constructor.apply(this, arguments);
50529     this.getEl().setStyle("outline", "0px none");
50530     this.getEl().unselectable();
50531     if (this.dragGroup) {
50532                 this.setDraggable(this.dragGroup.split(","));
50533     }
50534     if (this.dropGroup) {
50535                 this.setDroppable(this.dropGroup.split(","));
50536     }
50537     if (this.deletable) {
50538         this.setDeletable();
50539     }
50540     this.isDirtyFlag = false;
50541         this.addEvents({
50542                 "drop" : true
50543         });
50544 };
50545
50546 Roo.extend(Roo.DDView, Roo.View, {
50547 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
50548 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
50549 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
50550 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
50551
50552         isFormField: true,
50553
50554         reset: Roo.emptyFn,
50555         
50556         clearInvalid: Roo.form.Field.prototype.clearInvalid,
50557
50558         validate: function() {
50559                 return true;
50560         },
50561         
50562         destroy: function() {
50563                 this.purgeListeners();
50564                 this.getEl.removeAllListeners();
50565                 this.getEl().remove();
50566                 if (this.dragZone) {
50567                         if (this.dragZone.destroy) {
50568                                 this.dragZone.destroy();
50569                         }
50570                 }
50571                 if (this.dropZone) {
50572                         if (this.dropZone.destroy) {
50573                                 this.dropZone.destroy();
50574                         }
50575                 }
50576         },
50577
50578 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
50579         getName: function() {
50580                 return this.name;
50581         },
50582
50583 /**     Loads the View from a JSON string representing the Records to put into the Store. */
50584         setValue: function(v) {
50585                 if (!this.store) {
50586                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
50587                 }
50588                 var data = {};
50589                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
50590                 this.store.proxy = new Roo.data.MemoryProxy(data);
50591                 this.store.load();
50592         },
50593
50594 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
50595         getValue: function() {
50596                 var result = '(';
50597                 this.store.each(function(rec) {
50598                         result += rec.id + ',';
50599                 });
50600                 return result.substr(0, result.length - 1) + ')';
50601         },
50602         
50603         getIds: function() {
50604                 var i = 0, result = new Array(this.store.getCount());
50605                 this.store.each(function(rec) {
50606                         result[i++] = rec.id;
50607                 });
50608                 return result;
50609         },
50610         
50611         isDirty: function() {
50612                 return this.isDirtyFlag;
50613         },
50614
50615 /**
50616  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
50617  *      whole Element becomes the target, and this causes the drop gesture to append.
50618  */
50619     getTargetFromEvent : function(e) {
50620                 var target = e.getTarget();
50621                 while ((target !== null) && (target.parentNode != this.el.dom)) {
50622                 target = target.parentNode;
50623                 }
50624                 if (!target) {
50625                         target = this.el.dom.lastChild || this.el.dom;
50626                 }
50627                 return target;
50628     },
50629
50630 /**
50631  *      Create the drag data which consists of an object which has the property "ddel" as
50632  *      the drag proxy element. 
50633  */
50634     getDragData : function(e) {
50635         var target = this.findItemFromChild(e.getTarget());
50636                 if(target) {
50637                         this.handleSelection(e);
50638                         var selNodes = this.getSelectedNodes();
50639             var dragData = {
50640                 source: this,
50641                 copy: this.copy || (this.allowCopy && e.ctrlKey),
50642                 nodes: selNodes,
50643                 records: []
50644                         };
50645                         var selectedIndices = this.getSelectedIndexes();
50646                         for (var i = 0; i < selectedIndices.length; i++) {
50647                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
50648                         }
50649                         if (selNodes.length == 1) {
50650                                 dragData.ddel = target.cloneNode(true); // the div element
50651                         } else {
50652                                 var div = document.createElement('div'); // create the multi element drag "ghost"
50653                                 div.className = 'multi-proxy';
50654                                 for (var i = 0, len = selNodes.length; i < len; i++) {
50655                                         div.appendChild(selNodes[i].cloneNode(true));
50656                                 }
50657                                 dragData.ddel = div;
50658                         }
50659             //console.log(dragData)
50660             //console.log(dragData.ddel.innerHTML)
50661                         return dragData;
50662                 }
50663         //console.log('nodragData')
50664                 return false;
50665     },
50666     
50667 /**     Specify to which ddGroup items in this DDView may be dragged. */
50668     setDraggable: function(ddGroup) {
50669         if (ddGroup instanceof Array) {
50670                 Roo.each(ddGroup, this.setDraggable, this);
50671                 return;
50672         }
50673         if (this.dragZone) {
50674                 this.dragZone.addToGroup(ddGroup);
50675         } else {
50676                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
50677                                 containerScroll: true,
50678                                 ddGroup: ddGroup 
50679
50680                         });
50681 //                      Draggability implies selection. DragZone's mousedown selects the element.
50682                         if (!this.multiSelect) { this.singleSelect = true; }
50683
50684 //                      Wire the DragZone's handlers up to methods in *this*
50685                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
50686                 }
50687     },
50688
50689 /**     Specify from which ddGroup this DDView accepts drops. */
50690     setDroppable: function(ddGroup) {
50691         if (ddGroup instanceof Array) {
50692                 Roo.each(ddGroup, this.setDroppable, this);
50693                 return;
50694         }
50695         if (this.dropZone) {
50696                 this.dropZone.addToGroup(ddGroup);
50697         } else {
50698                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
50699                                 containerScroll: true,
50700                                 ddGroup: ddGroup
50701                         });
50702
50703 //                      Wire the DropZone's handlers up to methods in *this*
50704                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
50705                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
50706                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
50707                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
50708                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
50709                 }
50710     },
50711
50712 /**     Decide whether to drop above or below a View node. */
50713     getDropPoint : function(e, n, dd){
50714         if (n == this.el.dom) { return "above"; }
50715                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
50716                 var c = t + (b - t) / 2;
50717                 var y = Roo.lib.Event.getPageY(e);
50718                 if(y <= c) {
50719                         return "above";
50720                 }else{
50721                         return "below";
50722                 }
50723     },
50724
50725     onNodeEnter : function(n, dd, e, data){
50726                 return false;
50727     },
50728     
50729     onNodeOver : function(n, dd, e, data){
50730                 var pt = this.getDropPoint(e, n, dd);
50731                 // set the insert point style on the target node
50732                 var dragElClass = this.dropNotAllowed;
50733                 if (pt) {
50734                         var targetElClass;
50735                         if (pt == "above"){
50736                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
50737                                 targetElClass = "x-view-drag-insert-above";
50738                         } else {
50739                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
50740                                 targetElClass = "x-view-drag-insert-below";
50741                         }
50742                         if (this.lastInsertClass != targetElClass){
50743                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
50744                                 this.lastInsertClass = targetElClass;
50745                         }
50746                 }
50747                 return dragElClass;
50748         },
50749
50750     onNodeOut : function(n, dd, e, data){
50751                 this.removeDropIndicators(n);
50752     },
50753
50754     onNodeDrop : function(n, dd, e, data){
50755         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
50756                 return false;
50757         }
50758         var pt = this.getDropPoint(e, n, dd);
50759                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
50760                 if (pt == "below") { insertAt++; }
50761                 for (var i = 0; i < data.records.length; i++) {
50762                         var r = data.records[i];
50763                         var dup = this.store.getById(r.id);
50764                         if (dup && (dd != this.dragZone)) {
50765                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
50766                         } else {
50767                                 if (data.copy) {
50768                                         this.store.insert(insertAt++, r.copy());
50769                                 } else {
50770                                         data.source.isDirtyFlag = true;
50771                                         r.store.remove(r);
50772                                         this.store.insert(insertAt++, r);
50773                                 }
50774                                 this.isDirtyFlag = true;
50775                         }
50776                 }
50777                 this.dragZone.cachedTarget = null;
50778                 return true;
50779     },
50780
50781     removeDropIndicators : function(n){
50782                 if(n){
50783                         Roo.fly(n).removeClass([
50784                                 "x-view-drag-insert-above",
50785                                 "x-view-drag-insert-below"]);
50786                         this.lastInsertClass = "_noclass";
50787                 }
50788     },
50789
50790 /**
50791  *      Utility method. Add a delete option to the DDView's context menu.
50792  *      @param {String} imageUrl The URL of the "delete" icon image.
50793  */
50794         setDeletable: function(imageUrl) {
50795                 if (!this.singleSelect && !this.multiSelect) {
50796                         this.singleSelect = true;
50797                 }
50798                 var c = this.getContextMenu();
50799                 this.contextMenu.on("itemclick", function(item) {
50800                         switch (item.id) {
50801                                 case "delete":
50802                                         this.remove(this.getSelectedIndexes());
50803                                         break;
50804                         }
50805                 }, this);
50806                 this.contextMenu.add({
50807                         icon: imageUrl,
50808                         id: "delete",
50809                         text: 'Delete'
50810                 });
50811         },
50812         
50813 /**     Return the context menu for this DDView. */
50814         getContextMenu: function() {
50815                 if (!this.contextMenu) {
50816 //                      Create the View's context menu
50817                         this.contextMenu = new Roo.menu.Menu({
50818                                 id: this.id + "-contextmenu"
50819                         });
50820                         this.el.on("contextmenu", this.showContextMenu, this);
50821                 }
50822                 return this.contextMenu;
50823         },
50824         
50825         disableContextMenu: function() {
50826                 if (this.contextMenu) {
50827                         this.el.un("contextmenu", this.showContextMenu, this);
50828                 }
50829         },
50830
50831         showContextMenu: function(e, item) {
50832         item = this.findItemFromChild(e.getTarget());
50833                 if (item) {
50834                         e.stopEvent();
50835                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
50836                         this.contextMenu.showAt(e.getXY());
50837             }
50838     },
50839
50840 /**
50841  *      Remove {@link Roo.data.Record}s at the specified indices.
50842  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
50843  */
50844     remove: function(selectedIndices) {
50845                 selectedIndices = [].concat(selectedIndices);
50846                 for (var i = 0; i < selectedIndices.length; i++) {
50847                         var rec = this.store.getAt(selectedIndices[i]);
50848                         this.store.remove(rec);
50849                 }
50850     },
50851
50852 /**
50853  *      Double click fires the event, but also, if this is draggable, and there is only one other
50854  *      related DropZone, it transfers the selected node.
50855  */
50856     onDblClick : function(e){
50857         var item = this.findItemFromChild(e.getTarget());
50858         if(item){
50859             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
50860                 return false;
50861             }
50862             if (this.dragGroup) {
50863                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
50864                     while (targets.indexOf(this.dropZone) > -1) {
50865                             targets.remove(this.dropZone);
50866                                 }
50867                     if (targets.length == 1) {
50868                                         this.dragZone.cachedTarget = null;
50869                         var el = Roo.get(targets[0].getEl());
50870                         var box = el.getBox(true);
50871                         targets[0].onNodeDrop(el.dom, {
50872                                 target: el.dom,
50873                                 xy: [box.x, box.y + box.height - 1]
50874                         }, null, this.getDragData(e));
50875                     }
50876                 }
50877         }
50878     },
50879     
50880     handleSelection: function(e) {
50881                 this.dragZone.cachedTarget = null;
50882         var item = this.findItemFromChild(e.getTarget());
50883         if (!item) {
50884                 this.clearSelections(true);
50885                 return;
50886         }
50887                 if (item && (this.multiSelect || this.singleSelect)){
50888                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
50889                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
50890                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
50891                                 this.unselect(item);
50892                         } else {
50893                                 this.select(item, this.multiSelect && e.ctrlKey);
50894                                 this.lastSelection = item;
50895                         }
50896                 }
50897     },
50898
50899     onItemClick : function(item, index, e){
50900                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
50901                         return false;
50902                 }
50903                 return true;
50904     },
50905
50906     unselect : function(nodeInfo, suppressEvent){
50907                 var node = this.getNode(nodeInfo);
50908                 if(node && this.isSelected(node)){
50909                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
50910                                 Roo.fly(node).removeClass(this.selectedClass);
50911                                 this.selections.remove(node);
50912                                 if(!suppressEvent){
50913                                         this.fireEvent("selectionchange", this, this.selections);
50914                                 }
50915                         }
50916                 }
50917     }
50918 });
50919 /*
50920  * Based on:
50921  * Ext JS Library 1.1.1
50922  * Copyright(c) 2006-2007, Ext JS, LLC.
50923  *
50924  * Originally Released Under LGPL - original licence link has changed is not relivant.
50925  *
50926  * Fork - LGPL
50927  * <script type="text/javascript">
50928  */
50929  
50930 /**
50931  * @class Roo.LayoutManager
50932  * @extends Roo.util.Observable
50933  * Base class for layout managers.
50934  */
50935 Roo.LayoutManager = function(container, config){
50936     Roo.LayoutManager.superclass.constructor.call(this);
50937     this.el = Roo.get(container);
50938     // ie scrollbar fix
50939     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
50940         document.body.scroll = "no";
50941     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
50942         this.el.position('relative');
50943     }
50944     this.id = this.el.id;
50945     this.el.addClass("x-layout-container");
50946     /** false to disable window resize monitoring @type Boolean */
50947     this.monitorWindowResize = true;
50948     this.regions = {};
50949     this.addEvents({
50950         /**
50951          * @event layout
50952          * Fires when a layout is performed. 
50953          * @param {Roo.LayoutManager} this
50954          */
50955         "layout" : true,
50956         /**
50957          * @event regionresized
50958          * Fires when the user resizes a region. 
50959          * @param {Roo.LayoutRegion} region The resized region
50960          * @param {Number} newSize The new size (width for east/west, height for north/south)
50961          */
50962         "regionresized" : true,
50963         /**
50964          * @event regioncollapsed
50965          * Fires when a region is collapsed. 
50966          * @param {Roo.LayoutRegion} region The collapsed region
50967          */
50968         "regioncollapsed" : true,
50969         /**
50970          * @event regionexpanded
50971          * Fires when a region is expanded.  
50972          * @param {Roo.LayoutRegion} region The expanded region
50973          */
50974         "regionexpanded" : true
50975     });
50976     this.updating = false;
50977     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
50978 };
50979
50980 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
50981     /**
50982      * Returns true if this layout is currently being updated
50983      * @return {Boolean}
50984      */
50985     isUpdating : function(){
50986         return this.updating; 
50987     },
50988     
50989     /**
50990      * Suspend the LayoutManager from doing auto-layouts while
50991      * making multiple add or remove calls
50992      */
50993     beginUpdate : function(){
50994         this.updating = true;    
50995     },
50996     
50997     /**
50998      * Restore auto-layouts and optionally disable the manager from performing a layout
50999      * @param {Boolean} noLayout true to disable a layout update 
51000      */
51001     endUpdate : function(noLayout){
51002         this.updating = false;
51003         if(!noLayout){
51004             this.layout();
51005         }    
51006     },
51007     
51008     layout: function(){
51009         
51010     },
51011     
51012     onRegionResized : function(region, newSize){
51013         this.fireEvent("regionresized", region, newSize);
51014         this.layout();
51015     },
51016     
51017     onRegionCollapsed : function(region){
51018         this.fireEvent("regioncollapsed", region);
51019     },
51020     
51021     onRegionExpanded : function(region){
51022         this.fireEvent("regionexpanded", region);
51023     },
51024         
51025     /**
51026      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51027      * performs box-model adjustments.
51028      * @return {Object} The size as an object {width: (the width), height: (the height)}
51029      */
51030     getViewSize : function(){
51031         var size;
51032         if(this.el.dom != document.body){
51033             size = this.el.getSize();
51034         }else{
51035             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51036         }
51037         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51038         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51039         return size;
51040     },
51041     
51042     /**
51043      * Returns the Element this layout is bound to.
51044      * @return {Roo.Element}
51045      */
51046     getEl : function(){
51047         return this.el;
51048     },
51049     
51050     /**
51051      * Returns the specified region.
51052      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51053      * @return {Roo.LayoutRegion}
51054      */
51055     getRegion : function(target){
51056         return this.regions[target.toLowerCase()];
51057     },
51058     
51059     onWindowResize : function(){
51060         if(this.monitorWindowResize){
51061             this.layout();
51062         }
51063     }
51064 });/*
51065  * Based on:
51066  * Ext JS Library 1.1.1
51067  * Copyright(c) 2006-2007, Ext JS, LLC.
51068  *
51069  * Originally Released Under LGPL - original licence link has changed is not relivant.
51070  *
51071  * Fork - LGPL
51072  * <script type="text/javascript">
51073  */
51074 /**
51075  * @class Roo.BorderLayout
51076  * @extends Roo.LayoutManager
51077  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51078  * please see: <br><br>
51079  * <a href="http://www.jackslocum.com/yui/2006/10/19/cross-browser-web-20-layouts-with-yahoo-ui/">Cross Browser Layouts - Part 1</a><br>
51080  * <a href="http://www.jackslocum.com/yui/2006/10/28/cross-browser-web-20-layouts-part-2-ajax-feed-viewer-20/">Cross Browser Layouts - Part 2</a><br><br>
51081  * Example:
51082  <pre><code>
51083  var layout = new Roo.BorderLayout(document.body, {
51084     north: {
51085         initialSize: 25,
51086         titlebar: false
51087     },
51088     west: {
51089         split:true,
51090         initialSize: 200,
51091         minSize: 175,
51092         maxSize: 400,
51093         titlebar: true,
51094         collapsible: true
51095     },
51096     east: {
51097         split:true,
51098         initialSize: 202,
51099         minSize: 175,
51100         maxSize: 400,
51101         titlebar: true,
51102         collapsible: true
51103     },
51104     south: {
51105         split:true,
51106         initialSize: 100,
51107         minSize: 100,
51108         maxSize: 200,
51109         titlebar: true,
51110         collapsible: true
51111     },
51112     center: {
51113         titlebar: true,
51114         autoScroll:true,
51115         resizeTabs: true,
51116         minTabWidth: 50,
51117         preferredTabWidth: 150
51118     }
51119 });
51120
51121 // shorthand
51122 var CP = Roo.ContentPanel;
51123
51124 layout.beginUpdate();
51125 layout.add("north", new CP("north", "North"));
51126 layout.add("south", new CP("south", {title: "South", closable: true}));
51127 layout.add("west", new CP("west", {title: "West"}));
51128 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51129 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51130 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51131 layout.getRegion("center").showPanel("center1");
51132 layout.endUpdate();
51133 </code></pre>
51134
51135 <b>The container the layout is rendered into can be either the body element or any other element.
51136 If it is not the body element, the container needs to either be an absolute positioned element,
51137 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51138 the container size if it is not the body element.</b>
51139
51140 * @constructor
51141 * Create a new BorderLayout
51142 * @param {String/HTMLElement/Element} container The container this layout is bound to
51143 * @param {Object} config Configuration options
51144  */
51145 Roo.BorderLayout = function(container, config){
51146     config = config || {};
51147     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51148     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51149     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51150         var target = this.factory.validRegions[i];
51151         if(config[target]){
51152             this.addRegion(target, config[target]);
51153         }
51154     }
51155 };
51156
51157 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51158     /**
51159      * Creates and adds a new region if it doesn't already exist.
51160      * @param {String} target The target region key (north, south, east, west or center).
51161      * @param {Object} config The regions config object
51162      * @return {BorderLayoutRegion} The new region
51163      */
51164     addRegion : function(target, config){
51165         if(!this.regions[target]){
51166             var r = this.factory.create(target, this, config);
51167             this.bindRegion(target, r);
51168         }
51169         return this.regions[target];
51170     },
51171
51172     // private (kinda)
51173     bindRegion : function(name, r){
51174         this.regions[name] = r;
51175         r.on("visibilitychange", this.layout, this);
51176         r.on("paneladded", this.layout, this);
51177         r.on("panelremoved", this.layout, this);
51178         r.on("invalidated", this.layout, this);
51179         r.on("resized", this.onRegionResized, this);
51180         r.on("collapsed", this.onRegionCollapsed, this);
51181         r.on("expanded", this.onRegionExpanded, this);
51182     },
51183
51184     /**
51185      * Performs a layout update.
51186      */
51187     layout : function(){
51188         if(this.updating) {
51189             return;
51190         }
51191         var size = this.getViewSize();
51192         var w = size.width;
51193         var h = size.height;
51194         var centerW = w;
51195         var centerH = h;
51196         var centerY = 0;
51197         var centerX = 0;
51198         //var x = 0, y = 0;
51199
51200         var rs = this.regions;
51201         var north = rs["north"];
51202         var south = rs["south"]; 
51203         var west = rs["west"];
51204         var east = rs["east"];
51205         var center = rs["center"];
51206         //if(this.hideOnLayout){ // not supported anymore
51207             //c.el.setStyle("display", "none");
51208         //}
51209         if(north && north.isVisible()){
51210             var b = north.getBox();
51211             var m = north.getMargins();
51212             b.width = w - (m.left+m.right);
51213             b.x = m.left;
51214             b.y = m.top;
51215             centerY = b.height + b.y + m.bottom;
51216             centerH -= centerY;
51217             north.updateBox(this.safeBox(b));
51218         }
51219         if(south && south.isVisible()){
51220             var b = south.getBox();
51221             var m = south.getMargins();
51222             b.width = w - (m.left+m.right);
51223             b.x = m.left;
51224             var totalHeight = (b.height + m.top + m.bottom);
51225             b.y = h - totalHeight + m.top;
51226             centerH -= totalHeight;
51227             south.updateBox(this.safeBox(b));
51228         }
51229         if(west && west.isVisible()){
51230             var b = west.getBox();
51231             var m = west.getMargins();
51232             b.height = centerH - (m.top+m.bottom);
51233             b.x = m.left;
51234             b.y = centerY + m.top;
51235             var totalWidth = (b.width + m.left + m.right);
51236             centerX += totalWidth;
51237             centerW -= totalWidth;
51238             west.updateBox(this.safeBox(b));
51239         }
51240         if(east && east.isVisible()){
51241             var b = east.getBox();
51242             var m = east.getMargins();
51243             b.height = centerH - (m.top+m.bottom);
51244             var totalWidth = (b.width + m.left + m.right);
51245             b.x = w - totalWidth + m.left;
51246             b.y = centerY + m.top;
51247             centerW -= totalWidth;
51248             east.updateBox(this.safeBox(b));
51249         }
51250         if(center){
51251             var m = center.getMargins();
51252             var centerBox = {
51253                 x: centerX + m.left,
51254                 y: centerY + m.top,
51255                 width: centerW - (m.left+m.right),
51256                 height: centerH - (m.top+m.bottom)
51257             };
51258             //if(this.hideOnLayout){
51259                 //center.el.setStyle("display", "block");
51260             //}
51261             center.updateBox(this.safeBox(centerBox));
51262         }
51263         this.el.repaint();
51264         this.fireEvent("layout", this);
51265     },
51266
51267     // private
51268     safeBox : function(box){
51269         box.width = Math.max(0, box.width);
51270         box.height = Math.max(0, box.height);
51271         return box;
51272     },
51273
51274     /**
51275      * Adds a ContentPanel (or subclass) to this layout.
51276      * @param {String} target The target region key (north, south, east, west or center).
51277      * @param {Roo.ContentPanel} panel The panel to add
51278      * @return {Roo.ContentPanel} The added panel
51279      */
51280     add : function(target, panel){
51281          
51282         target = target.toLowerCase();
51283         return this.regions[target].add(panel);
51284     },
51285
51286     /**
51287      * Remove a ContentPanel (or subclass) to this layout.
51288      * @param {String} target The target region key (north, south, east, west or center).
51289      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
51290      * @return {Roo.ContentPanel} The removed panel
51291      */
51292     remove : function(target, panel){
51293         target = target.toLowerCase();
51294         return this.regions[target].remove(panel);
51295     },
51296
51297     /**
51298      * Searches all regions for a panel with the specified id
51299      * @param {String} panelId
51300      * @return {Roo.ContentPanel} The panel or null if it wasn't found
51301      */
51302     findPanel : function(panelId){
51303         var rs = this.regions;
51304         for(var target in rs){
51305             if(typeof rs[target] != "function"){
51306                 var p = rs[target].getPanel(panelId);
51307                 if(p){
51308                     return p;
51309                 }
51310             }
51311         }
51312         return null;
51313     },
51314
51315     /**
51316      * Searches all regions for a panel with the specified id and activates (shows) it.
51317      * @param {String/ContentPanel} panelId The panels id or the panel itself
51318      * @return {Roo.ContentPanel} The shown panel or null
51319      */
51320     showPanel : function(panelId) {
51321       var rs = this.regions;
51322       for(var target in rs){
51323          var r = rs[target];
51324          if(typeof r != "function"){
51325             if(r.hasPanel(panelId)){
51326                return r.showPanel(panelId);
51327             }
51328          }
51329       }
51330       return null;
51331    },
51332
51333    /**
51334      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
51335      * @param {Roo.state.Provider} provider (optional) An alternate state provider
51336      */
51337     restoreState : function(provider){
51338         if(!provider){
51339             provider = Roo.state.Manager;
51340         }
51341         var sm = new Roo.LayoutStateManager();
51342         sm.init(this, provider);
51343     },
51344
51345     /**
51346      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
51347      * object should contain properties for each region to add ContentPanels to, and each property's value should be
51348      * a valid ContentPanel config object.  Example:
51349      * <pre><code>
51350 // Create the main layout
51351 var layout = new Roo.BorderLayout('main-ct', {
51352     west: {
51353         split:true,
51354         minSize: 175,
51355         titlebar: true
51356     },
51357     center: {
51358         title:'Components'
51359     }
51360 }, 'main-ct');
51361
51362 // Create and add multiple ContentPanels at once via configs
51363 layout.batchAdd({
51364    west: {
51365        id: 'source-files',
51366        autoCreate:true,
51367        title:'Ext Source Files',
51368        autoScroll:true,
51369        fitToFrame:true
51370    },
51371    center : {
51372        el: cview,
51373        autoScroll:true,
51374        fitToFrame:true,
51375        toolbar: tb,
51376        resizeEl:'cbody'
51377    }
51378 });
51379 </code></pre>
51380      * @param {Object} regions An object containing ContentPanel configs by region name
51381      */
51382     batchAdd : function(regions){
51383         this.beginUpdate();
51384         for(var rname in regions){
51385             var lr = this.regions[rname];
51386             if(lr){
51387                 this.addTypedPanels(lr, regions[rname]);
51388             }
51389         }
51390         this.endUpdate();
51391     },
51392
51393     // private
51394     addTypedPanels : function(lr, ps){
51395         if(typeof ps == 'string'){
51396             lr.add(new Roo.ContentPanel(ps));
51397         }
51398         else if(ps instanceof Array){
51399             for(var i =0, len = ps.length; i < len; i++){
51400                 this.addTypedPanels(lr, ps[i]);
51401             }
51402         }
51403         else if(!ps.events){ // raw config?
51404             var el = ps.el;
51405             delete ps.el; // prevent conflict
51406             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
51407         }
51408         else {  // panel object assumed!
51409             lr.add(ps);
51410         }
51411     },
51412     /**
51413      * Adds a xtype elements to the layout.
51414      * <pre><code>
51415
51416 layout.addxtype({
51417        xtype : 'ContentPanel',
51418        region: 'west',
51419        items: [ .... ]
51420    }
51421 );
51422
51423 layout.addxtype({
51424         xtype : 'NestedLayoutPanel',
51425         region: 'west',
51426         layout: {
51427            center: { },
51428            west: { }   
51429         },
51430         items : [ ... list of content panels or nested layout panels.. ]
51431    }
51432 );
51433 </code></pre>
51434      * @param {Object} cfg Xtype definition of item to add.
51435      */
51436     addxtype : function(cfg)
51437     {
51438         // basically accepts a pannel...
51439         // can accept a layout region..!?!?
51440         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
51441         
51442         if (!cfg.xtype.match(/Panel$/)) {
51443             return false;
51444         }
51445         var ret = false;
51446         
51447         if (typeof(cfg.region) == 'undefined') {
51448             Roo.log("Failed to add Panel, region was not set");
51449             Roo.log(cfg);
51450             return false;
51451         }
51452         var region = cfg.region;
51453         delete cfg.region;
51454         
51455           
51456         var xitems = [];
51457         if (cfg.items) {
51458             xitems = cfg.items;
51459             delete cfg.items;
51460         }
51461         var nb = false;
51462         
51463         switch(cfg.xtype) 
51464         {
51465             case 'ContentPanel':  // ContentPanel (el, cfg)
51466             case 'ScrollPanel':  // ContentPanel (el, cfg)
51467             case 'ViewPanel': 
51468                 if(cfg.autoCreate) {
51469                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51470                 } else {
51471                     var el = this.el.createChild();
51472                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
51473                 }
51474                 
51475                 this.add(region, ret);
51476                 break;
51477             
51478             
51479             case 'TreePanel': // our new panel!
51480                 cfg.el = this.el.createChild();
51481                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51482                 this.add(region, ret);
51483                 break;
51484             
51485             case 'NestedLayoutPanel': 
51486                 // create a new Layout (which is  a Border Layout...
51487                 var el = this.el.createChild();
51488                 var clayout = cfg.layout;
51489                 delete cfg.layout;
51490                 clayout.items   = clayout.items  || [];
51491                 // replace this exitems with the clayout ones..
51492                 xitems = clayout.items;
51493                  
51494                 
51495                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
51496                     cfg.background = false;
51497                 }
51498                 var layout = new Roo.BorderLayout(el, clayout);
51499                 
51500                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
51501                 //console.log('adding nested layout panel '  + cfg.toSource());
51502                 this.add(region, ret);
51503                 nb = {}; /// find first...
51504                 break;
51505                 
51506             case 'GridPanel': 
51507             
51508                 // needs grid and region
51509                 
51510                 //var el = this.getRegion(region).el.createChild();
51511                 var el = this.el.createChild();
51512                 // create the grid first...
51513                 
51514                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
51515                 delete cfg.grid;
51516                 if (region == 'center' && this.active ) {
51517                     cfg.background = false;
51518                 }
51519                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
51520                 
51521                 this.add(region, ret);
51522                 if (cfg.background) {
51523                     ret.on('activate', function(gp) {
51524                         if (!gp.grid.rendered) {
51525                             gp.grid.render();
51526                         }
51527                     });
51528                 } else {
51529                     grid.render();
51530                 }
51531                 break;
51532            
51533            
51534            
51535                 
51536                 
51537                 
51538             default:
51539                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
51540                     
51541                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
51542                     this.add(region, ret);
51543                 } else {
51544                 
51545                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
51546                     return null;
51547                 }
51548                 
51549              // GridPanel (grid, cfg)
51550             
51551         }
51552         this.beginUpdate();
51553         // add children..
51554         var region = '';
51555         var abn = {};
51556         Roo.each(xitems, function(i)  {
51557             region = nb && i.region ? i.region : false;
51558             
51559             var add = ret.addxtype(i);
51560            
51561             if (region) {
51562                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
51563                 if (!i.background) {
51564                     abn[region] = nb[region] ;
51565                 }
51566             }
51567             
51568         });
51569         this.endUpdate();
51570
51571         // make the last non-background panel active..
51572         //if (nb) { Roo.log(abn); }
51573         if (nb) {
51574             
51575             for(var r in abn) {
51576                 region = this.getRegion(r);
51577                 if (region) {
51578                     // tried using nb[r], but it does not work..
51579                      
51580                     region.showPanel(abn[r]);
51581                    
51582                 }
51583             }
51584         }
51585         return ret;
51586         
51587     }
51588 });
51589
51590 /**
51591  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
51592  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
51593  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
51594  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
51595  * <pre><code>
51596 // shorthand
51597 var CP = Roo.ContentPanel;
51598
51599 var layout = Roo.BorderLayout.create({
51600     north: {
51601         initialSize: 25,
51602         titlebar: false,
51603         panels: [new CP("north", "North")]
51604     },
51605     west: {
51606         split:true,
51607         initialSize: 200,
51608         minSize: 175,
51609         maxSize: 400,
51610         titlebar: true,
51611         collapsible: true,
51612         panels: [new CP("west", {title: "West"})]
51613     },
51614     east: {
51615         split:true,
51616         initialSize: 202,
51617         minSize: 175,
51618         maxSize: 400,
51619         titlebar: true,
51620         collapsible: true,
51621         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
51622     },
51623     south: {
51624         split:true,
51625         initialSize: 100,
51626         minSize: 100,
51627         maxSize: 200,
51628         titlebar: true,
51629         collapsible: true,
51630         panels: [new CP("south", {title: "South", closable: true})]
51631     },
51632     center: {
51633         titlebar: true,
51634         autoScroll:true,
51635         resizeTabs: true,
51636         minTabWidth: 50,
51637         preferredTabWidth: 150,
51638         panels: [
51639             new CP("center1", {title: "Close Me", closable: true}),
51640             new CP("center2", {title: "Center Panel", closable: false})
51641         ]
51642     }
51643 }, document.body);
51644
51645 layout.getRegion("center").showPanel("center1");
51646 </code></pre>
51647  * @param config
51648  * @param targetEl
51649  */
51650 Roo.BorderLayout.create = function(config, targetEl){
51651     var layout = new Roo.BorderLayout(targetEl || document.body, config);
51652     layout.beginUpdate();
51653     var regions = Roo.BorderLayout.RegionFactory.validRegions;
51654     for(var j = 0, jlen = regions.length; j < jlen; j++){
51655         var lr = regions[j];
51656         if(layout.regions[lr] && config[lr].panels){
51657             var r = layout.regions[lr];
51658             var ps = config[lr].panels;
51659             layout.addTypedPanels(r, ps);
51660         }
51661     }
51662     layout.endUpdate();
51663     return layout;
51664 };
51665
51666 // private
51667 Roo.BorderLayout.RegionFactory = {
51668     // private
51669     validRegions : ["north","south","east","west","center"],
51670
51671     // private
51672     create : function(target, mgr, config){
51673         target = target.toLowerCase();
51674         if(config.lightweight || config.basic){
51675             return new Roo.BasicLayoutRegion(mgr, config, target);
51676         }
51677         switch(target){
51678             case "north":
51679                 return new Roo.NorthLayoutRegion(mgr, config);
51680             case "south":
51681                 return new Roo.SouthLayoutRegion(mgr, config);
51682             case "east":
51683                 return new Roo.EastLayoutRegion(mgr, config);
51684             case "west":
51685                 return new Roo.WestLayoutRegion(mgr, config);
51686             case "center":
51687                 return new Roo.CenterLayoutRegion(mgr, config);
51688         }
51689         throw 'Layout region "'+target+'" not supported.';
51690     }
51691 };/*
51692  * Based on:
51693  * Ext JS Library 1.1.1
51694  * Copyright(c) 2006-2007, Ext JS, LLC.
51695  *
51696  * Originally Released Under LGPL - original licence link has changed is not relivant.
51697  *
51698  * Fork - LGPL
51699  * <script type="text/javascript">
51700  */
51701  
51702 /**
51703  * @class Roo.BasicLayoutRegion
51704  * @extends Roo.util.Observable
51705  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
51706  * and does not have a titlebar, tabs or any other features. All it does is size and position 
51707  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
51708  */
51709 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
51710     this.mgr = mgr;
51711     this.position  = pos;
51712     this.events = {
51713         /**
51714          * @scope Roo.BasicLayoutRegion
51715          */
51716         
51717         /**
51718          * @event beforeremove
51719          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
51720          * @param {Roo.LayoutRegion} this
51721          * @param {Roo.ContentPanel} panel The panel
51722          * @param {Object} e The cancel event object
51723          */
51724         "beforeremove" : true,
51725         /**
51726          * @event invalidated
51727          * Fires when the layout for this region is changed.
51728          * @param {Roo.LayoutRegion} this
51729          */
51730         "invalidated" : true,
51731         /**
51732          * @event visibilitychange
51733          * Fires when this region is shown or hidden 
51734          * @param {Roo.LayoutRegion} this
51735          * @param {Boolean} visibility true or false
51736          */
51737         "visibilitychange" : true,
51738         /**
51739          * @event paneladded
51740          * Fires when a panel is added. 
51741          * @param {Roo.LayoutRegion} this
51742          * @param {Roo.ContentPanel} panel The panel
51743          */
51744         "paneladded" : true,
51745         /**
51746          * @event panelremoved
51747          * Fires when a panel is removed. 
51748          * @param {Roo.LayoutRegion} this
51749          * @param {Roo.ContentPanel} panel The panel
51750          */
51751         "panelremoved" : true,
51752         /**
51753          * @event beforecollapse
51754          * Fires when this region before collapse.
51755          * @param {Roo.LayoutRegion} this
51756          */
51757         "beforecollapse" : true,
51758         /**
51759          * @event collapsed
51760          * Fires when this region is collapsed.
51761          * @param {Roo.LayoutRegion} this
51762          */
51763         "collapsed" : true,
51764         /**
51765          * @event expanded
51766          * Fires when this region is expanded.
51767          * @param {Roo.LayoutRegion} this
51768          */
51769         "expanded" : true,
51770         /**
51771          * @event slideshow
51772          * Fires when this region is slid into view.
51773          * @param {Roo.LayoutRegion} this
51774          */
51775         "slideshow" : true,
51776         /**
51777          * @event slidehide
51778          * Fires when this region slides out of view. 
51779          * @param {Roo.LayoutRegion} this
51780          */
51781         "slidehide" : true,
51782         /**
51783          * @event panelactivated
51784          * Fires when a panel is activated. 
51785          * @param {Roo.LayoutRegion} this
51786          * @param {Roo.ContentPanel} panel The activated panel
51787          */
51788         "panelactivated" : true,
51789         /**
51790          * @event resized
51791          * Fires when the user resizes this region. 
51792          * @param {Roo.LayoutRegion} this
51793          * @param {Number} newSize The new size (width for east/west, height for north/south)
51794          */
51795         "resized" : true
51796     };
51797     /** A collection of panels in this region. @type Roo.util.MixedCollection */
51798     this.panels = new Roo.util.MixedCollection();
51799     this.panels.getKey = this.getPanelId.createDelegate(this);
51800     this.box = null;
51801     this.activePanel = null;
51802     // ensure listeners are added...
51803     
51804     if (config.listeners || config.events) {
51805         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
51806             listeners : config.listeners || {},
51807             events : config.events || {}
51808         });
51809     }
51810     
51811     if(skipConfig !== true){
51812         this.applyConfig(config);
51813     }
51814 };
51815
51816 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
51817     getPanelId : function(p){
51818         return p.getId();
51819     },
51820     
51821     applyConfig : function(config){
51822         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
51823         this.config = config;
51824         
51825     },
51826     
51827     /**
51828      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
51829      * the width, for horizontal (north, south) the height.
51830      * @param {Number} newSize The new width or height
51831      */
51832     resizeTo : function(newSize){
51833         var el = this.el ? this.el :
51834                  (this.activePanel ? this.activePanel.getEl() : null);
51835         if(el){
51836             switch(this.position){
51837                 case "east":
51838                 case "west":
51839                     el.setWidth(newSize);
51840                     this.fireEvent("resized", this, newSize);
51841                 break;
51842                 case "north":
51843                 case "south":
51844                     el.setHeight(newSize);
51845                     this.fireEvent("resized", this, newSize);
51846                 break;                
51847             }
51848         }
51849     },
51850     
51851     getBox : function(){
51852         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
51853     },
51854     
51855     getMargins : function(){
51856         return this.margins;
51857     },
51858     
51859     updateBox : function(box){
51860         this.box = box;
51861         var el = this.activePanel.getEl();
51862         el.dom.style.left = box.x + "px";
51863         el.dom.style.top = box.y + "px";
51864         this.activePanel.setSize(box.width, box.height);
51865     },
51866     
51867     /**
51868      * Returns the container element for this region.
51869      * @return {Roo.Element}
51870      */
51871     getEl : function(){
51872         return this.activePanel;
51873     },
51874     
51875     /**
51876      * Returns true if this region is currently visible.
51877      * @return {Boolean}
51878      */
51879     isVisible : function(){
51880         return this.activePanel ? true : false;
51881     },
51882     
51883     setActivePanel : function(panel){
51884         panel = this.getPanel(panel);
51885         if(this.activePanel && this.activePanel != panel){
51886             this.activePanel.setActiveState(false);
51887             this.activePanel.getEl().setLeftTop(-10000,-10000);
51888         }
51889         this.activePanel = panel;
51890         panel.setActiveState(true);
51891         if(this.box){
51892             panel.setSize(this.box.width, this.box.height);
51893         }
51894         this.fireEvent("panelactivated", this, panel);
51895         this.fireEvent("invalidated");
51896     },
51897     
51898     /**
51899      * Show the specified panel.
51900      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
51901      * @return {Roo.ContentPanel} The shown panel or null
51902      */
51903     showPanel : function(panel){
51904         if(panel = this.getPanel(panel)){
51905             this.setActivePanel(panel);
51906         }
51907         return panel;
51908     },
51909     
51910     /**
51911      * Get the active panel for this region.
51912      * @return {Roo.ContentPanel} The active panel or null
51913      */
51914     getActivePanel : function(){
51915         return this.activePanel;
51916     },
51917     
51918     /**
51919      * Add the passed ContentPanel(s)
51920      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
51921      * @return {Roo.ContentPanel} The panel added (if only one was added)
51922      */
51923     add : function(panel){
51924         if(arguments.length > 1){
51925             for(var i = 0, len = arguments.length; i < len; i++) {
51926                 this.add(arguments[i]);
51927             }
51928             return null;
51929         }
51930         if(this.hasPanel(panel)){
51931             this.showPanel(panel);
51932             return panel;
51933         }
51934         var el = panel.getEl();
51935         if(el.dom.parentNode != this.mgr.el.dom){
51936             this.mgr.el.dom.appendChild(el.dom);
51937         }
51938         if(panel.setRegion){
51939             panel.setRegion(this);
51940         }
51941         this.panels.add(panel);
51942         el.setStyle("position", "absolute");
51943         if(!panel.background){
51944             this.setActivePanel(panel);
51945             if(this.config.initialSize && this.panels.getCount()==1){
51946                 this.resizeTo(this.config.initialSize);
51947             }
51948         }
51949         this.fireEvent("paneladded", this, panel);
51950         return panel;
51951     },
51952     
51953     /**
51954      * Returns true if the panel is in this region.
51955      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
51956      * @return {Boolean}
51957      */
51958     hasPanel : function(panel){
51959         if(typeof panel == "object"){ // must be panel obj
51960             panel = panel.getId();
51961         }
51962         return this.getPanel(panel) ? true : false;
51963     },
51964     
51965     /**
51966      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
51967      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
51968      * @param {Boolean} preservePanel Overrides the config preservePanel option
51969      * @return {Roo.ContentPanel} The panel that was removed
51970      */
51971     remove : function(panel, preservePanel){
51972         panel = this.getPanel(panel);
51973         if(!panel){
51974             return null;
51975         }
51976         var e = {};
51977         this.fireEvent("beforeremove", this, panel, e);
51978         if(e.cancel === true){
51979             return null;
51980         }
51981         var panelId = panel.getId();
51982         this.panels.removeKey(panelId);
51983         return panel;
51984     },
51985     
51986     /**
51987      * Returns the panel specified or null if it's not in this region.
51988      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
51989      * @return {Roo.ContentPanel}
51990      */
51991     getPanel : function(id){
51992         if(typeof id == "object"){ // must be panel obj
51993             return id;
51994         }
51995         return this.panels.get(id);
51996     },
51997     
51998     /**
51999      * Returns this regions position (north/south/east/west/center).
52000      * @return {String} 
52001      */
52002     getPosition: function(){
52003         return this.position;    
52004     }
52005 });/*
52006  * Based on:
52007  * Ext JS Library 1.1.1
52008  * Copyright(c) 2006-2007, Ext JS, LLC.
52009  *
52010  * Originally Released Under LGPL - original licence link has changed is not relivant.
52011  *
52012  * Fork - LGPL
52013  * <script type="text/javascript">
52014  */
52015  
52016 /**
52017  * @class Roo.LayoutRegion
52018  * @extends Roo.BasicLayoutRegion
52019  * This class represents a region in a layout manager.
52020  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52021  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52022  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52023  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52024  * @cfg {Object}    cmargins        Margins for the element when collapsed (defaults to: north/south {top: 2, left: 0, right:0, bottom: 2} or east/west {top: 0, left: 2, right:2, bottom: 0})
52025  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52026  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52027  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52028  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52029  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52030  * @cfg {String}    title           The title for the region (overrides panel titles)
52031  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52032  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52033  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52034  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52035  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52036  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52037  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52038  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52039  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52040  * @cfg {Boolean}   showPin         True to show a pin button
52041  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52042  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52043  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52044  * @cfg {Number}    width           For East/West panels
52045  * @cfg {Number}    height          For North/South panels
52046  * @cfg {Boolean}   split           To show the splitter
52047  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52048  */
52049 Roo.LayoutRegion = function(mgr, config, pos){
52050     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52051     var dh = Roo.DomHelper;
52052     /** This region's container element 
52053     * @type Roo.Element */
52054     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52055     /** This region's title element 
52056     * @type Roo.Element */
52057
52058     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52059         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52060         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52061     ]}, true);
52062     this.titleEl.enableDisplayMode();
52063     /** This region's title text element 
52064     * @type HTMLElement */
52065     this.titleTextEl = this.titleEl.dom.firstChild;
52066     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52067     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52068     this.closeBtn.enableDisplayMode();
52069     this.closeBtn.on("click", this.closeClicked, this);
52070     this.closeBtn.hide();
52071
52072     this.createBody(config);
52073     this.visible = true;
52074     this.collapsed = false;
52075
52076     if(config.hideWhenEmpty){
52077         this.hide();
52078         this.on("paneladded", this.validateVisibility, this);
52079         this.on("panelremoved", this.validateVisibility, this);
52080     }
52081     this.applyConfig(config);
52082 };
52083
52084 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52085
52086     createBody : function(){
52087         /** This region's body element 
52088         * @type Roo.Element */
52089         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52090     },
52091
52092     applyConfig : function(c){
52093         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52094             var dh = Roo.DomHelper;
52095             if(c.titlebar !== false){
52096                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52097                 this.collapseBtn.on("click", this.collapse, this);
52098                 this.collapseBtn.enableDisplayMode();
52099
52100                 if(c.showPin === true || this.showPin){
52101                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52102                     this.stickBtn.enableDisplayMode();
52103                     this.stickBtn.on("click", this.expand, this);
52104                     this.stickBtn.hide();
52105                 }
52106             }
52107             /** This region's collapsed element
52108             * @type Roo.Element */
52109             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52110                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52111             ]}, true);
52112             if(c.floatable !== false){
52113                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52114                this.collapsedEl.on("click", this.collapseClick, this);
52115             }
52116
52117             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52118                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52119                    id: "message", unselectable: "on", style:{"float":"left"}});
52120                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52121              }
52122             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52123             this.expandBtn.on("click", this.expand, this);
52124         }
52125         if(this.collapseBtn){
52126             this.collapseBtn.setVisible(c.collapsible == true);
52127         }
52128         this.cmargins = c.cmargins || this.cmargins ||
52129                          (this.position == "west" || this.position == "east" ?
52130                              {top: 0, left: 2, right:2, bottom: 0} :
52131                              {top: 2, left: 0, right:0, bottom: 2});
52132         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52133         this.bottomTabs = c.tabPosition != "top";
52134         this.autoScroll = c.autoScroll || false;
52135         if(this.autoScroll){
52136             this.bodyEl.setStyle("overflow", "auto");
52137         }else{
52138             this.bodyEl.setStyle("overflow", "hidden");
52139         }
52140         //if(c.titlebar !== false){
52141             if((!c.titlebar && !c.title) || c.titlebar === false){
52142                 this.titleEl.hide();
52143             }else{
52144                 this.titleEl.show();
52145                 if(c.title){
52146                     this.titleTextEl.innerHTML = c.title;
52147                 }
52148             }
52149         //}
52150         this.duration = c.duration || .30;
52151         this.slideDuration = c.slideDuration || .45;
52152         this.config = c;
52153         if(c.collapsed){
52154             this.collapse(true);
52155         }
52156         if(c.hidden){
52157             this.hide();
52158         }
52159     },
52160     /**
52161      * Returns true if this region is currently visible.
52162      * @return {Boolean}
52163      */
52164     isVisible : function(){
52165         return this.visible;
52166     },
52167
52168     /**
52169      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52170      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52171      */
52172     setCollapsedTitle : function(title){
52173         title = title || "&#160;";
52174         if(this.collapsedTitleTextEl){
52175             this.collapsedTitleTextEl.innerHTML = title;
52176         }
52177     },
52178
52179     getBox : function(){
52180         var b;
52181         if(!this.collapsed){
52182             b = this.el.getBox(false, true);
52183         }else{
52184             b = this.collapsedEl.getBox(false, true);
52185         }
52186         return b;
52187     },
52188
52189     getMargins : function(){
52190         return this.collapsed ? this.cmargins : this.margins;
52191     },
52192
52193     highlight : function(){
52194         this.el.addClass("x-layout-panel-dragover");
52195     },
52196
52197     unhighlight : function(){
52198         this.el.removeClass("x-layout-panel-dragover");
52199     },
52200
52201     updateBox : function(box){
52202         this.box = box;
52203         if(!this.collapsed){
52204             this.el.dom.style.left = box.x + "px";
52205             this.el.dom.style.top = box.y + "px";
52206             this.updateBody(box.width, box.height);
52207         }else{
52208             this.collapsedEl.dom.style.left = box.x + "px";
52209             this.collapsedEl.dom.style.top = box.y + "px";
52210             this.collapsedEl.setSize(box.width, box.height);
52211         }
52212         if(this.tabs){
52213             this.tabs.autoSizeTabs();
52214         }
52215     },
52216
52217     updateBody : function(w, h){
52218         if(w !== null){
52219             this.el.setWidth(w);
52220             w -= this.el.getBorderWidth("rl");
52221             if(this.config.adjustments){
52222                 w += this.config.adjustments[0];
52223             }
52224         }
52225         if(h !== null){
52226             this.el.setHeight(h);
52227             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52228             h -= this.el.getBorderWidth("tb");
52229             if(this.config.adjustments){
52230                 h += this.config.adjustments[1];
52231             }
52232             this.bodyEl.setHeight(h);
52233             if(this.tabs){
52234                 h = this.tabs.syncHeight(h);
52235             }
52236         }
52237         if(this.panelSize){
52238             w = w !== null ? w : this.panelSize.width;
52239             h = h !== null ? h : this.panelSize.height;
52240         }
52241         if(this.activePanel){
52242             var el = this.activePanel.getEl();
52243             w = w !== null ? w : el.getWidth();
52244             h = h !== null ? h : el.getHeight();
52245             this.panelSize = {width: w, height: h};
52246             this.activePanel.setSize(w, h);
52247         }
52248         if(Roo.isIE && this.tabs){
52249             this.tabs.el.repaint();
52250         }
52251     },
52252
52253     /**
52254      * Returns the container element for this region.
52255      * @return {Roo.Element}
52256      */
52257     getEl : function(){
52258         return this.el;
52259     },
52260
52261     /**
52262      * Hides this region.
52263      */
52264     hide : function(){
52265         if(!this.collapsed){
52266             this.el.dom.style.left = "-2000px";
52267             this.el.hide();
52268         }else{
52269             this.collapsedEl.dom.style.left = "-2000px";
52270             this.collapsedEl.hide();
52271         }
52272         this.visible = false;
52273         this.fireEvent("visibilitychange", this, false);
52274     },
52275
52276     /**
52277      * Shows this region if it was previously hidden.
52278      */
52279     show : function(){
52280         if(!this.collapsed){
52281             this.el.show();
52282         }else{
52283             this.collapsedEl.show();
52284         }
52285         this.visible = true;
52286         this.fireEvent("visibilitychange", this, true);
52287     },
52288
52289     closeClicked : function(){
52290         if(this.activePanel){
52291             this.remove(this.activePanel);
52292         }
52293     },
52294
52295     collapseClick : function(e){
52296         if(this.isSlid){
52297            e.stopPropagation();
52298            this.slideIn();
52299         }else{
52300            e.stopPropagation();
52301            this.slideOut();
52302         }
52303     },
52304
52305     /**
52306      * Collapses this region.
52307      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
52308      */
52309     collapse : function(skipAnim, skipCheck = false){
52310         if(this.collapsed) {
52311             return;
52312         }
52313         
52314         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
52315             
52316             this.collapsed = true;
52317             if(this.split){
52318                 this.split.el.hide();
52319             }
52320             if(this.config.animate && skipAnim !== true){
52321                 this.fireEvent("invalidated", this);
52322                 this.animateCollapse();
52323             }else{
52324                 this.el.setLocation(-20000,-20000);
52325                 this.el.hide();
52326                 this.collapsedEl.show();
52327                 this.fireEvent("collapsed", this);
52328                 this.fireEvent("invalidated", this);
52329             }
52330         }
52331         
52332     },
52333
52334     animateCollapse : function(){
52335         // overridden
52336     },
52337
52338     /**
52339      * Expands this region if it was previously collapsed.
52340      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
52341      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
52342      */
52343     expand : function(e, skipAnim){
52344         if(e) {
52345             e.stopPropagation();
52346         }
52347         if(!this.collapsed || this.el.hasActiveFx()) {
52348             return;
52349         }
52350         if(this.isSlid){
52351             this.afterSlideIn();
52352             skipAnim = true;
52353         }
52354         this.collapsed = false;
52355         if(this.config.animate && skipAnim !== true){
52356             this.animateExpand();
52357         }else{
52358             this.el.show();
52359             if(this.split){
52360                 this.split.el.show();
52361             }
52362             this.collapsedEl.setLocation(-2000,-2000);
52363             this.collapsedEl.hide();
52364             this.fireEvent("invalidated", this);
52365             this.fireEvent("expanded", this);
52366         }
52367     },
52368
52369     animateExpand : function(){
52370         // overridden
52371     },
52372
52373     initTabs : function()
52374     {
52375         this.bodyEl.setStyle("overflow", "hidden");
52376         var ts = new Roo.TabPanel(
52377                 this.bodyEl.dom,
52378                 {
52379                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
52380                     disableTooltips: this.config.disableTabTips,
52381                     toolbar : this.config.toolbar
52382                 }
52383         );
52384         if(this.config.hideTabs){
52385             ts.stripWrap.setDisplayed(false);
52386         }
52387         this.tabs = ts;
52388         ts.resizeTabs = this.config.resizeTabs === true;
52389         ts.minTabWidth = this.config.minTabWidth || 40;
52390         ts.maxTabWidth = this.config.maxTabWidth || 250;
52391         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
52392         ts.monitorResize = false;
52393         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52394         ts.bodyEl.addClass('x-layout-tabs-body');
52395         this.panels.each(this.initPanelAsTab, this);
52396     },
52397
52398     initPanelAsTab : function(panel){
52399         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
52400                     this.config.closeOnTab && panel.isClosable());
52401         if(panel.tabTip !== undefined){
52402             ti.setTooltip(panel.tabTip);
52403         }
52404         ti.on("activate", function(){
52405               this.setActivePanel(panel);
52406         }, this);
52407         if(this.config.closeOnTab){
52408             ti.on("beforeclose", function(t, e){
52409                 e.cancel = true;
52410                 this.remove(panel);
52411             }, this);
52412         }
52413         return ti;
52414     },
52415
52416     updatePanelTitle : function(panel, title){
52417         if(this.activePanel == panel){
52418             this.updateTitle(title);
52419         }
52420         if(this.tabs){
52421             var ti = this.tabs.getTab(panel.getEl().id);
52422             ti.setText(title);
52423             if(panel.tabTip !== undefined){
52424                 ti.setTooltip(panel.tabTip);
52425             }
52426         }
52427     },
52428
52429     updateTitle : function(title){
52430         if(this.titleTextEl && !this.config.title){
52431             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
52432         }
52433     },
52434
52435     setActivePanel : function(panel){
52436         panel = this.getPanel(panel);
52437         if(this.activePanel && this.activePanel != panel){
52438             this.activePanel.setActiveState(false);
52439         }
52440         this.activePanel = panel;
52441         panel.setActiveState(true);
52442         if(this.panelSize){
52443             panel.setSize(this.panelSize.width, this.panelSize.height);
52444         }
52445         if(this.closeBtn){
52446             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
52447         }
52448         this.updateTitle(panel.getTitle());
52449         if(this.tabs){
52450             this.fireEvent("invalidated", this);
52451         }
52452         this.fireEvent("panelactivated", this, panel);
52453     },
52454
52455     /**
52456      * Shows the specified panel.
52457      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
52458      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
52459      */
52460     showPanel : function(panel)
52461     {
52462         panel = this.getPanel(panel);
52463         if(panel){
52464             if(this.tabs){
52465                 var tab = this.tabs.getTab(panel.getEl().id);
52466                 if(tab.isHidden()){
52467                     this.tabs.unhideTab(tab.id);
52468                 }
52469                 tab.activate();
52470             }else{
52471                 this.setActivePanel(panel);
52472             }
52473         }
52474         return panel;
52475     },
52476
52477     /**
52478      * Get the active panel for this region.
52479      * @return {Roo.ContentPanel} The active panel or null
52480      */
52481     getActivePanel : function(){
52482         return this.activePanel;
52483     },
52484
52485     validateVisibility : function(){
52486         if(this.panels.getCount() < 1){
52487             this.updateTitle("&#160;");
52488             this.closeBtn.hide();
52489             this.hide();
52490         }else{
52491             if(!this.isVisible()){
52492                 this.show();
52493             }
52494         }
52495     },
52496
52497     /**
52498      * Adds the passed ContentPanel(s) to this region.
52499      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52500      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
52501      */
52502     add : function(panel){
52503         if(arguments.length > 1){
52504             for(var i = 0, len = arguments.length; i < len; i++) {
52505                 this.add(arguments[i]);
52506             }
52507             return null;
52508         }
52509         if(this.hasPanel(panel)){
52510             this.showPanel(panel);
52511             return panel;
52512         }
52513         panel.setRegion(this);
52514         this.panels.add(panel);
52515         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
52516             this.bodyEl.dom.appendChild(panel.getEl().dom);
52517             if(panel.background !== true){
52518                 this.setActivePanel(panel);
52519             }
52520             this.fireEvent("paneladded", this, panel);
52521             return panel;
52522         }
52523         if(!this.tabs){
52524             this.initTabs();
52525         }else{
52526             this.initPanelAsTab(panel);
52527         }
52528         if(panel.background !== true){
52529             this.tabs.activate(panel.getEl().id);
52530         }
52531         this.fireEvent("paneladded", this, panel);
52532         return panel;
52533     },
52534
52535     /**
52536      * Hides the tab for the specified panel.
52537      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52538      */
52539     hidePanel : function(panel){
52540         if(this.tabs && (panel = this.getPanel(panel))){
52541             this.tabs.hideTab(panel.getEl().id);
52542         }
52543     },
52544
52545     /**
52546      * Unhides the tab for a previously hidden panel.
52547      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52548      */
52549     unhidePanel : function(panel){
52550         if(this.tabs && (panel = this.getPanel(panel))){
52551             this.tabs.unhideTab(panel.getEl().id);
52552         }
52553     },
52554
52555     clearPanels : function(){
52556         while(this.panels.getCount() > 0){
52557              this.remove(this.panels.first());
52558         }
52559     },
52560
52561     /**
52562      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52563      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
52564      * @param {Boolean} preservePanel Overrides the config preservePanel option
52565      * @return {Roo.ContentPanel} The panel that was removed
52566      */
52567     remove : function(panel, preservePanel){
52568         panel = this.getPanel(panel);
52569         if(!panel){
52570             return null;
52571         }
52572         var e = {};
52573         this.fireEvent("beforeremove", this, panel, e);
52574         if(e.cancel === true){
52575             return null;
52576         }
52577         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
52578         var panelId = panel.getId();
52579         this.panels.removeKey(panelId);
52580         if(preservePanel){
52581             document.body.appendChild(panel.getEl().dom);
52582         }
52583         if(this.tabs){
52584             this.tabs.removeTab(panel.getEl().id);
52585         }else if (!preservePanel){
52586             this.bodyEl.dom.removeChild(panel.getEl().dom);
52587         }
52588         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
52589             var p = this.panels.first();
52590             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
52591             tempEl.appendChild(p.getEl().dom);
52592             this.bodyEl.update("");
52593             this.bodyEl.dom.appendChild(p.getEl().dom);
52594             tempEl = null;
52595             this.updateTitle(p.getTitle());
52596             this.tabs = null;
52597             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
52598             this.setActivePanel(p);
52599         }
52600         panel.setRegion(null);
52601         if(this.activePanel == panel){
52602             this.activePanel = null;
52603         }
52604         if(this.config.autoDestroy !== false && preservePanel !== true){
52605             try{panel.destroy();}catch(e){}
52606         }
52607         this.fireEvent("panelremoved", this, panel);
52608         return panel;
52609     },
52610
52611     /**
52612      * Returns the TabPanel component used by this region
52613      * @return {Roo.TabPanel}
52614      */
52615     getTabs : function(){
52616         return this.tabs;
52617     },
52618
52619     createTool : function(parentEl, className){
52620         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
52621             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
52622         btn.addClassOnOver("x-layout-tools-button-over");
52623         return btn;
52624     }
52625 });/*
52626  * Based on:
52627  * Ext JS Library 1.1.1
52628  * Copyright(c) 2006-2007, Ext JS, LLC.
52629  *
52630  * Originally Released Under LGPL - original licence link has changed is not relivant.
52631  *
52632  * Fork - LGPL
52633  * <script type="text/javascript">
52634  */
52635  
52636
52637
52638 /**
52639  * @class Roo.SplitLayoutRegion
52640  * @extends Roo.LayoutRegion
52641  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
52642  */
52643 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
52644     this.cursor = cursor;
52645     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
52646 };
52647
52648 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
52649     splitTip : "Drag to resize.",
52650     collapsibleSplitTip : "Drag to resize. Double click to hide.",
52651     useSplitTips : false,
52652
52653     applyConfig : function(config){
52654         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
52655         if(config.split){
52656             if(!this.split){
52657                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
52658                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
52659                 /** The SplitBar for this region 
52660                 * @type Roo.SplitBar */
52661                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
52662                 this.split.on("moved", this.onSplitMove, this);
52663                 this.split.useShim = config.useShim === true;
52664                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
52665                 if(this.useSplitTips){
52666                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
52667                 }
52668                 if(config.collapsible){
52669                     this.split.el.on("dblclick", this.collapse,  this);
52670                 }
52671             }
52672             if(typeof config.minSize != "undefined"){
52673                 this.split.minSize = config.minSize;
52674             }
52675             if(typeof config.maxSize != "undefined"){
52676                 this.split.maxSize = config.maxSize;
52677             }
52678             if(config.hideWhenEmpty || config.hidden || config.collapsed){
52679                 this.hideSplitter();
52680             }
52681         }
52682     },
52683
52684     getHMaxSize : function(){
52685          var cmax = this.config.maxSize || 10000;
52686          var center = this.mgr.getRegion("center");
52687          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
52688     },
52689
52690     getVMaxSize : function(){
52691          var cmax = this.config.maxSize || 10000;
52692          var center = this.mgr.getRegion("center");
52693          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
52694     },
52695
52696     onSplitMove : function(split, newSize){
52697         this.fireEvent("resized", this, newSize);
52698     },
52699     
52700     /** 
52701      * Returns the {@link Roo.SplitBar} for this region.
52702      * @return {Roo.SplitBar}
52703      */
52704     getSplitBar : function(){
52705         return this.split;
52706     },
52707     
52708     hide : function(){
52709         this.hideSplitter();
52710         Roo.SplitLayoutRegion.superclass.hide.call(this);
52711     },
52712
52713     hideSplitter : function(){
52714         if(this.split){
52715             this.split.el.setLocation(-2000,-2000);
52716             this.split.el.hide();
52717         }
52718     },
52719
52720     show : function(){
52721         if(this.split){
52722             this.split.el.show();
52723         }
52724         Roo.SplitLayoutRegion.superclass.show.call(this);
52725     },
52726     
52727     beforeSlide: function(){
52728         if(Roo.isGecko){// firefox overflow auto bug workaround
52729             this.bodyEl.clip();
52730             if(this.tabs) {
52731                 this.tabs.bodyEl.clip();
52732             }
52733             if(this.activePanel){
52734                 this.activePanel.getEl().clip();
52735                 
52736                 if(this.activePanel.beforeSlide){
52737                     this.activePanel.beforeSlide();
52738                 }
52739             }
52740         }
52741     },
52742     
52743     afterSlide : function(){
52744         if(Roo.isGecko){// firefox overflow auto bug workaround
52745             this.bodyEl.unclip();
52746             if(this.tabs) {
52747                 this.tabs.bodyEl.unclip();
52748             }
52749             if(this.activePanel){
52750                 this.activePanel.getEl().unclip();
52751                 if(this.activePanel.afterSlide){
52752                     this.activePanel.afterSlide();
52753                 }
52754             }
52755         }
52756     },
52757
52758     initAutoHide : function(){
52759         if(this.autoHide !== false){
52760             if(!this.autoHideHd){
52761                 var st = new Roo.util.DelayedTask(this.slideIn, this);
52762                 this.autoHideHd = {
52763                     "mouseout": function(e){
52764                         if(!e.within(this.el, true)){
52765                             st.delay(500);
52766                         }
52767                     },
52768                     "mouseover" : function(e){
52769                         st.cancel();
52770                     },
52771                     scope : this
52772                 };
52773             }
52774             this.el.on(this.autoHideHd);
52775         }
52776     },
52777
52778     clearAutoHide : function(){
52779         if(this.autoHide !== false){
52780             this.el.un("mouseout", this.autoHideHd.mouseout);
52781             this.el.un("mouseover", this.autoHideHd.mouseover);
52782         }
52783     },
52784
52785     clearMonitor : function(){
52786         Roo.get(document).un("click", this.slideInIf, this);
52787     },
52788
52789     // these names are backwards but not changed for compat
52790     slideOut : function(){
52791         if(this.isSlid || this.el.hasActiveFx()){
52792             return;
52793         }
52794         this.isSlid = true;
52795         if(this.collapseBtn){
52796             this.collapseBtn.hide();
52797         }
52798         this.closeBtnState = this.closeBtn.getStyle('display');
52799         this.closeBtn.hide();
52800         if(this.stickBtn){
52801             this.stickBtn.show();
52802         }
52803         this.el.show();
52804         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
52805         this.beforeSlide();
52806         this.el.setStyle("z-index", 10001);
52807         this.el.slideIn(this.getSlideAnchor(), {
52808             callback: function(){
52809                 this.afterSlide();
52810                 this.initAutoHide();
52811                 Roo.get(document).on("click", this.slideInIf, this);
52812                 this.fireEvent("slideshow", this);
52813             },
52814             scope: this,
52815             block: true
52816         });
52817     },
52818
52819     afterSlideIn : function(){
52820         this.clearAutoHide();
52821         this.isSlid = false;
52822         this.clearMonitor();
52823         this.el.setStyle("z-index", "");
52824         if(this.collapseBtn){
52825             this.collapseBtn.show();
52826         }
52827         this.closeBtn.setStyle('display', this.closeBtnState);
52828         if(this.stickBtn){
52829             this.stickBtn.hide();
52830         }
52831         this.fireEvent("slidehide", this);
52832     },
52833
52834     slideIn : function(cb){
52835         if(!this.isSlid || this.el.hasActiveFx()){
52836             Roo.callback(cb);
52837             return;
52838         }
52839         this.isSlid = false;
52840         this.beforeSlide();
52841         this.el.slideOut(this.getSlideAnchor(), {
52842             callback: function(){
52843                 this.el.setLeftTop(-10000, -10000);
52844                 this.afterSlide();
52845                 this.afterSlideIn();
52846                 Roo.callback(cb);
52847             },
52848             scope: this,
52849             block: true
52850         });
52851     },
52852     
52853     slideInIf : function(e){
52854         if(!e.within(this.el)){
52855             this.slideIn();
52856         }
52857     },
52858
52859     animateCollapse : function(){
52860         this.beforeSlide();
52861         this.el.setStyle("z-index", 20000);
52862         var anchor = this.getSlideAnchor();
52863         this.el.slideOut(anchor, {
52864             callback : function(){
52865                 this.el.setStyle("z-index", "");
52866                 this.collapsedEl.slideIn(anchor, {duration:.3});
52867                 this.afterSlide();
52868                 this.el.setLocation(-10000,-10000);
52869                 this.el.hide();
52870                 this.fireEvent("collapsed", this);
52871             },
52872             scope: this,
52873             block: true
52874         });
52875     },
52876
52877     animateExpand : function(){
52878         this.beforeSlide();
52879         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
52880         this.el.setStyle("z-index", 20000);
52881         this.collapsedEl.hide({
52882             duration:.1
52883         });
52884         this.el.slideIn(this.getSlideAnchor(), {
52885             callback : function(){
52886                 this.el.setStyle("z-index", "");
52887                 this.afterSlide();
52888                 if(this.split){
52889                     this.split.el.show();
52890                 }
52891                 this.fireEvent("invalidated", this);
52892                 this.fireEvent("expanded", this);
52893             },
52894             scope: this,
52895             block: true
52896         });
52897     },
52898
52899     anchors : {
52900         "west" : "left",
52901         "east" : "right",
52902         "north" : "top",
52903         "south" : "bottom"
52904     },
52905
52906     sanchors : {
52907         "west" : "l",
52908         "east" : "r",
52909         "north" : "t",
52910         "south" : "b"
52911     },
52912
52913     canchors : {
52914         "west" : "tl-tr",
52915         "east" : "tr-tl",
52916         "north" : "tl-bl",
52917         "south" : "bl-tl"
52918     },
52919
52920     getAnchor : function(){
52921         return this.anchors[this.position];
52922     },
52923
52924     getCollapseAnchor : function(){
52925         return this.canchors[this.position];
52926     },
52927
52928     getSlideAnchor : function(){
52929         return this.sanchors[this.position];
52930     },
52931
52932     getAlignAdj : function(){
52933         var cm = this.cmargins;
52934         switch(this.position){
52935             case "west":
52936                 return [0, 0];
52937             break;
52938             case "east":
52939                 return [0, 0];
52940             break;
52941             case "north":
52942                 return [0, 0];
52943             break;
52944             case "south":
52945                 return [0, 0];
52946             break;
52947         }
52948     },
52949
52950     getExpandAdj : function(){
52951         var c = this.collapsedEl, cm = this.cmargins;
52952         switch(this.position){
52953             case "west":
52954                 return [-(cm.right+c.getWidth()+cm.left), 0];
52955             break;
52956             case "east":
52957                 return [cm.right+c.getWidth()+cm.left, 0];
52958             break;
52959             case "north":
52960                 return [0, -(cm.top+cm.bottom+c.getHeight())];
52961             break;
52962             case "south":
52963                 return [0, cm.top+cm.bottom+c.getHeight()];
52964             break;
52965         }
52966     }
52967 });/*
52968  * Based on:
52969  * Ext JS Library 1.1.1
52970  * Copyright(c) 2006-2007, Ext JS, LLC.
52971  *
52972  * Originally Released Under LGPL - original licence link has changed is not relivant.
52973  *
52974  * Fork - LGPL
52975  * <script type="text/javascript">
52976  */
52977 /*
52978  * These classes are private internal classes
52979  */
52980 Roo.CenterLayoutRegion = function(mgr, config){
52981     Roo.LayoutRegion.call(this, mgr, config, "center");
52982     this.visible = true;
52983     this.minWidth = config.minWidth || 20;
52984     this.minHeight = config.minHeight || 20;
52985 };
52986
52987 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
52988     hide : function(){
52989         // center panel can't be hidden
52990     },
52991     
52992     show : function(){
52993         // center panel can't be hidden
52994     },
52995     
52996     getMinWidth: function(){
52997         return this.minWidth;
52998     },
52999     
53000     getMinHeight: function(){
53001         return this.minHeight;
53002     }
53003 });
53004
53005
53006 Roo.NorthLayoutRegion = function(mgr, config){
53007     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53008     if(this.split){
53009         this.split.placement = Roo.SplitBar.TOP;
53010         this.split.orientation = Roo.SplitBar.VERTICAL;
53011         this.split.el.addClass("x-layout-split-v");
53012     }
53013     var size = config.initialSize || config.height;
53014     if(typeof size != "undefined"){
53015         this.el.setHeight(size);
53016     }
53017 };
53018 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53019     orientation: Roo.SplitBar.VERTICAL,
53020     getBox : function(){
53021         if(this.collapsed){
53022             return this.collapsedEl.getBox();
53023         }
53024         var box = this.el.getBox();
53025         if(this.split){
53026             box.height += this.split.el.getHeight();
53027         }
53028         return box;
53029     },
53030     
53031     updateBox : function(box){
53032         if(this.split && !this.collapsed){
53033             box.height -= this.split.el.getHeight();
53034             this.split.el.setLeft(box.x);
53035             this.split.el.setTop(box.y+box.height);
53036             this.split.el.setWidth(box.width);
53037         }
53038         if(this.collapsed){
53039             this.updateBody(box.width, null);
53040         }
53041         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53042     }
53043 });
53044
53045 Roo.SouthLayoutRegion = function(mgr, config){
53046     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53047     if(this.split){
53048         this.split.placement = Roo.SplitBar.BOTTOM;
53049         this.split.orientation = Roo.SplitBar.VERTICAL;
53050         this.split.el.addClass("x-layout-split-v");
53051     }
53052     var size = config.initialSize || config.height;
53053     if(typeof size != "undefined"){
53054         this.el.setHeight(size);
53055     }
53056 };
53057 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53058     orientation: Roo.SplitBar.VERTICAL,
53059     getBox : function(){
53060         if(this.collapsed){
53061             return this.collapsedEl.getBox();
53062         }
53063         var box = this.el.getBox();
53064         if(this.split){
53065             var sh = this.split.el.getHeight();
53066             box.height += sh;
53067             box.y -= sh;
53068         }
53069         return box;
53070     },
53071     
53072     updateBox : function(box){
53073         if(this.split && !this.collapsed){
53074             var sh = this.split.el.getHeight();
53075             box.height -= sh;
53076             box.y += sh;
53077             this.split.el.setLeft(box.x);
53078             this.split.el.setTop(box.y-sh);
53079             this.split.el.setWidth(box.width);
53080         }
53081         if(this.collapsed){
53082             this.updateBody(box.width, null);
53083         }
53084         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53085     }
53086 });
53087
53088 Roo.EastLayoutRegion = function(mgr, config){
53089     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53090     if(this.split){
53091         this.split.placement = Roo.SplitBar.RIGHT;
53092         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53093         this.split.el.addClass("x-layout-split-h");
53094     }
53095     var size = config.initialSize || config.width;
53096     if(typeof size != "undefined"){
53097         this.el.setWidth(size);
53098     }
53099 };
53100 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53101     orientation: Roo.SplitBar.HORIZONTAL,
53102     getBox : function(){
53103         if(this.collapsed){
53104             return this.collapsedEl.getBox();
53105         }
53106         var box = this.el.getBox();
53107         if(this.split){
53108             var sw = this.split.el.getWidth();
53109             box.width += sw;
53110             box.x -= sw;
53111         }
53112         return box;
53113     },
53114
53115     updateBox : function(box){
53116         if(this.split && !this.collapsed){
53117             var sw = this.split.el.getWidth();
53118             box.width -= sw;
53119             this.split.el.setLeft(box.x);
53120             this.split.el.setTop(box.y);
53121             this.split.el.setHeight(box.height);
53122             box.x += sw;
53123         }
53124         if(this.collapsed){
53125             this.updateBody(null, box.height);
53126         }
53127         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53128     }
53129 });
53130
53131 Roo.WestLayoutRegion = function(mgr, config){
53132     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53133     if(this.split){
53134         this.split.placement = Roo.SplitBar.LEFT;
53135         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53136         this.split.el.addClass("x-layout-split-h");
53137     }
53138     var size = config.initialSize || config.width;
53139     if(typeof size != "undefined"){
53140         this.el.setWidth(size);
53141     }
53142 };
53143 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53144     orientation: Roo.SplitBar.HORIZONTAL,
53145     getBox : function(){
53146         if(this.collapsed){
53147             return this.collapsedEl.getBox();
53148         }
53149         var box = this.el.getBox();
53150         if(this.split){
53151             box.width += this.split.el.getWidth();
53152         }
53153         return box;
53154     },
53155     
53156     updateBox : function(box){
53157         if(this.split && !this.collapsed){
53158             var sw = this.split.el.getWidth();
53159             box.width -= sw;
53160             this.split.el.setLeft(box.x+box.width);
53161             this.split.el.setTop(box.y);
53162             this.split.el.setHeight(box.height);
53163         }
53164         if(this.collapsed){
53165             this.updateBody(null, box.height);
53166         }
53167         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53168     }
53169 });
53170 /*
53171  * Based on:
53172  * Ext JS Library 1.1.1
53173  * Copyright(c) 2006-2007, Ext JS, LLC.
53174  *
53175  * Originally Released Under LGPL - original licence link has changed is not relivant.
53176  *
53177  * Fork - LGPL
53178  * <script type="text/javascript">
53179  */
53180  
53181  
53182 /*
53183  * Private internal class for reading and applying state
53184  */
53185 Roo.LayoutStateManager = function(layout){
53186      // default empty state
53187      this.state = {
53188         north: {},
53189         south: {},
53190         east: {},
53191         west: {}       
53192     };
53193 };
53194
53195 Roo.LayoutStateManager.prototype = {
53196     init : function(layout, provider){
53197         this.provider = provider;
53198         var state = provider.get(layout.id+"-layout-state");
53199         if(state){
53200             var wasUpdating = layout.isUpdating();
53201             if(!wasUpdating){
53202                 layout.beginUpdate();
53203             }
53204             for(var key in state){
53205                 if(typeof state[key] != "function"){
53206                     var rstate = state[key];
53207                     var r = layout.getRegion(key);
53208                     if(r && rstate){
53209                         if(rstate.size){
53210                             r.resizeTo(rstate.size);
53211                         }
53212                         if(rstate.collapsed == true){
53213                             r.collapse(true);
53214                         }else{
53215                             r.expand(null, true);
53216                         }
53217                     }
53218                 }
53219             }
53220             if(!wasUpdating){
53221                 layout.endUpdate();
53222             }
53223             this.state = state; 
53224         }
53225         this.layout = layout;
53226         layout.on("regionresized", this.onRegionResized, this);
53227         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53228         layout.on("regionexpanded", this.onRegionExpanded, this);
53229     },
53230     
53231     storeState : function(){
53232         this.provider.set(this.layout.id+"-layout-state", this.state);
53233     },
53234     
53235     onRegionResized : function(region, newSize){
53236         this.state[region.getPosition()].size = newSize;
53237         this.storeState();
53238     },
53239     
53240     onRegionCollapsed : function(region){
53241         this.state[region.getPosition()].collapsed = true;
53242         this.storeState();
53243     },
53244     
53245     onRegionExpanded : function(region){
53246         this.state[region.getPosition()].collapsed = false;
53247         this.storeState();
53248     }
53249 };/*
53250  * Based on:
53251  * Ext JS Library 1.1.1
53252  * Copyright(c) 2006-2007, Ext JS, LLC.
53253  *
53254  * Originally Released Under LGPL - original licence link has changed is not relivant.
53255  *
53256  * Fork - LGPL
53257  * <script type="text/javascript">
53258  */
53259 /**
53260  * @class Roo.ContentPanel
53261  * @extends Roo.util.Observable
53262  * A basic ContentPanel element.
53263  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53264  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53265  * @cfg {Boolean/Object} autoCreate True to auto generate the DOM element for this panel, or a {@link Roo.DomHelper} config of the element to create
53266  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53267  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53268  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53269  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53270  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53271  * @cfg {String} title          The title for this panel
53272  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53273  * @cfg {String} url            Calls {@link #setUrl} with this value
53274  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
53275  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
53276  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
53277  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
53278
53279  * @constructor
53280  * Create a new ContentPanel.
53281  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
53282  * @param {String/Object} config A string to set only the title or a config object
53283  * @param {String} content (optional) Set the HTML content for this panel
53284  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
53285  */
53286 Roo.ContentPanel = function(el, config, content){
53287     
53288      
53289     /*
53290     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
53291         config = el;
53292         el = Roo.id();
53293     }
53294     if (config && config.parentLayout) { 
53295         el = config.parentLayout.el.createChild(); 
53296     }
53297     */
53298     if(el.autoCreate){ // xtype is available if this is called from factory
53299         config = el;
53300         el = Roo.id();
53301     }
53302     this.el = Roo.get(el);
53303     if(!this.el && config && config.autoCreate){
53304         if(typeof config.autoCreate == "object"){
53305             if(!config.autoCreate.id){
53306                 config.autoCreate.id = config.id||el;
53307             }
53308             this.el = Roo.DomHelper.append(document.body,
53309                         config.autoCreate, true);
53310         }else{
53311             this.el = Roo.DomHelper.append(document.body,
53312                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
53313         }
53314     }
53315     this.closable = false;
53316     this.loaded = false;
53317     this.active = false;
53318     if(typeof config == "string"){
53319         this.title = config;
53320     }else{
53321         Roo.apply(this, config);
53322     }
53323     
53324     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
53325         this.wrapEl = this.el.wrap();
53326         this.toolbar.container = this.el.insertSibling(false, 'before');
53327         this.toolbar = new Roo.Toolbar(this.toolbar);
53328     }
53329     
53330     // xtype created footer. - not sure if will work as we normally have to render first..
53331     if (this.footer && !this.footer.el && this.footer.xtype) {
53332         if (!this.wrapEl) {
53333             this.wrapEl = this.el.wrap();
53334         }
53335     
53336         this.footer.container = this.wrapEl.createChild();
53337          
53338         this.footer = Roo.factory(this.footer, Roo);
53339         
53340     }
53341     
53342     if(this.resizeEl){
53343         this.resizeEl = Roo.get(this.resizeEl, true);
53344     }else{
53345         this.resizeEl = this.el;
53346     }
53347     // handle view.xtype
53348     
53349  
53350     
53351     
53352     this.addEvents({
53353         /**
53354          * @event activate
53355          * Fires when this panel is activated. 
53356          * @param {Roo.ContentPanel} this
53357          */
53358         "activate" : true,
53359         /**
53360          * @event deactivate
53361          * Fires when this panel is activated. 
53362          * @param {Roo.ContentPanel} this
53363          */
53364         "deactivate" : true,
53365
53366         /**
53367          * @event resize
53368          * Fires when this panel is resized if fitToFrame is true.
53369          * @param {Roo.ContentPanel} this
53370          * @param {Number} width The width after any component adjustments
53371          * @param {Number} height The height after any component adjustments
53372          */
53373         "resize" : true,
53374         
53375          /**
53376          * @event render
53377          * Fires when this tab is created
53378          * @param {Roo.ContentPanel} this
53379          */
53380         "render" : true
53381          
53382         
53383     });
53384     
53385
53386     
53387     
53388     if(this.autoScroll){
53389         this.resizeEl.setStyle("overflow", "auto");
53390     } else {
53391         // fix randome scrolling
53392         this.el.on('scroll', function() {
53393             Roo.log('fix random scolling');
53394             this.scrollTo('top',0); 
53395         });
53396     }
53397     content = content || this.content;
53398     if(content){
53399         this.setContent(content);
53400     }
53401     if(config && config.url){
53402         this.setUrl(this.url, this.params, this.loadOnce);
53403     }
53404     
53405     
53406     
53407     Roo.ContentPanel.superclass.constructor.call(this);
53408     
53409     if (this.view && typeof(this.view.xtype) != 'undefined') {
53410         this.view.el = this.el.appendChild(document.createElement("div"));
53411         this.view = Roo.factory(this.view); 
53412         this.view.render  &&  this.view.render(false, '');  
53413     }
53414     
53415     
53416     this.fireEvent('render', this);
53417 };
53418
53419 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
53420     tabTip:'',
53421     setRegion : function(region){
53422         this.region = region;
53423         if(region){
53424            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
53425         }else{
53426            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
53427         } 
53428     },
53429     
53430     /**
53431      * Returns the toolbar for this Panel if one was configured. 
53432      * @return {Roo.Toolbar} 
53433      */
53434     getToolbar : function(){
53435         return this.toolbar;
53436     },
53437     
53438     setActiveState : function(active){
53439         this.active = active;
53440         if(!active){
53441             this.fireEvent("deactivate", this);
53442         }else{
53443             this.fireEvent("activate", this);
53444         }
53445     },
53446     /**
53447      * Updates this panel's element
53448      * @param {String} content The new content
53449      * @param {Boolean} loadScripts (optional) true to look for and process scripts
53450     */
53451     setContent : function(content, loadScripts){
53452         this.el.update(content, loadScripts);
53453     },
53454
53455     ignoreResize : function(w, h){
53456         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
53457             return true;
53458         }else{
53459             this.lastSize = {width: w, height: h};
53460             return false;
53461         }
53462     },
53463     /**
53464      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
53465      * @return {Roo.UpdateManager} The UpdateManager
53466      */
53467     getUpdateManager : function(){
53468         return this.el.getUpdateManager();
53469     },
53470      /**
53471      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
53472      * @param {Object/String/Function} url The url for this request or a function to call to get the url or a config object containing any of the following options:
53473 <pre><code>
53474 panel.load({
53475     url: "your-url.php",
53476     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
53477     callback: yourFunction,
53478     scope: yourObject, //(optional scope)
53479     discardUrl: false,
53480     nocache: false,
53481     text: "Loading...",
53482     timeout: 30,
53483     scripts: false
53484 });
53485 </code></pre>
53486      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
53487      * are shorthand for <i>disableCaching</i>, <i>indicatorText</i> and <i>loadScripts</i> and are used to set their associated property on this panel UpdateManager instance.
53488      * @param {String/Object} params (optional) The parameters to pass as either a URL encoded string "param1=1&amp;param2=2" or an object {param1: 1, param2: 2}
53489      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
53490      * @param {Boolean} discardUrl (optional) By default when you execute an update the defaultUrl is changed to the last used URL. If true, it will not store the URL.
53491      * @return {Roo.ContentPanel} this
53492      */
53493     load : function(){
53494         var um = this.el.getUpdateManager();
53495         um.update.apply(um, arguments);
53496         return this;
53497     },
53498
53499
53500     /**
53501      * Set a URL to be used to load the content for this panel. When this panel is activated, the content will be loaded from that URL.
53502      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
53503      * @param {String/Object} params (optional) The string params for the update call or an object of the params. See {@link Roo.UpdateManager#update} for more details. (Defaults to null)
53504      * @param {Boolean} loadOnce (optional) Whether to only load the content once. If this is false it makes the Ajax call every time this panel is activated. (Defaults to false)
53505      * @return {Roo.UpdateManager} The UpdateManager
53506      */
53507     setUrl : function(url, params, loadOnce){
53508         if(this.refreshDelegate){
53509             this.removeListener("activate", this.refreshDelegate);
53510         }
53511         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
53512         this.on("activate", this.refreshDelegate);
53513         return this.el.getUpdateManager();
53514     },
53515     
53516     _handleRefresh : function(url, params, loadOnce){
53517         if(!loadOnce || !this.loaded){
53518             var updater = this.el.getUpdateManager();
53519             updater.update(url, params, this._setLoaded.createDelegate(this));
53520         }
53521     },
53522     
53523     _setLoaded : function(){
53524         this.loaded = true;
53525     }, 
53526     
53527     /**
53528      * Returns this panel's id
53529      * @return {String} 
53530      */
53531     getId : function(){
53532         return this.el.id;
53533     },
53534     
53535     /** 
53536      * Returns this panel's element - used by regiosn to add.
53537      * @return {Roo.Element} 
53538      */
53539     getEl : function(){
53540         return this.wrapEl || this.el;
53541     },
53542     
53543     adjustForComponents : function(width, height)
53544     {
53545         //Roo.log('adjustForComponents ');
53546         if(this.resizeEl != this.el){
53547             width -= this.el.getFrameWidth('lr');
53548             height -= this.el.getFrameWidth('tb');
53549         }
53550         if(this.toolbar){
53551             var te = this.toolbar.getEl();
53552             height -= te.getHeight();
53553             te.setWidth(width);
53554         }
53555         if(this.footer){
53556             var te = this.footer.getEl();
53557             //Roo.log("footer:" + te.getHeight());
53558             
53559             height -= te.getHeight();
53560             te.setWidth(width);
53561         }
53562         
53563         
53564         if(this.adjustments){
53565             width += this.adjustments[0];
53566             height += this.adjustments[1];
53567         }
53568         return {"width": width, "height": height};
53569     },
53570     
53571     setSize : function(width, height){
53572         if(this.fitToFrame && !this.ignoreResize(width, height)){
53573             if(this.fitContainer && this.resizeEl != this.el){
53574                 this.el.setSize(width, height);
53575             }
53576             var size = this.adjustForComponents(width, height);
53577             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
53578             this.fireEvent('resize', this, size.width, size.height);
53579         }
53580     },
53581     
53582     /**
53583      * Returns this panel's title
53584      * @return {String} 
53585      */
53586     getTitle : function(){
53587         return this.title;
53588     },
53589     
53590     /**
53591      * Set this panel's title
53592      * @param {String} title
53593      */
53594     setTitle : function(title){
53595         this.title = title;
53596         if(this.region){
53597             this.region.updatePanelTitle(this, title);
53598         }
53599     },
53600     
53601     /**
53602      * Returns true is this panel was configured to be closable
53603      * @return {Boolean} 
53604      */
53605     isClosable : function(){
53606         return this.closable;
53607     },
53608     
53609     beforeSlide : function(){
53610         this.el.clip();
53611         this.resizeEl.clip();
53612     },
53613     
53614     afterSlide : function(){
53615         this.el.unclip();
53616         this.resizeEl.unclip();
53617     },
53618     
53619     /**
53620      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
53621      *   Will fail silently if the {@link #setUrl} method has not been called.
53622      *   This does not activate the panel, just updates its content.
53623      */
53624     refresh : function(){
53625         if(this.refreshDelegate){
53626            this.loaded = false;
53627            this.refreshDelegate();
53628         }
53629     },
53630     
53631     /**
53632      * Destroys this panel
53633      */
53634     destroy : function(){
53635         this.el.removeAllListeners();
53636         var tempEl = document.createElement("span");
53637         tempEl.appendChild(this.el.dom);
53638         tempEl.innerHTML = "";
53639         this.el.remove();
53640         this.el = null;
53641     },
53642     
53643     /**
53644      * form - if the content panel contains a form - this is a reference to it.
53645      * @type {Roo.form.Form}
53646      */
53647     form : false,
53648     /**
53649      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
53650      *    This contains a reference to it.
53651      * @type {Roo.View}
53652      */
53653     view : false,
53654     
53655       /**
53656      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
53657      * <pre><code>
53658
53659 layout.addxtype({
53660        xtype : 'Form',
53661        items: [ .... ]
53662    }
53663 );
53664
53665 </code></pre>
53666      * @param {Object} cfg Xtype definition of item to add.
53667      */
53668     
53669     addxtype : function(cfg) {
53670         // add form..
53671         if (cfg.xtype.match(/^Form$/)) {
53672             
53673             var el;
53674             //if (this.footer) {
53675             //    el = this.footer.container.insertSibling(false, 'before');
53676             //} else {
53677                 el = this.el.createChild();
53678             //}
53679
53680             this.form = new  Roo.form.Form(cfg);
53681             
53682             
53683             if ( this.form.allItems.length) {
53684                 this.form.render(el.dom);
53685             }
53686             return this.form;
53687         }
53688         // should only have one of theses..
53689         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
53690             // views.. should not be just added - used named prop 'view''
53691             
53692             cfg.el = this.el.appendChild(document.createElement("div"));
53693             // factory?
53694             
53695             var ret = new Roo.factory(cfg);
53696              
53697              ret.render && ret.render(false, ''); // render blank..
53698             this.view = ret;
53699             return ret;
53700         }
53701         return false;
53702     }
53703 });
53704
53705 /**
53706  * @class Roo.GridPanel
53707  * @extends Roo.ContentPanel
53708  * @constructor
53709  * Create a new GridPanel.
53710  * @param {Roo.grid.Grid} grid The grid for this panel
53711  * @param {String/Object} config A string to set only the panel's title, or a config object
53712  */
53713 Roo.GridPanel = function(grid, config){
53714     
53715   
53716     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
53717         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
53718         
53719     this.wrapper.dom.appendChild(grid.getGridEl().dom);
53720     
53721     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
53722     
53723     if(this.toolbar){
53724         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
53725     }
53726     // xtype created footer. - not sure if will work as we normally have to render first..
53727     if (this.footer && !this.footer.el && this.footer.xtype) {
53728         
53729         this.footer.container = this.grid.getView().getFooterPanel(true);
53730         this.footer.dataSource = this.grid.dataSource;
53731         this.footer = Roo.factory(this.footer, Roo);
53732         
53733     }
53734     
53735     grid.monitorWindowResize = false; // turn off autosizing
53736     grid.autoHeight = false;
53737     grid.autoWidth = false;
53738     this.grid = grid;
53739     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
53740 };
53741
53742 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
53743     getId : function(){
53744         return this.grid.id;
53745     },
53746     
53747     /**
53748      * Returns the grid for this panel
53749      * @return {Roo.grid.Grid} 
53750      */
53751     getGrid : function(){
53752         return this.grid;    
53753     },
53754     
53755     setSize : function(width, height){
53756         if(!this.ignoreResize(width, height)){
53757             var grid = this.grid;
53758             var size = this.adjustForComponents(width, height);
53759             grid.getGridEl().setSize(size.width, size.height);
53760             grid.autoSize();
53761         }
53762     },
53763     
53764     beforeSlide : function(){
53765         this.grid.getView().scroller.clip();
53766     },
53767     
53768     afterSlide : function(){
53769         this.grid.getView().scroller.unclip();
53770     },
53771     
53772     destroy : function(){
53773         this.grid.destroy();
53774         delete this.grid;
53775         Roo.GridPanel.superclass.destroy.call(this); 
53776     }
53777 });
53778
53779
53780 /**
53781  * @class Roo.NestedLayoutPanel
53782  * @extends Roo.ContentPanel
53783  * @constructor
53784  * Create a new NestedLayoutPanel.
53785  * 
53786  * 
53787  * @param {Roo.BorderLayout} layout The layout for this panel
53788  * @param {String/Object} config A string to set only the title or a config object
53789  */
53790 Roo.NestedLayoutPanel = function(layout, config)
53791 {
53792     // construct with only one argument..
53793     /* FIXME - implement nicer consturctors
53794     if (layout.layout) {
53795         config = layout;
53796         layout = config.layout;
53797         delete config.layout;
53798     }
53799     if (layout.xtype && !layout.getEl) {
53800         // then layout needs constructing..
53801         layout = Roo.factory(layout, Roo);
53802     }
53803     */
53804     
53805     
53806     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
53807     
53808     layout.monitorWindowResize = false; // turn off autosizing
53809     this.layout = layout;
53810     this.layout.getEl().addClass("x-layout-nested-layout");
53811     
53812     
53813     
53814     
53815 };
53816
53817 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
53818
53819     setSize : function(width, height){
53820         if(!this.ignoreResize(width, height)){
53821             var size = this.adjustForComponents(width, height);
53822             var el = this.layout.getEl();
53823             el.setSize(size.width, size.height);
53824             var touch = el.dom.offsetWidth;
53825             this.layout.layout();
53826             // ie requires a double layout on the first pass
53827             if(Roo.isIE && !this.initialized){
53828                 this.initialized = true;
53829                 this.layout.layout();
53830             }
53831         }
53832     },
53833     
53834     // activate all subpanels if not currently active..
53835     
53836     setActiveState : function(active){
53837         this.active = active;
53838         if(!active){
53839             this.fireEvent("deactivate", this);
53840             return;
53841         }
53842         
53843         this.fireEvent("activate", this);
53844         // not sure if this should happen before or after..
53845         if (!this.layout) {
53846             return; // should not happen..
53847         }
53848         var reg = false;
53849         for (var r in this.layout.regions) {
53850             reg = this.layout.getRegion(r);
53851             if (reg.getActivePanel()) {
53852                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
53853                 reg.setActivePanel(reg.getActivePanel());
53854                 continue;
53855             }
53856             if (!reg.panels.length) {
53857                 continue;
53858             }
53859             reg.showPanel(reg.getPanel(0));
53860         }
53861         
53862         
53863         
53864         
53865     },
53866     
53867     /**
53868      * Returns the nested BorderLayout for this panel
53869      * @return {Roo.BorderLayout} 
53870      */
53871     getLayout : function(){
53872         return this.layout;
53873     },
53874     
53875      /**
53876      * Adds a xtype elements to the layout of the nested panel
53877      * <pre><code>
53878
53879 panel.addxtype({
53880        xtype : 'ContentPanel',
53881        region: 'west',
53882        items: [ .... ]
53883    }
53884 );
53885
53886 panel.addxtype({
53887         xtype : 'NestedLayoutPanel',
53888         region: 'west',
53889         layout: {
53890            center: { },
53891            west: { }   
53892         },
53893         items : [ ... list of content panels or nested layout panels.. ]
53894    }
53895 );
53896 </code></pre>
53897      * @param {Object} cfg Xtype definition of item to add.
53898      */
53899     addxtype : function(cfg) {
53900         return this.layout.addxtype(cfg);
53901     
53902     }
53903 });
53904
53905 Roo.ScrollPanel = function(el, config, content){
53906     config = config || {};
53907     config.fitToFrame = true;
53908     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
53909     
53910     this.el.dom.style.overflow = "hidden";
53911     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
53912     this.el.removeClass("x-layout-inactive-content");
53913     this.el.on("mousewheel", this.onWheel, this);
53914
53915     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
53916     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
53917     up.unselectable(); down.unselectable();
53918     up.on("click", this.scrollUp, this);
53919     down.on("click", this.scrollDown, this);
53920     up.addClassOnOver("x-scroller-btn-over");
53921     down.addClassOnOver("x-scroller-btn-over");
53922     up.addClassOnClick("x-scroller-btn-click");
53923     down.addClassOnClick("x-scroller-btn-click");
53924     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
53925
53926     this.resizeEl = this.el;
53927     this.el = wrap; this.up = up; this.down = down;
53928 };
53929
53930 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
53931     increment : 100,
53932     wheelIncrement : 5,
53933     scrollUp : function(){
53934         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
53935     },
53936
53937     scrollDown : function(){
53938         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
53939     },
53940
53941     afterScroll : function(){
53942         var el = this.resizeEl;
53943         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
53944         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
53945         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
53946     },
53947
53948     setSize : function(){
53949         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
53950         this.afterScroll();
53951     },
53952
53953     onWheel : function(e){
53954         var d = e.getWheelDelta();
53955         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
53956         this.afterScroll();
53957         e.stopEvent();
53958     },
53959
53960     setContent : function(content, loadScripts){
53961         this.resizeEl.update(content, loadScripts);
53962     }
53963
53964 });
53965
53966
53967
53968
53969
53970
53971
53972
53973
53974 /**
53975  * @class Roo.TreePanel
53976  * @extends Roo.ContentPanel
53977  * @constructor
53978  * Create a new TreePanel. - defaults to fit/scoll contents.
53979  * @param {String/Object} config A string to set only the panel's title, or a config object
53980  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
53981  */
53982 Roo.TreePanel = function(config){
53983     var el = config.el;
53984     var tree = config.tree;
53985     delete config.tree; 
53986     delete config.el; // hopefull!
53987     
53988     // wrapper for IE7 strict & safari scroll issue
53989     
53990     var treeEl = el.createChild();
53991     config.resizeEl = treeEl;
53992     
53993     
53994     
53995     Roo.TreePanel.superclass.constructor.call(this, el, config);
53996  
53997  
53998     this.tree = new Roo.tree.TreePanel(treeEl , tree);
53999     //console.log(tree);
54000     this.on('activate', function()
54001     {
54002         if (this.tree.rendered) {
54003             return;
54004         }
54005         //console.log('render tree');
54006         this.tree.render();
54007     });
54008     // this should not be needed.. - it's actually the 'el' that resizes?
54009     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54010     
54011     //this.on('resize',  function (cp, w, h) {
54012     //        this.tree.innerCt.setWidth(w);
54013     //        this.tree.innerCt.setHeight(h);
54014     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54015     //});
54016
54017         
54018     
54019 };
54020
54021 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54022     fitToFrame : true,
54023     autoScroll : true
54024 });
54025
54026
54027
54028
54029
54030
54031
54032
54033
54034
54035
54036 /*
54037  * Based on:
54038  * Ext JS Library 1.1.1
54039  * Copyright(c) 2006-2007, Ext JS, LLC.
54040  *
54041  * Originally Released Under LGPL - original licence link has changed is not relivant.
54042  *
54043  * Fork - LGPL
54044  * <script type="text/javascript">
54045  */
54046  
54047
54048 /**
54049  * @class Roo.ReaderLayout
54050  * @extends Roo.BorderLayout
54051  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54052  * center region containing two nested regions (a top one for a list view and one for item preview below),
54053  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54054  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54055  * expedites the setup of the overall layout and regions for this common application style.
54056  * Example:
54057  <pre><code>
54058 var reader = new Roo.ReaderLayout();
54059 var CP = Roo.ContentPanel;  // shortcut for adding
54060
54061 reader.beginUpdate();
54062 reader.add("north", new CP("north", "North"));
54063 reader.add("west", new CP("west", {title: "West"}));
54064 reader.add("east", new CP("east", {title: "East"}));
54065
54066 reader.regions.listView.add(new CP("listView", "List"));
54067 reader.regions.preview.add(new CP("preview", "Preview"));
54068 reader.endUpdate();
54069 </code></pre>
54070 * @constructor
54071 * Create a new ReaderLayout
54072 * @param {Object} config Configuration options
54073 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54074 * document.body if omitted)
54075 */
54076 Roo.ReaderLayout = function(config, renderTo){
54077     var c = config || {size:{}};
54078     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54079         north: c.north !== false ? Roo.apply({
54080             split:false,
54081             initialSize: 32,
54082             titlebar: false
54083         }, c.north) : false,
54084         west: c.west !== false ? Roo.apply({
54085             split:true,
54086             initialSize: 200,
54087             minSize: 175,
54088             maxSize: 400,
54089             titlebar: true,
54090             collapsible: true,
54091             animate: true,
54092             margins:{left:5,right:0,bottom:5,top:5},
54093             cmargins:{left:5,right:5,bottom:5,top:5}
54094         }, c.west) : false,
54095         east: c.east !== false ? Roo.apply({
54096             split:true,
54097             initialSize: 200,
54098             minSize: 175,
54099             maxSize: 400,
54100             titlebar: true,
54101             collapsible: true,
54102             animate: true,
54103             margins:{left:0,right:5,bottom:5,top:5},
54104             cmargins:{left:5,right:5,bottom:5,top:5}
54105         }, c.east) : false,
54106         center: Roo.apply({
54107             tabPosition: 'top',
54108             autoScroll:false,
54109             closeOnTab: true,
54110             titlebar:false,
54111             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54112         }, c.center)
54113     });
54114
54115     this.el.addClass('x-reader');
54116
54117     this.beginUpdate();
54118
54119     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54120         south: c.preview !== false ? Roo.apply({
54121             split:true,
54122             initialSize: 200,
54123             minSize: 100,
54124             autoScroll:true,
54125             collapsible:true,
54126             titlebar: true,
54127             cmargins:{top:5,left:0, right:0, bottom:0}
54128         }, c.preview) : false,
54129         center: Roo.apply({
54130             autoScroll:false,
54131             titlebar:false,
54132             minHeight:200
54133         }, c.listView)
54134     });
54135     this.add('center', new Roo.NestedLayoutPanel(inner,
54136             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54137
54138     this.endUpdate();
54139
54140     this.regions.preview = inner.getRegion('south');
54141     this.regions.listView = inner.getRegion('center');
54142 };
54143
54144 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54145  * Based on:
54146  * Ext JS Library 1.1.1
54147  * Copyright(c) 2006-2007, Ext JS, LLC.
54148  *
54149  * Originally Released Under LGPL - original licence link has changed is not relivant.
54150  *
54151  * Fork - LGPL
54152  * <script type="text/javascript">
54153  */
54154  
54155 /**
54156  * @class Roo.grid.Grid
54157  * @extends Roo.util.Observable
54158  * This class represents the primary interface of a component based grid control.
54159  * <br><br>Usage:<pre><code>
54160  var grid = new Roo.grid.Grid("my-container-id", {
54161      ds: myDataStore,
54162      cm: myColModel,
54163      selModel: mySelectionModel,
54164      autoSizeColumns: true,
54165      monitorWindowResize: false,
54166      trackMouseOver: true
54167  });
54168  // set any options
54169  grid.render();
54170  * </code></pre>
54171  * <b>Common Problems:</b><br/>
54172  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54173  * element will correct this<br/>
54174  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54175  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54176  * are unpredictable.<br/>
54177  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54178  * grid to calculate dimensions/offsets.<br/>
54179   * @constructor
54180  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54181  * The container MUST have some type of size defined for the grid to fill. The container will be
54182  * automatically set to position relative if it isn't already.
54183  * @param {Object} config A config object that sets properties on this grid.
54184  */
54185 Roo.grid.Grid = function(container, config){
54186         // initialize the container
54187         this.container = Roo.get(container);
54188         this.container.update("");
54189         this.container.setStyle("overflow", "hidden");
54190     this.container.addClass('x-grid-container');
54191
54192     this.id = this.container.id;
54193
54194     Roo.apply(this, config);
54195     // check and correct shorthanded configs
54196     if(this.ds){
54197         this.dataSource = this.ds;
54198         delete this.ds;
54199     }
54200     if(this.cm){
54201         this.colModel = this.cm;
54202         delete this.cm;
54203     }
54204     if(this.sm){
54205         this.selModel = this.sm;
54206         delete this.sm;
54207     }
54208
54209     if (this.selModel) {
54210         this.selModel = Roo.factory(this.selModel, Roo.grid);
54211         this.sm = this.selModel;
54212         this.sm.xmodule = this.xmodule || false;
54213     }
54214     if (typeof(this.colModel.config) == 'undefined') {
54215         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54216         this.cm = this.colModel;
54217         this.cm.xmodule = this.xmodule || false;
54218     }
54219     if (this.dataSource) {
54220         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54221         this.ds = this.dataSource;
54222         this.ds.xmodule = this.xmodule || false;
54223          
54224     }
54225     
54226     
54227     
54228     if(this.width){
54229         this.container.setWidth(this.width);
54230     }
54231
54232     if(this.height){
54233         this.container.setHeight(this.height);
54234     }
54235     /** @private */
54236         this.addEvents({
54237         // raw events
54238         /**
54239          * @event click
54240          * The raw click event for the entire grid.
54241          * @param {Roo.EventObject} e
54242          */
54243         "click" : true,
54244         /**
54245          * @event dblclick
54246          * The raw dblclick event for the entire grid.
54247          * @param {Roo.EventObject} e
54248          */
54249         "dblclick" : true,
54250         /**
54251          * @event contextmenu
54252          * The raw contextmenu event for the entire grid.
54253          * @param {Roo.EventObject} e
54254          */
54255         "contextmenu" : true,
54256         /**
54257          * @event mousedown
54258          * The raw mousedown event for the entire grid.
54259          * @param {Roo.EventObject} e
54260          */
54261         "mousedown" : true,
54262         /**
54263          * @event mouseup
54264          * The raw mouseup event for the entire grid.
54265          * @param {Roo.EventObject} e
54266          */
54267         "mouseup" : true,
54268         /**
54269          * @event mouseover
54270          * The raw mouseover event for the entire grid.
54271          * @param {Roo.EventObject} e
54272          */
54273         "mouseover" : true,
54274         /**
54275          * @event mouseout
54276          * The raw mouseout event for the entire grid.
54277          * @param {Roo.EventObject} e
54278          */
54279         "mouseout" : true,
54280         /**
54281          * @event keypress
54282          * The raw keypress event for the entire grid.
54283          * @param {Roo.EventObject} e
54284          */
54285         "keypress" : true,
54286         /**
54287          * @event keydown
54288          * The raw keydown event for the entire grid.
54289          * @param {Roo.EventObject} e
54290          */
54291         "keydown" : true,
54292
54293         // custom events
54294
54295         /**
54296          * @event cellclick
54297          * Fires when a cell is clicked
54298          * @param {Grid} this
54299          * @param {Number} rowIndex
54300          * @param {Number} columnIndex
54301          * @param {Roo.EventObject} e
54302          */
54303         "cellclick" : true,
54304         /**
54305          * @event celldblclick
54306          * Fires when a cell is double clicked
54307          * @param {Grid} this
54308          * @param {Number} rowIndex
54309          * @param {Number} columnIndex
54310          * @param {Roo.EventObject} e
54311          */
54312         "celldblclick" : true,
54313         /**
54314          * @event rowclick
54315          * Fires when a row is clicked
54316          * @param {Grid} this
54317          * @param {Number} rowIndex
54318          * @param {Roo.EventObject} e
54319          */
54320         "rowclick" : true,
54321         /**
54322          * @event rowdblclick
54323          * Fires when a row is double clicked
54324          * @param {Grid} this
54325          * @param {Number} rowIndex
54326          * @param {Roo.EventObject} e
54327          */
54328         "rowdblclick" : true,
54329         /**
54330          * @event headerclick
54331          * Fires when a header is clicked
54332          * @param {Grid} this
54333          * @param {Number} columnIndex
54334          * @param {Roo.EventObject} e
54335          */
54336         "headerclick" : true,
54337         /**
54338          * @event headerdblclick
54339          * Fires when a header cell is double clicked
54340          * @param {Grid} this
54341          * @param {Number} columnIndex
54342          * @param {Roo.EventObject} e
54343          */
54344         "headerdblclick" : true,
54345         /**
54346          * @event rowcontextmenu
54347          * Fires when a row is right clicked
54348          * @param {Grid} this
54349          * @param {Number} rowIndex
54350          * @param {Roo.EventObject} e
54351          */
54352         "rowcontextmenu" : true,
54353         /**
54354          * @event cellcontextmenu
54355          * Fires when a cell is right clicked
54356          * @param {Grid} this
54357          * @param {Number} rowIndex
54358          * @param {Number} cellIndex
54359          * @param {Roo.EventObject} e
54360          */
54361          "cellcontextmenu" : true,
54362         /**
54363          * @event headercontextmenu
54364          * Fires when a header is right clicked
54365          * @param {Grid} this
54366          * @param {Number} columnIndex
54367          * @param {Roo.EventObject} e
54368          */
54369         "headercontextmenu" : true,
54370         /**
54371          * @event bodyscroll
54372          * Fires when the body element is scrolled
54373          * @param {Number} scrollLeft
54374          * @param {Number} scrollTop
54375          */
54376         "bodyscroll" : true,
54377         /**
54378          * @event columnresize
54379          * Fires when the user resizes a column
54380          * @param {Number} columnIndex
54381          * @param {Number} newSize
54382          */
54383         "columnresize" : true,
54384         /**
54385          * @event columnmove
54386          * Fires when the user moves a column
54387          * @param {Number} oldIndex
54388          * @param {Number} newIndex
54389          */
54390         "columnmove" : true,
54391         /**
54392          * @event startdrag
54393          * Fires when row(s) start being dragged
54394          * @param {Grid} this
54395          * @param {Roo.GridDD} dd The drag drop object
54396          * @param {event} e The raw browser event
54397          */
54398         "startdrag" : true,
54399         /**
54400          * @event enddrag
54401          * Fires when a drag operation is complete
54402          * @param {Grid} this
54403          * @param {Roo.GridDD} dd The drag drop object
54404          * @param {event} e The raw browser event
54405          */
54406         "enddrag" : true,
54407         /**
54408          * @event dragdrop
54409          * Fires when dragged row(s) are dropped on a valid DD target
54410          * @param {Grid} this
54411          * @param {Roo.GridDD} dd The drag drop object
54412          * @param {String} targetId The target drag drop object
54413          * @param {event} e The raw browser event
54414          */
54415         "dragdrop" : true,
54416         /**
54417          * @event dragover
54418          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
54419          * @param {Grid} this
54420          * @param {Roo.GridDD} dd The drag drop object
54421          * @param {String} targetId The target drag drop object
54422          * @param {event} e The raw browser event
54423          */
54424         "dragover" : true,
54425         /**
54426          * @event dragenter
54427          *  Fires when the dragged row(s) first cross another DD target while being dragged
54428          * @param {Grid} this
54429          * @param {Roo.GridDD} dd The drag drop object
54430          * @param {String} targetId The target drag drop object
54431          * @param {event} e The raw browser event
54432          */
54433         "dragenter" : true,
54434         /**
54435          * @event dragout
54436          * Fires when the dragged row(s) leave another DD target while being dragged
54437          * @param {Grid} this
54438          * @param {Roo.GridDD} dd The drag drop object
54439          * @param {String} targetId The target drag drop object
54440          * @param {event} e The raw browser event
54441          */
54442         "dragout" : true,
54443         /**
54444          * @event rowclass
54445          * Fires when a row is rendered, so you can change add a style to it.
54446          * @param {GridView} gridview   The grid view
54447          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
54448          */
54449         'rowclass' : true,
54450
54451         /**
54452          * @event render
54453          * Fires when the grid is rendered
54454          * @param {Grid} grid
54455          */
54456         'render' : true
54457     });
54458
54459     Roo.grid.Grid.superclass.constructor.call(this);
54460 };
54461 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
54462     
54463     /**
54464      * @cfg {String} ddGroup - drag drop group.
54465      */
54466
54467     /**
54468      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
54469      */
54470     minColumnWidth : 25,
54471
54472     /**
54473      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
54474      * <b>on initial render.</b> It is more efficient to explicitly size the columns
54475      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
54476      */
54477     autoSizeColumns : false,
54478
54479     /**
54480      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
54481      */
54482     autoSizeHeaders : true,
54483
54484     /**
54485      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
54486      */
54487     monitorWindowResize : true,
54488
54489     /**
54490      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
54491      * rows measured to get a columns size. Default is 0 (all rows).
54492      */
54493     maxRowsToMeasure : 0,
54494
54495     /**
54496      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
54497      */
54498     trackMouseOver : true,
54499
54500     /**
54501     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
54502     */
54503     
54504     /**
54505     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
54506     */
54507     enableDragDrop : false,
54508     
54509     /**
54510     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
54511     */
54512     enableColumnMove : true,
54513     
54514     /**
54515     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
54516     */
54517     enableColumnHide : true,
54518     
54519     /**
54520     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
54521     */
54522     enableRowHeightSync : false,
54523     
54524     /**
54525     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
54526     */
54527     stripeRows : true,
54528     
54529     /**
54530     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
54531     */
54532     autoHeight : false,
54533
54534     /**
54535      * @cfg {String} autoExpandColumn The id (or dataIndex) of a column in this grid that should expand to fill unused space. This id can not be 0. Default is false.
54536      */
54537     autoExpandColumn : false,
54538
54539     /**
54540     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
54541     * Default is 50.
54542     */
54543     autoExpandMin : 50,
54544
54545     /**
54546     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
54547     */
54548     autoExpandMax : 1000,
54549
54550     /**
54551     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
54552     */
54553     view : null,
54554
54555     /**
54556     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
54557     */
54558     loadMask : false,
54559     /**
54560     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
54561     */
54562     dropTarget: false,
54563     
54564    
54565     
54566     // private
54567     rendered : false,
54568
54569     /**
54570     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
54571     * of a fixed width. Default is false.
54572     */
54573     /**
54574     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
54575     */
54576     /**
54577      * Called once after all setup has been completed and the grid is ready to be rendered.
54578      * @return {Roo.grid.Grid} this
54579      */
54580     render : function()
54581     {
54582         var c = this.container;
54583         // try to detect autoHeight/width mode
54584         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
54585             this.autoHeight = true;
54586         }
54587         var view = this.getView();
54588         view.init(this);
54589
54590         c.on("click", this.onClick, this);
54591         c.on("dblclick", this.onDblClick, this);
54592         c.on("contextmenu", this.onContextMenu, this);
54593         c.on("keydown", this.onKeyDown, this);
54594         if (Roo.isTouch) {
54595             c.on("touchstart", this.onTouchStart, this);
54596         }
54597
54598         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
54599
54600         this.getSelectionModel().init(this);
54601
54602         view.render();
54603
54604         if(this.loadMask){
54605             this.loadMask = new Roo.LoadMask(this.container,
54606                     Roo.apply({store:this.dataSource}, this.loadMask));
54607         }
54608         
54609         
54610         if (this.toolbar && this.toolbar.xtype) {
54611             this.toolbar.container = this.getView().getHeaderPanel(true);
54612             this.toolbar = new Roo.Toolbar(this.toolbar);
54613         }
54614         if (this.footer && this.footer.xtype) {
54615             this.footer.dataSource = this.getDataSource();
54616             this.footer.container = this.getView().getFooterPanel(true);
54617             this.footer = Roo.factory(this.footer, Roo);
54618         }
54619         if (this.dropTarget && this.dropTarget.xtype) {
54620             delete this.dropTarget.xtype;
54621             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
54622         }
54623         
54624         
54625         this.rendered = true;
54626         this.fireEvent('render', this);
54627         return this;
54628     },
54629
54630         /**
54631          * Reconfigures the grid to use a different Store and Column Model.
54632          * The View will be bound to the new objects and refreshed.
54633          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
54634          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
54635          */
54636     reconfigure : function(dataSource, colModel){
54637         if(this.loadMask){
54638             this.loadMask.destroy();
54639             this.loadMask = new Roo.LoadMask(this.container,
54640                     Roo.apply({store:dataSource}, this.loadMask));
54641         }
54642         this.view.bind(dataSource, colModel);
54643         this.dataSource = dataSource;
54644         this.colModel = colModel;
54645         this.view.refresh(true);
54646     },
54647
54648     // private
54649     onKeyDown : function(e){
54650         this.fireEvent("keydown", e);
54651     },
54652
54653     /**
54654      * Destroy this grid.
54655      * @param {Boolean} removeEl True to remove the element
54656      */
54657     destroy : function(removeEl, keepListeners){
54658         if(this.loadMask){
54659             this.loadMask.destroy();
54660         }
54661         var c = this.container;
54662         c.removeAllListeners();
54663         this.view.destroy();
54664         this.colModel.purgeListeners();
54665         if(!keepListeners){
54666             this.purgeListeners();
54667         }
54668         c.update("");
54669         if(removeEl === true){
54670             c.remove();
54671         }
54672     },
54673
54674     // private
54675     processEvent : function(name, e){
54676         // does this fire select???
54677         //Roo.log('grid:processEvent '  + name);
54678         
54679         if (name != 'touchstart' ) {
54680             this.fireEvent(name, e);    
54681         }
54682         
54683         var t = e.getTarget();
54684         var v = this.view;
54685         var header = v.findHeaderIndex(t);
54686         if(header !== false){
54687             var ename = name == 'touchstart' ? 'click' : name;
54688              
54689             this.fireEvent("header" + ename, this, header, e);
54690         }else{
54691             var row = v.findRowIndex(t);
54692             var cell = v.findCellIndex(t);
54693             if (name == 'touchstart') {
54694                 // first touch is always a click.
54695                 // hopefull this happens after selection is updated.?
54696                 name = false;
54697                 
54698                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
54699                     var cs = this.selModel.getSelectedCell();
54700                     if (row == cs[0] && cell == cs[1]){
54701                         name = 'dblclick';
54702                     }
54703                 }
54704                 if (typeof(this.selModel.getSelections) != 'undefined') {
54705                     var cs = this.selModel.getSelections();
54706                     var ds = this.dataSource;
54707                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
54708                         name = 'dblclick';
54709                     }
54710                 }
54711                 if (!name) {
54712                     return;
54713                 }
54714             }
54715             
54716             
54717             if(row !== false){
54718                 this.fireEvent("row" + name, this, row, e);
54719                 if(cell !== false){
54720                     this.fireEvent("cell" + name, this, row, cell, e);
54721                 }
54722             }
54723         }
54724     },
54725
54726     // private
54727     onClick : function(e){
54728         this.processEvent("click", e);
54729     },
54730    // private
54731     onTouchStart : function(e){
54732         this.processEvent("touchstart", e);
54733     },
54734
54735     // private
54736     onContextMenu : function(e, t){
54737         this.processEvent("contextmenu", e);
54738     },
54739
54740     // private
54741     onDblClick : function(e){
54742         this.processEvent("dblclick", e);
54743     },
54744
54745     // private
54746     walkCells : function(row, col, step, fn, scope){
54747         var cm = this.colModel, clen = cm.getColumnCount();
54748         var ds = this.dataSource, rlen = ds.getCount(), first = true;
54749         if(step < 0){
54750             if(col < 0){
54751                 row--;
54752                 first = false;
54753             }
54754             while(row >= 0){
54755                 if(!first){
54756                     col = clen-1;
54757                 }
54758                 first = false;
54759                 while(col >= 0){
54760                     if(fn.call(scope || this, row, col, cm) === true){
54761                         return [row, col];
54762                     }
54763                     col--;
54764                 }
54765                 row--;
54766             }
54767         } else {
54768             if(col >= clen){
54769                 row++;
54770                 first = false;
54771             }
54772             while(row < rlen){
54773                 if(!first){
54774                     col = 0;
54775                 }
54776                 first = false;
54777                 while(col < clen){
54778                     if(fn.call(scope || this, row, col, cm) === true){
54779                         return [row, col];
54780                     }
54781                     col++;
54782                 }
54783                 row++;
54784             }
54785         }
54786         return null;
54787     },
54788
54789     // private
54790     getSelections : function(){
54791         return this.selModel.getSelections();
54792     },
54793
54794     /**
54795      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
54796      * but if manual update is required this method will initiate it.
54797      */
54798     autoSize : function(){
54799         if(this.rendered){
54800             this.view.layout();
54801             if(this.view.adjustForScroll){
54802                 this.view.adjustForScroll();
54803             }
54804         }
54805     },
54806
54807     /**
54808      * Returns the grid's underlying element.
54809      * @return {Element} The element
54810      */
54811     getGridEl : function(){
54812         return this.container;
54813     },
54814
54815     // private for compatibility, overridden by editor grid
54816     stopEditing : function(){},
54817
54818     /**
54819      * Returns the grid's SelectionModel.
54820      * @return {SelectionModel}
54821      */
54822     getSelectionModel : function(){
54823         if(!this.selModel){
54824             this.selModel = new Roo.grid.RowSelectionModel();
54825         }
54826         return this.selModel;
54827     },
54828
54829     /**
54830      * Returns the grid's DataSource.
54831      * @return {DataSource}
54832      */
54833     getDataSource : function(){
54834         return this.dataSource;
54835     },
54836
54837     /**
54838      * Returns the grid's ColumnModel.
54839      * @return {ColumnModel}
54840      */
54841     getColumnModel : function(){
54842         return this.colModel;
54843     },
54844
54845     /**
54846      * Returns the grid's GridView object.
54847      * @return {GridView}
54848      */
54849     getView : function(){
54850         if(!this.view){
54851             this.view = new Roo.grid.GridView(this.viewConfig);
54852         }
54853         return this.view;
54854     },
54855     /**
54856      * Called to get grid's drag proxy text, by default returns this.ddText.
54857      * @return {String}
54858      */
54859     getDragDropText : function(){
54860         var count = this.selModel.getCount();
54861         return String.format(this.ddText, count, count == 1 ? '' : 's');
54862     }
54863 });
54864 /**
54865  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
54866  * %0 is replaced with the number of selected rows.
54867  * @type String
54868  */
54869 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
54870  * Based on:
54871  * Ext JS Library 1.1.1
54872  * Copyright(c) 2006-2007, Ext JS, LLC.
54873  *
54874  * Originally Released Under LGPL - original licence link has changed is not relivant.
54875  *
54876  * Fork - LGPL
54877  * <script type="text/javascript">
54878  */
54879  
54880 Roo.grid.AbstractGridView = function(){
54881         this.grid = null;
54882         
54883         this.events = {
54884             "beforerowremoved" : true,
54885             "beforerowsinserted" : true,
54886             "beforerefresh" : true,
54887             "rowremoved" : true,
54888             "rowsinserted" : true,
54889             "rowupdated" : true,
54890             "refresh" : true
54891         };
54892     Roo.grid.AbstractGridView.superclass.constructor.call(this);
54893 };
54894
54895 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
54896     rowClass : "x-grid-row",
54897     cellClass : "x-grid-cell",
54898     tdClass : "x-grid-td",
54899     hdClass : "x-grid-hd",
54900     splitClass : "x-grid-hd-split",
54901     
54902     init: function(grid){
54903         this.grid = grid;
54904                 var cid = this.grid.getGridEl().id;
54905         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
54906         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
54907         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
54908         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
54909         },
54910         
54911     getColumnRenderers : function(){
54912         var renderers = [];
54913         var cm = this.grid.colModel;
54914         var colCount = cm.getColumnCount();
54915         for(var i = 0; i < colCount; i++){
54916             renderers[i] = cm.getRenderer(i);
54917         }
54918         return renderers;
54919     },
54920     
54921     getColumnIds : function(){
54922         var ids = [];
54923         var cm = this.grid.colModel;
54924         var colCount = cm.getColumnCount();
54925         for(var i = 0; i < colCount; i++){
54926             ids[i] = cm.getColumnId(i);
54927         }
54928         return ids;
54929     },
54930     
54931     getDataIndexes : function(){
54932         if(!this.indexMap){
54933             this.indexMap = this.buildIndexMap();
54934         }
54935         return this.indexMap.colToData;
54936     },
54937     
54938     getColumnIndexByDataIndex : function(dataIndex){
54939         if(!this.indexMap){
54940             this.indexMap = this.buildIndexMap();
54941         }
54942         return this.indexMap.dataToCol[dataIndex];
54943     },
54944     
54945     /**
54946      * Set a css style for a column dynamically. 
54947      * @param {Number} colIndex The index of the column
54948      * @param {String} name The css property name
54949      * @param {String} value The css value
54950      */
54951     setCSSStyle : function(colIndex, name, value){
54952         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
54953         Roo.util.CSS.updateRule(selector, name, value);
54954     },
54955     
54956     generateRules : function(cm){
54957         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
54958         Roo.util.CSS.removeStyleSheet(rulesId);
54959         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
54960             var cid = cm.getColumnId(i);
54961             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
54962                          this.tdSelector, cid, " {\n}\n",
54963                          this.hdSelector, cid, " {\n}\n",
54964                          this.splitSelector, cid, " {\n}\n");
54965         }
54966         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
54967     }
54968 });/*
54969  * Based on:
54970  * Ext JS Library 1.1.1
54971  * Copyright(c) 2006-2007, Ext JS, LLC.
54972  *
54973  * Originally Released Under LGPL - original licence link has changed is not relivant.
54974  *
54975  * Fork - LGPL
54976  * <script type="text/javascript">
54977  */
54978
54979 // private
54980 // This is a support class used internally by the Grid components
54981 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
54982     this.grid = grid;
54983     this.view = grid.getView();
54984     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
54985     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
54986     if(hd2){
54987         this.setHandleElId(Roo.id(hd));
54988         this.setOuterHandleElId(Roo.id(hd2));
54989     }
54990     this.scroll = false;
54991 };
54992 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
54993     maxDragWidth: 120,
54994     getDragData : function(e){
54995         var t = Roo.lib.Event.getTarget(e);
54996         var h = this.view.findHeaderCell(t);
54997         if(h){
54998             return {ddel: h.firstChild, header:h};
54999         }
55000         return false;
55001     },
55002
55003     onInitDrag : function(e){
55004         this.view.headersDisabled = true;
55005         var clone = this.dragData.ddel.cloneNode(true);
55006         clone.id = Roo.id();
55007         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55008         this.proxy.update(clone);
55009         return true;
55010     },
55011
55012     afterValidDrop : function(){
55013         var v = this.view;
55014         setTimeout(function(){
55015             v.headersDisabled = false;
55016         }, 50);
55017     },
55018
55019     afterInvalidDrop : function(){
55020         var v = this.view;
55021         setTimeout(function(){
55022             v.headersDisabled = false;
55023         }, 50);
55024     }
55025 });
55026 /*
55027  * Based on:
55028  * Ext JS Library 1.1.1
55029  * Copyright(c) 2006-2007, Ext JS, LLC.
55030  *
55031  * Originally Released Under LGPL - original licence link has changed is not relivant.
55032  *
55033  * Fork - LGPL
55034  * <script type="text/javascript">
55035  */
55036 // private
55037 // This is a support class used internally by the Grid components
55038 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55039     this.grid = grid;
55040     this.view = grid.getView();
55041     // split the proxies so they don't interfere with mouse events
55042     this.proxyTop = Roo.DomHelper.append(document.body, {
55043         cls:"col-move-top", html:"&#160;"
55044     }, true);
55045     this.proxyBottom = Roo.DomHelper.append(document.body, {
55046         cls:"col-move-bottom", html:"&#160;"
55047     }, true);
55048     this.proxyTop.hide = this.proxyBottom.hide = function(){
55049         this.setLeftTop(-100,-100);
55050         this.setStyle("visibility", "hidden");
55051     };
55052     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55053     // temporarily disabled
55054     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55055     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55056 };
55057 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55058     proxyOffsets : [-4, -9],
55059     fly: Roo.Element.fly,
55060
55061     getTargetFromEvent : function(e){
55062         var t = Roo.lib.Event.getTarget(e);
55063         var cindex = this.view.findCellIndex(t);
55064         if(cindex !== false){
55065             return this.view.getHeaderCell(cindex);
55066         }
55067         return null;
55068     },
55069
55070     nextVisible : function(h){
55071         var v = this.view, cm = this.grid.colModel;
55072         h = h.nextSibling;
55073         while(h){
55074             if(!cm.isHidden(v.getCellIndex(h))){
55075                 return h;
55076             }
55077             h = h.nextSibling;
55078         }
55079         return null;
55080     },
55081
55082     prevVisible : function(h){
55083         var v = this.view, cm = this.grid.colModel;
55084         h = h.prevSibling;
55085         while(h){
55086             if(!cm.isHidden(v.getCellIndex(h))){
55087                 return h;
55088             }
55089             h = h.prevSibling;
55090         }
55091         return null;
55092     },
55093
55094     positionIndicator : function(h, n, e){
55095         var x = Roo.lib.Event.getPageX(e);
55096         var r = Roo.lib.Dom.getRegion(n.firstChild);
55097         var px, pt, py = r.top + this.proxyOffsets[1];
55098         if((r.right - x) <= (r.right-r.left)/2){
55099             px = r.right+this.view.borderWidth;
55100             pt = "after";
55101         }else{
55102             px = r.left;
55103             pt = "before";
55104         }
55105         var oldIndex = this.view.getCellIndex(h);
55106         var newIndex = this.view.getCellIndex(n);
55107
55108         if(this.grid.colModel.isFixed(newIndex)){
55109             return false;
55110         }
55111
55112         var locked = this.grid.colModel.isLocked(newIndex);
55113
55114         if(pt == "after"){
55115             newIndex++;
55116         }
55117         if(oldIndex < newIndex){
55118             newIndex--;
55119         }
55120         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55121             return false;
55122         }
55123         px +=  this.proxyOffsets[0];
55124         this.proxyTop.setLeftTop(px, py);
55125         this.proxyTop.show();
55126         if(!this.bottomOffset){
55127             this.bottomOffset = this.view.mainHd.getHeight();
55128         }
55129         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55130         this.proxyBottom.show();
55131         return pt;
55132     },
55133
55134     onNodeEnter : function(n, dd, e, data){
55135         if(data.header != n){
55136             this.positionIndicator(data.header, n, e);
55137         }
55138     },
55139
55140     onNodeOver : function(n, dd, e, data){
55141         var result = false;
55142         if(data.header != n){
55143             result = this.positionIndicator(data.header, n, e);
55144         }
55145         if(!result){
55146             this.proxyTop.hide();
55147             this.proxyBottom.hide();
55148         }
55149         return result ? this.dropAllowed : this.dropNotAllowed;
55150     },
55151
55152     onNodeOut : function(n, dd, e, data){
55153         this.proxyTop.hide();
55154         this.proxyBottom.hide();
55155     },
55156
55157     onNodeDrop : function(n, dd, e, data){
55158         var h = data.header;
55159         if(h != n){
55160             var cm = this.grid.colModel;
55161             var x = Roo.lib.Event.getPageX(e);
55162             var r = Roo.lib.Dom.getRegion(n.firstChild);
55163             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55164             var oldIndex = this.view.getCellIndex(h);
55165             var newIndex = this.view.getCellIndex(n);
55166             var locked = cm.isLocked(newIndex);
55167             if(pt == "after"){
55168                 newIndex++;
55169             }
55170             if(oldIndex < newIndex){
55171                 newIndex--;
55172             }
55173             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55174                 return false;
55175             }
55176             cm.setLocked(oldIndex, locked, true);
55177             cm.moveColumn(oldIndex, newIndex);
55178             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55179             return true;
55180         }
55181         return false;
55182     }
55183 });
55184 /*
55185  * Based on:
55186  * Ext JS Library 1.1.1
55187  * Copyright(c) 2006-2007, Ext JS, LLC.
55188  *
55189  * Originally Released Under LGPL - original licence link has changed is not relivant.
55190  *
55191  * Fork - LGPL
55192  * <script type="text/javascript">
55193  */
55194   
55195 /**
55196  * @class Roo.grid.GridView
55197  * @extends Roo.util.Observable
55198  *
55199  * @constructor
55200  * @param {Object} config
55201  */
55202 Roo.grid.GridView = function(config){
55203     Roo.grid.GridView.superclass.constructor.call(this);
55204     this.el = null;
55205
55206     Roo.apply(this, config);
55207 };
55208
55209 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55210
55211     unselectable :  'unselectable="on"',
55212     unselectableCls :  'x-unselectable',
55213     
55214     
55215     rowClass : "x-grid-row",
55216
55217     cellClass : "x-grid-col",
55218
55219     tdClass : "x-grid-td",
55220
55221     hdClass : "x-grid-hd",
55222
55223     splitClass : "x-grid-split",
55224
55225     sortClasses : ["sort-asc", "sort-desc"],
55226
55227     enableMoveAnim : false,
55228
55229     hlColor: "C3DAF9",
55230
55231     dh : Roo.DomHelper,
55232
55233     fly : Roo.Element.fly,
55234
55235     css : Roo.util.CSS,
55236
55237     borderWidth: 1,
55238
55239     splitOffset: 3,
55240
55241     scrollIncrement : 22,
55242
55243     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55244
55245     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55246
55247     bind : function(ds, cm){
55248         if(this.ds){
55249             this.ds.un("load", this.onLoad, this);
55250             this.ds.un("datachanged", this.onDataChange, this);
55251             this.ds.un("add", this.onAdd, this);
55252             this.ds.un("remove", this.onRemove, this);
55253             this.ds.un("update", this.onUpdate, this);
55254             this.ds.un("clear", this.onClear, this);
55255         }
55256         if(ds){
55257             ds.on("load", this.onLoad, this);
55258             ds.on("datachanged", this.onDataChange, this);
55259             ds.on("add", this.onAdd, this);
55260             ds.on("remove", this.onRemove, this);
55261             ds.on("update", this.onUpdate, this);
55262             ds.on("clear", this.onClear, this);
55263         }
55264         this.ds = ds;
55265
55266         if(this.cm){
55267             this.cm.un("widthchange", this.onColWidthChange, this);
55268             this.cm.un("headerchange", this.onHeaderChange, this);
55269             this.cm.un("hiddenchange", this.onHiddenChange, this);
55270             this.cm.un("columnmoved", this.onColumnMove, this);
55271             this.cm.un("columnlockchange", this.onColumnLock, this);
55272         }
55273         if(cm){
55274             this.generateRules(cm);
55275             cm.on("widthchange", this.onColWidthChange, this);
55276             cm.on("headerchange", this.onHeaderChange, this);
55277             cm.on("hiddenchange", this.onHiddenChange, this);
55278             cm.on("columnmoved", this.onColumnMove, this);
55279             cm.on("columnlockchange", this.onColumnLock, this);
55280         }
55281         this.cm = cm;
55282     },
55283
55284     init: function(grid){
55285         Roo.grid.GridView.superclass.init.call(this, grid);
55286
55287         this.bind(grid.dataSource, grid.colModel);
55288
55289         grid.on("headerclick", this.handleHeaderClick, this);
55290
55291         if(grid.trackMouseOver){
55292             grid.on("mouseover", this.onRowOver, this);
55293             grid.on("mouseout", this.onRowOut, this);
55294         }
55295         grid.cancelTextSelection = function(){};
55296         this.gridId = grid.id;
55297
55298         var tpls = this.templates || {};
55299
55300         if(!tpls.master){
55301             tpls.master = new Roo.Template(
55302                '<div class="x-grid" hidefocus="true">',
55303                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
55304                   '<div class="x-grid-topbar"></div>',
55305                   '<div class="x-grid-scroller"><div></div></div>',
55306                   '<div class="x-grid-locked">',
55307                       '<div class="x-grid-header">{lockedHeader}</div>',
55308                       '<div class="x-grid-body">{lockedBody}</div>',
55309                   "</div>",
55310                   '<div class="x-grid-viewport">',
55311                       '<div class="x-grid-header">{header}</div>',
55312                       '<div class="x-grid-body">{body}</div>',
55313                   "</div>",
55314                   '<div class="x-grid-bottombar"></div>',
55315                  
55316                   '<div class="x-grid-resize-proxy">&#160;</div>',
55317                "</div>"
55318             );
55319             tpls.master.disableformats = true;
55320         }
55321
55322         if(!tpls.header){
55323             tpls.header = new Roo.Template(
55324                '<table border="0" cellspacing="0" cellpadding="0">',
55325                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
55326                "</table>{splits}"
55327             );
55328             tpls.header.disableformats = true;
55329         }
55330         tpls.header.compile();
55331
55332         if(!tpls.hcell){
55333             tpls.hcell = new Roo.Template(
55334                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
55335                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
55336                 "</div></td>"
55337              );
55338              tpls.hcell.disableFormats = true;
55339         }
55340         tpls.hcell.compile();
55341
55342         if(!tpls.hsplit){
55343             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
55344                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
55345             tpls.hsplit.disableFormats = true;
55346         }
55347         tpls.hsplit.compile();
55348
55349         if(!tpls.body){
55350             tpls.body = new Roo.Template(
55351                '<table border="0" cellspacing="0" cellpadding="0">',
55352                "<tbody>{rows}</tbody>",
55353                "</table>"
55354             );
55355             tpls.body.disableFormats = true;
55356         }
55357         tpls.body.compile();
55358
55359         if(!tpls.row){
55360             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
55361             tpls.row.disableFormats = true;
55362         }
55363         tpls.row.compile();
55364
55365         if(!tpls.cell){
55366             tpls.cell = new Roo.Template(
55367                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
55368                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
55369                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
55370                 "</td>"
55371             );
55372             tpls.cell.disableFormats = true;
55373         }
55374         tpls.cell.compile();
55375
55376         this.templates = tpls;
55377     },
55378
55379     // remap these for backwards compat
55380     onColWidthChange : function(){
55381         this.updateColumns.apply(this, arguments);
55382     },
55383     onHeaderChange : function(){
55384         this.updateHeaders.apply(this, arguments);
55385     }, 
55386     onHiddenChange : function(){
55387         this.handleHiddenChange.apply(this, arguments);
55388     },
55389     onColumnMove : function(){
55390         this.handleColumnMove.apply(this, arguments);
55391     },
55392     onColumnLock : function(){
55393         this.handleLockChange.apply(this, arguments);
55394     },
55395
55396     onDataChange : function(){
55397         this.refresh();
55398         this.updateHeaderSortState();
55399     },
55400
55401     onClear : function(){
55402         this.refresh();
55403     },
55404
55405     onUpdate : function(ds, record){
55406         this.refreshRow(record);
55407     },
55408
55409     refreshRow : function(record){
55410         var ds = this.ds, index;
55411         if(typeof record == 'number'){
55412             index = record;
55413             record = ds.getAt(index);
55414         }else{
55415             index = ds.indexOf(record);
55416         }
55417         this.insertRows(ds, index, index, true);
55418         this.onRemove(ds, record, index+1, true);
55419         this.syncRowHeights(index, index);
55420         this.layout();
55421         this.fireEvent("rowupdated", this, index, record);
55422     },
55423
55424     onAdd : function(ds, records, index){
55425         this.insertRows(ds, index, index + (records.length-1));
55426     },
55427
55428     onRemove : function(ds, record, index, isUpdate){
55429         if(isUpdate !== true){
55430             this.fireEvent("beforerowremoved", this, index, record);
55431         }
55432         var bt = this.getBodyTable(), lt = this.getLockedTable();
55433         if(bt.rows[index]){
55434             bt.firstChild.removeChild(bt.rows[index]);
55435         }
55436         if(lt.rows[index]){
55437             lt.firstChild.removeChild(lt.rows[index]);
55438         }
55439         if(isUpdate !== true){
55440             this.stripeRows(index);
55441             this.syncRowHeights(index, index);
55442             this.layout();
55443             this.fireEvent("rowremoved", this, index, record);
55444         }
55445     },
55446
55447     onLoad : function(){
55448         this.scrollToTop();
55449     },
55450
55451     /**
55452      * Scrolls the grid to the top
55453      */
55454     scrollToTop : function(){
55455         if(this.scroller){
55456             this.scroller.dom.scrollTop = 0;
55457             this.syncScroll();
55458         }
55459     },
55460
55461     /**
55462      * Gets a panel in the header of the grid that can be used for toolbars etc.
55463      * After modifying the contents of this panel a call to grid.autoSize() may be
55464      * required to register any changes in size.
55465      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
55466      * @return Roo.Element
55467      */
55468     getHeaderPanel : function(doShow){
55469         if(doShow){
55470             this.headerPanel.show();
55471         }
55472         return this.headerPanel;
55473     },
55474
55475     /**
55476      * Gets a panel in the footer of the grid that can be used for toolbars etc.
55477      * After modifying the contents of this panel a call to grid.autoSize() may be
55478      * required to register any changes in size.
55479      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
55480      * @return Roo.Element
55481      */
55482     getFooterPanel : function(doShow){
55483         if(doShow){
55484             this.footerPanel.show();
55485         }
55486         return this.footerPanel;
55487     },
55488
55489     initElements : function(){
55490         var E = Roo.Element;
55491         var el = this.grid.getGridEl().dom.firstChild;
55492         var cs = el.childNodes;
55493
55494         this.el = new E(el);
55495         
55496          this.focusEl = new E(el.firstChild);
55497         this.focusEl.swallowEvent("click", true);
55498         
55499         this.headerPanel = new E(cs[1]);
55500         this.headerPanel.enableDisplayMode("block");
55501
55502         this.scroller = new E(cs[2]);
55503         this.scrollSizer = new E(this.scroller.dom.firstChild);
55504
55505         this.lockedWrap = new E(cs[3]);
55506         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
55507         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
55508
55509         this.mainWrap = new E(cs[4]);
55510         this.mainHd = new E(this.mainWrap.dom.firstChild);
55511         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
55512
55513         this.footerPanel = new E(cs[5]);
55514         this.footerPanel.enableDisplayMode("block");
55515
55516         this.resizeProxy = new E(cs[6]);
55517
55518         this.headerSelector = String.format(
55519            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
55520            this.lockedHd.id, this.mainHd.id
55521         );
55522
55523         this.splitterSelector = String.format(
55524            '#{0} div.x-grid-split, #{1} div.x-grid-split',
55525            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
55526         );
55527     },
55528     idToCssName : function(s)
55529     {
55530         return s.replace(/[^a-z0-9]+/ig, '-');
55531     },
55532
55533     getHeaderCell : function(index){
55534         return Roo.DomQuery.select(this.headerSelector)[index];
55535     },
55536
55537     getHeaderCellMeasure : function(index){
55538         return this.getHeaderCell(index).firstChild;
55539     },
55540
55541     getHeaderCellText : function(index){
55542         return this.getHeaderCell(index).firstChild.firstChild;
55543     },
55544
55545     getLockedTable : function(){
55546         return this.lockedBody.dom.firstChild;
55547     },
55548
55549     getBodyTable : function(){
55550         return this.mainBody.dom.firstChild;
55551     },
55552
55553     getLockedRow : function(index){
55554         return this.getLockedTable().rows[index];
55555     },
55556
55557     getRow : function(index){
55558         return this.getBodyTable().rows[index];
55559     },
55560
55561     getRowComposite : function(index){
55562         if(!this.rowEl){
55563             this.rowEl = new Roo.CompositeElementLite();
55564         }
55565         var els = [], lrow, mrow;
55566         if(lrow = this.getLockedRow(index)){
55567             els.push(lrow);
55568         }
55569         if(mrow = this.getRow(index)){
55570             els.push(mrow);
55571         }
55572         this.rowEl.elements = els;
55573         return this.rowEl;
55574     },
55575     /**
55576      * Gets the 'td' of the cell
55577      * 
55578      * @param {Integer} rowIndex row to select
55579      * @param {Integer} colIndex column to select
55580      * 
55581      * @return {Object} 
55582      */
55583     getCell : function(rowIndex, colIndex){
55584         var locked = this.cm.getLockedCount();
55585         var source;
55586         if(colIndex < locked){
55587             source = this.lockedBody.dom.firstChild;
55588         }else{
55589             source = this.mainBody.dom.firstChild;
55590             colIndex -= locked;
55591         }
55592         return source.rows[rowIndex].childNodes[colIndex];
55593     },
55594
55595     getCellText : function(rowIndex, colIndex){
55596         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
55597     },
55598
55599     getCellBox : function(cell){
55600         var b = this.fly(cell).getBox();
55601         if(Roo.isOpera){ // opera fails to report the Y
55602             b.y = cell.offsetTop + this.mainBody.getY();
55603         }
55604         return b;
55605     },
55606
55607     getCellIndex : function(cell){
55608         var id = String(cell.className).match(this.cellRE);
55609         if(id){
55610             return parseInt(id[1], 10);
55611         }
55612         return 0;
55613     },
55614
55615     findHeaderIndex : function(n){
55616         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55617         return r ? this.getCellIndex(r) : false;
55618     },
55619
55620     findHeaderCell : function(n){
55621         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
55622         return r ? r : false;
55623     },
55624
55625     findRowIndex : function(n){
55626         if(!n){
55627             return false;
55628         }
55629         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
55630         return r ? r.rowIndex : false;
55631     },
55632
55633     findCellIndex : function(node){
55634         var stop = this.el.dom;
55635         while(node && node != stop){
55636             if(this.findRE.test(node.className)){
55637                 return this.getCellIndex(node);
55638             }
55639             node = node.parentNode;
55640         }
55641         return false;
55642     },
55643
55644     getColumnId : function(index){
55645         return this.cm.getColumnId(index);
55646     },
55647
55648     getSplitters : function()
55649     {
55650         if(this.splitterSelector){
55651            return Roo.DomQuery.select(this.splitterSelector);
55652         }else{
55653             return null;
55654       }
55655     },
55656
55657     getSplitter : function(index){
55658         return this.getSplitters()[index];
55659     },
55660
55661     onRowOver : function(e, t){
55662         var row;
55663         if((row = this.findRowIndex(t)) !== false){
55664             this.getRowComposite(row).addClass("x-grid-row-over");
55665         }
55666     },
55667
55668     onRowOut : function(e, t){
55669         var row;
55670         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
55671             this.getRowComposite(row).removeClass("x-grid-row-over");
55672         }
55673     },
55674
55675     renderHeaders : function(){
55676         var cm = this.cm;
55677         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
55678         var cb = [], lb = [], sb = [], lsb = [], p = {};
55679         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55680             p.cellId = "x-grid-hd-0-" + i;
55681             p.splitId = "x-grid-csplit-0-" + i;
55682             p.id = cm.getColumnId(i);
55683             p.value = cm.getColumnHeader(i) || "";
55684             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
55685             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
55686             if(!cm.isLocked(i)){
55687                 cb[cb.length] = ct.apply(p);
55688                 sb[sb.length] = st.apply(p);
55689             }else{
55690                 lb[lb.length] = ct.apply(p);
55691                 lsb[lsb.length] = st.apply(p);
55692             }
55693         }
55694         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
55695                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
55696     },
55697
55698     updateHeaders : function(){
55699         var html = this.renderHeaders();
55700         this.lockedHd.update(html[0]);
55701         this.mainHd.update(html[1]);
55702     },
55703
55704     /**
55705      * Focuses the specified row.
55706      * @param {Number} row The row index
55707      */
55708     focusRow : function(row)
55709     {
55710         //Roo.log('GridView.focusRow');
55711         var x = this.scroller.dom.scrollLeft;
55712         this.focusCell(row, 0, false);
55713         this.scroller.dom.scrollLeft = x;
55714     },
55715
55716     /**
55717      * Focuses the specified cell.
55718      * @param {Number} row The row index
55719      * @param {Number} col The column index
55720      * @param {Boolean} hscroll false to disable horizontal scrolling
55721      */
55722     focusCell : function(row, col, hscroll)
55723     {
55724         //Roo.log('GridView.focusCell');
55725         var el = this.ensureVisible(row, col, hscroll);
55726         this.focusEl.alignTo(el, "tl-tl");
55727         if(Roo.isGecko){
55728             this.focusEl.focus();
55729         }else{
55730             this.focusEl.focus.defer(1, this.focusEl);
55731         }
55732     },
55733
55734     /**
55735      * Scrolls the specified cell into view
55736      * @param {Number} row The row index
55737      * @param {Number} col The column index
55738      * @param {Boolean} hscroll false to disable horizontal scrolling
55739      */
55740     ensureVisible : function(row, col, hscroll)
55741     {
55742         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
55743         //return null; //disable for testing.
55744         if(typeof row != "number"){
55745             row = row.rowIndex;
55746         }
55747         if(row < 0 && row >= this.ds.getCount()){
55748             return  null;
55749         }
55750         col = (col !== undefined ? col : 0);
55751         var cm = this.grid.colModel;
55752         while(cm.isHidden(col)){
55753             col++;
55754         }
55755
55756         var el = this.getCell(row, col);
55757         if(!el){
55758             return null;
55759         }
55760         var c = this.scroller.dom;
55761
55762         var ctop = parseInt(el.offsetTop, 10);
55763         var cleft = parseInt(el.offsetLeft, 10);
55764         var cbot = ctop + el.offsetHeight;
55765         var cright = cleft + el.offsetWidth;
55766         
55767         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
55768         var stop = parseInt(c.scrollTop, 10);
55769         var sleft = parseInt(c.scrollLeft, 10);
55770         var sbot = stop + ch;
55771         var sright = sleft + c.clientWidth;
55772         /*
55773         Roo.log('GridView.ensureVisible:' +
55774                 ' ctop:' + ctop +
55775                 ' c.clientHeight:' + c.clientHeight +
55776                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
55777                 ' stop:' + stop +
55778                 ' cbot:' + cbot +
55779                 ' sbot:' + sbot +
55780                 ' ch:' + ch  
55781                 );
55782         */
55783         if(ctop < stop){
55784              c.scrollTop = ctop;
55785             //Roo.log("set scrolltop to ctop DISABLE?");
55786         }else if(cbot > sbot){
55787             //Roo.log("set scrolltop to cbot-ch");
55788             c.scrollTop = cbot-ch;
55789         }
55790         
55791         if(hscroll !== false){
55792             if(cleft < sleft){
55793                 c.scrollLeft = cleft;
55794             }else if(cright > sright){
55795                 c.scrollLeft = cright-c.clientWidth;
55796             }
55797         }
55798          
55799         return el;
55800     },
55801
55802     updateColumns : function(){
55803         this.grid.stopEditing();
55804         var cm = this.grid.colModel, colIds = this.getColumnIds();
55805         //var totalWidth = cm.getTotalWidth();
55806         var pos = 0;
55807         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55808             //if(cm.isHidden(i)) continue;
55809             var w = cm.getColumnWidth(i);
55810             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
55811             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
55812         }
55813         this.updateSplitters();
55814     },
55815
55816     generateRules : function(cm){
55817         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
55818         Roo.util.CSS.removeStyleSheet(rulesId);
55819         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55820             var cid = cm.getColumnId(i);
55821             var align = '';
55822             if(cm.config[i].align){
55823                 align = 'text-align:'+cm.config[i].align+';';
55824             }
55825             var hidden = '';
55826             if(cm.isHidden(i)){
55827                 hidden = 'display:none;';
55828             }
55829             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
55830             ruleBuf.push(
55831                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
55832                     this.hdSelector, cid, " {\n", align, width, "}\n",
55833                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
55834                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
55835         }
55836         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55837     },
55838
55839     updateSplitters : function(){
55840         var cm = this.cm, s = this.getSplitters();
55841         if(s){ // splitters not created yet
55842             var pos = 0, locked = true;
55843             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55844                 if(cm.isHidden(i)) {
55845                     continue;
55846                 }
55847                 var w = cm.getColumnWidth(i); // make sure it's a number
55848                 if(!cm.isLocked(i) && locked){
55849                     pos = 0;
55850                     locked = false;
55851                 }
55852                 pos += w;
55853                 s[i].style.left = (pos-this.splitOffset) + "px";
55854             }
55855         }
55856     },
55857
55858     handleHiddenChange : function(colModel, colIndex, hidden){
55859         if(hidden){
55860             this.hideColumn(colIndex);
55861         }else{
55862             this.unhideColumn(colIndex);
55863         }
55864     },
55865
55866     hideColumn : function(colIndex){
55867         var cid = this.getColumnId(colIndex);
55868         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
55869         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
55870         if(Roo.isSafari){
55871             this.updateHeaders();
55872         }
55873         this.updateSplitters();
55874         this.layout();
55875     },
55876
55877     unhideColumn : function(colIndex){
55878         var cid = this.getColumnId(colIndex);
55879         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
55880         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
55881
55882         if(Roo.isSafari){
55883             this.updateHeaders();
55884         }
55885         this.updateSplitters();
55886         this.layout();
55887     },
55888
55889     insertRows : function(dm, firstRow, lastRow, isUpdate){
55890         if(firstRow == 0 && lastRow == dm.getCount()-1){
55891             this.refresh();
55892         }else{
55893             if(!isUpdate){
55894                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
55895             }
55896             var s = this.getScrollState();
55897             var markup = this.renderRows(firstRow, lastRow);
55898             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
55899             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
55900             this.restoreScroll(s);
55901             if(!isUpdate){
55902                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
55903                 this.syncRowHeights(firstRow, lastRow);
55904                 this.stripeRows(firstRow);
55905                 this.layout();
55906             }
55907         }
55908     },
55909
55910     bufferRows : function(markup, target, index){
55911         var before = null, trows = target.rows, tbody = target.tBodies[0];
55912         if(index < trows.length){
55913             before = trows[index];
55914         }
55915         var b = document.createElement("div");
55916         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
55917         var rows = b.firstChild.rows;
55918         for(var i = 0, len = rows.length; i < len; i++){
55919             if(before){
55920                 tbody.insertBefore(rows[0], before);
55921             }else{
55922                 tbody.appendChild(rows[0]);
55923             }
55924         }
55925         b.innerHTML = "";
55926         b = null;
55927     },
55928
55929     deleteRows : function(dm, firstRow, lastRow){
55930         if(dm.getRowCount()<1){
55931             this.fireEvent("beforerefresh", this);
55932             this.mainBody.update("");
55933             this.lockedBody.update("");
55934             this.fireEvent("refresh", this);
55935         }else{
55936             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
55937             var bt = this.getBodyTable();
55938             var tbody = bt.firstChild;
55939             var rows = bt.rows;
55940             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
55941                 tbody.removeChild(rows[firstRow]);
55942             }
55943             this.stripeRows(firstRow);
55944             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
55945         }
55946     },
55947
55948     updateRows : function(dataSource, firstRow, lastRow){
55949         var s = this.getScrollState();
55950         this.refresh();
55951         this.restoreScroll(s);
55952     },
55953
55954     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
55955         if(!noRefresh){
55956            this.refresh();
55957         }
55958         this.updateHeaderSortState();
55959     },
55960
55961     getScrollState : function(){
55962         
55963         var sb = this.scroller.dom;
55964         return {left: sb.scrollLeft, top: sb.scrollTop};
55965     },
55966
55967     stripeRows : function(startRow){
55968         if(!this.grid.stripeRows || this.ds.getCount() < 1){
55969             return;
55970         }
55971         startRow = startRow || 0;
55972         var rows = this.getBodyTable().rows;
55973         var lrows = this.getLockedTable().rows;
55974         var cls = ' x-grid-row-alt ';
55975         for(var i = startRow, len = rows.length; i < len; i++){
55976             var row = rows[i], lrow = lrows[i];
55977             var isAlt = ((i+1) % 2 == 0);
55978             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
55979             if(isAlt == hasAlt){
55980                 continue;
55981             }
55982             if(isAlt){
55983                 row.className += " x-grid-row-alt";
55984             }else{
55985                 row.className = row.className.replace("x-grid-row-alt", "");
55986             }
55987             if(lrow){
55988                 lrow.className = row.className;
55989             }
55990         }
55991     },
55992
55993     restoreScroll : function(state){
55994         //Roo.log('GridView.restoreScroll');
55995         var sb = this.scroller.dom;
55996         sb.scrollLeft = state.left;
55997         sb.scrollTop = state.top;
55998         this.syncScroll();
55999     },
56000
56001     syncScroll : function(){
56002         //Roo.log('GridView.syncScroll');
56003         var sb = this.scroller.dom;
56004         var sh = this.mainHd.dom;
56005         var bs = this.mainBody.dom;
56006         var lv = this.lockedBody.dom;
56007         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56008         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56009     },
56010
56011     handleScroll : function(e){
56012         this.syncScroll();
56013         var sb = this.scroller.dom;
56014         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56015         e.stopEvent();
56016     },
56017
56018     handleWheel : function(e){
56019         var d = e.getWheelDelta();
56020         this.scroller.dom.scrollTop -= d*22;
56021         // set this here to prevent jumpy scrolling on large tables
56022         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56023         e.stopEvent();
56024     },
56025
56026     renderRows : function(startRow, endRow){
56027         // pull in all the crap needed to render rows
56028         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56029         var colCount = cm.getColumnCount();
56030
56031         if(ds.getCount() < 1){
56032             return ["", ""];
56033         }
56034
56035         // build a map for all the columns
56036         var cs = [];
56037         for(var i = 0; i < colCount; i++){
56038             var name = cm.getDataIndex(i);
56039             cs[i] = {
56040                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56041                 renderer : cm.getRenderer(i),
56042                 id : cm.getColumnId(i),
56043                 locked : cm.isLocked(i),
56044                 has_editor : cm.isCellEditable(i)
56045             };
56046         }
56047
56048         startRow = startRow || 0;
56049         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56050
56051         // records to render
56052         var rs = ds.getRange(startRow, endRow);
56053
56054         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56055     },
56056
56057     // As much as I hate to duplicate code, this was branched because FireFox really hates
56058     // [].join("") on strings. The performance difference was substantial enough to
56059     // branch this function
56060     doRender : Roo.isGecko ?
56061             function(cs, rs, ds, startRow, colCount, stripe){
56062                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56063                 // buffers
56064                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56065                 
56066                 var hasListener = this.grid.hasListener('rowclass');
56067                 var rowcfg = {};
56068                 for(var j = 0, len = rs.length; j < len; j++){
56069                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56070                     for(var i = 0; i < colCount; i++){
56071                         c = cs[i];
56072                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56073                         p.id = c.id;
56074                         p.css = p.attr = "";
56075                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56076                         if(p.value == undefined || p.value === "") {
56077                             p.value = "&#160;";
56078                         }
56079                         if(c.has_editor){
56080                             p.css += ' x-grid-editable-cell';
56081                         }
56082                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56083                             p.css +=  ' x-grid-dirty-cell';
56084                         }
56085                         var markup = ct.apply(p);
56086                         if(!c.locked){
56087                             cb+= markup;
56088                         }else{
56089                             lcb+= markup;
56090                         }
56091                     }
56092                     var alt = [];
56093                     if(stripe && ((rowIndex+1) % 2 == 0)){
56094                         alt.push("x-grid-row-alt")
56095                     }
56096                     if(r.dirty){
56097                         alt.push(  " x-grid-dirty-row");
56098                     }
56099                     rp.cells = lcb;
56100                     if(this.getRowClass){
56101                         alt.push(this.getRowClass(r, rowIndex));
56102                     }
56103                     if (hasListener) {
56104                         rowcfg = {
56105                              
56106                             record: r,
56107                             rowIndex : rowIndex,
56108                             rowClass : ''
56109                         };
56110                         this.grid.fireEvent('rowclass', this, rowcfg);
56111                         alt.push(rowcfg.rowClass);
56112                     }
56113                     rp.alt = alt.join(" ");
56114                     lbuf+= rt.apply(rp);
56115                     rp.cells = cb;
56116                     buf+=  rt.apply(rp);
56117                 }
56118                 return [lbuf, buf];
56119             } :
56120             function(cs, rs, ds, startRow, colCount, stripe){
56121                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56122                 // buffers
56123                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56124                 var hasListener = this.grid.hasListener('rowclass');
56125  
56126                 var rowcfg = {};
56127                 for(var j = 0, len = rs.length; j < len; j++){
56128                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56129                     for(var i = 0; i < colCount; i++){
56130                         c = cs[i];
56131                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56132                         p.id = c.id;
56133                         p.css = p.attr = "";
56134                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56135                         if(p.value == undefined || p.value === "") {
56136                             p.value = "&#160;";
56137                         }
56138                         //Roo.log(c);
56139                          if(c.has_editor){
56140                             p.css += ' x-grid-editable-cell';
56141                         }
56142                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56143                             p.css += ' x-grid-dirty-cell' 
56144                         }
56145                         
56146                         var markup = ct.apply(p);
56147                         if(!c.locked){
56148                             cb[cb.length] = markup;
56149                         }else{
56150                             lcb[lcb.length] = markup;
56151                         }
56152                     }
56153                     var alt = [];
56154                     if(stripe && ((rowIndex+1) % 2 == 0)){
56155                         alt.push( "x-grid-row-alt");
56156                     }
56157                     if(r.dirty){
56158                         alt.push(" x-grid-dirty-row");
56159                     }
56160                     rp.cells = lcb;
56161                     if(this.getRowClass){
56162                         alt.push( this.getRowClass(r, rowIndex));
56163                     }
56164                     if (hasListener) {
56165                         rowcfg = {
56166                              
56167                             record: r,
56168                             rowIndex : rowIndex,
56169                             rowClass : ''
56170                         };
56171                         this.grid.fireEvent('rowclass', this, rowcfg);
56172                         alt.push(rowcfg.rowClass);
56173                     }
56174                     
56175                     rp.alt = alt.join(" ");
56176                     rp.cells = lcb.join("");
56177                     lbuf[lbuf.length] = rt.apply(rp);
56178                     rp.cells = cb.join("");
56179                     buf[buf.length] =  rt.apply(rp);
56180                 }
56181                 return [lbuf.join(""), buf.join("")];
56182             },
56183
56184     renderBody : function(){
56185         var markup = this.renderRows();
56186         var bt = this.templates.body;
56187         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56188     },
56189
56190     /**
56191      * Refreshes the grid
56192      * @param {Boolean} headersToo
56193      */
56194     refresh : function(headersToo){
56195         this.fireEvent("beforerefresh", this);
56196         this.grid.stopEditing();
56197         var result = this.renderBody();
56198         this.lockedBody.update(result[0]);
56199         this.mainBody.update(result[1]);
56200         if(headersToo === true){
56201             this.updateHeaders();
56202             this.updateColumns();
56203             this.updateSplitters();
56204             this.updateHeaderSortState();
56205         }
56206         this.syncRowHeights();
56207         this.layout();
56208         this.fireEvent("refresh", this);
56209     },
56210
56211     handleColumnMove : function(cm, oldIndex, newIndex){
56212         this.indexMap = null;
56213         var s = this.getScrollState();
56214         this.refresh(true);
56215         this.restoreScroll(s);
56216         this.afterMove(newIndex);
56217     },
56218
56219     afterMove : function(colIndex){
56220         if(this.enableMoveAnim && Roo.enableFx){
56221             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56222         }
56223         // if multisort - fix sortOrder, and reload..
56224         if (this.grid.dataSource.multiSort) {
56225             // the we can call sort again..
56226             var dm = this.grid.dataSource;
56227             var cm = this.grid.colModel;
56228             var so = [];
56229             for(var i = 0; i < cm.config.length; i++ ) {
56230                 
56231                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56232                     continue; // dont' bother, it's not in sort list or being set.
56233                 }
56234                 
56235                 so.push(cm.config[i].dataIndex);
56236             };
56237             dm.sortOrder = so;
56238             dm.load(dm.lastOptions);
56239             
56240             
56241         }
56242         
56243     },
56244
56245     updateCell : function(dm, rowIndex, dataIndex){
56246         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56247         if(typeof colIndex == "undefined"){ // not present in grid
56248             return;
56249         }
56250         var cm = this.grid.colModel;
56251         var cell = this.getCell(rowIndex, colIndex);
56252         var cellText = this.getCellText(rowIndex, colIndex);
56253
56254         var p = {
56255             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56256             id : cm.getColumnId(colIndex),
56257             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56258         };
56259         var renderer = cm.getRenderer(colIndex);
56260         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56261         if(typeof val == "undefined" || val === "") {
56262             val = "&#160;";
56263         }
56264         cellText.innerHTML = val;
56265         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56266         this.syncRowHeights(rowIndex, rowIndex);
56267     },
56268
56269     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56270         var maxWidth = 0;
56271         if(this.grid.autoSizeHeaders){
56272             var h = this.getHeaderCellMeasure(colIndex);
56273             maxWidth = Math.max(maxWidth, h.scrollWidth);
56274         }
56275         var tb, index;
56276         if(this.cm.isLocked(colIndex)){
56277             tb = this.getLockedTable();
56278             index = colIndex;
56279         }else{
56280             tb = this.getBodyTable();
56281             index = colIndex - this.cm.getLockedCount();
56282         }
56283         if(tb && tb.rows){
56284             var rows = tb.rows;
56285             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
56286             for(var i = 0; i < stopIndex; i++){
56287                 var cell = rows[i].childNodes[index].firstChild;
56288                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
56289             }
56290         }
56291         return maxWidth + /*margin for error in IE*/ 5;
56292     },
56293     /**
56294      * Autofit a column to its content.
56295      * @param {Number} colIndex
56296      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
56297      */
56298      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
56299          if(this.cm.isHidden(colIndex)){
56300              return; // can't calc a hidden column
56301          }
56302         if(forceMinSize){
56303             var cid = this.cm.getColumnId(colIndex);
56304             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
56305            if(this.grid.autoSizeHeaders){
56306                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
56307            }
56308         }
56309         var newWidth = this.calcColumnWidth(colIndex);
56310         this.cm.setColumnWidth(colIndex,
56311             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
56312         if(!suppressEvent){
56313             this.grid.fireEvent("columnresize", colIndex, newWidth);
56314         }
56315     },
56316
56317     /**
56318      * Autofits all columns to their content and then expands to fit any extra space in the grid
56319      */
56320      autoSizeColumns : function(){
56321         var cm = this.grid.colModel;
56322         var colCount = cm.getColumnCount();
56323         for(var i = 0; i < colCount; i++){
56324             this.autoSizeColumn(i, true, true);
56325         }
56326         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
56327             this.fitColumns();
56328         }else{
56329             this.updateColumns();
56330             this.layout();
56331         }
56332     },
56333
56334     /**
56335      * Autofits all columns to the grid's width proportionate with their current size
56336      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
56337      */
56338     fitColumns : function(reserveScrollSpace){
56339         var cm = this.grid.colModel;
56340         var colCount = cm.getColumnCount();
56341         var cols = [];
56342         var width = 0;
56343         var i, w;
56344         for (i = 0; i < colCount; i++){
56345             if(!cm.isHidden(i) && !cm.isFixed(i)){
56346                 w = cm.getColumnWidth(i);
56347                 cols.push(i);
56348                 cols.push(w);
56349                 width += w;
56350             }
56351         }
56352         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
56353         if(reserveScrollSpace){
56354             avail -= 17;
56355         }
56356         var frac = (avail - cm.getTotalWidth())/width;
56357         while (cols.length){
56358             w = cols.pop();
56359             i = cols.pop();
56360             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
56361         }
56362         this.updateColumns();
56363         this.layout();
56364     },
56365
56366     onRowSelect : function(rowIndex){
56367         var row = this.getRowComposite(rowIndex);
56368         row.addClass("x-grid-row-selected");
56369     },
56370
56371     onRowDeselect : function(rowIndex){
56372         var row = this.getRowComposite(rowIndex);
56373         row.removeClass("x-grid-row-selected");
56374     },
56375
56376     onCellSelect : function(row, col){
56377         var cell = this.getCell(row, col);
56378         if(cell){
56379             Roo.fly(cell).addClass("x-grid-cell-selected");
56380         }
56381     },
56382
56383     onCellDeselect : function(row, col){
56384         var cell = this.getCell(row, col);
56385         if(cell){
56386             Roo.fly(cell).removeClass("x-grid-cell-selected");
56387         }
56388     },
56389
56390     updateHeaderSortState : function(){
56391         
56392         // sort state can be single { field: xxx, direction : yyy}
56393         // or   { xxx=>ASC , yyy : DESC ..... }
56394         
56395         var mstate = {};
56396         if (!this.ds.multiSort) { 
56397             var state = this.ds.getSortState();
56398             if(!state){
56399                 return;
56400             }
56401             mstate[state.field] = state.direction;
56402             // FIXME... - this is not used here.. but might be elsewhere..
56403             this.sortState = state;
56404             
56405         } else {
56406             mstate = this.ds.sortToggle;
56407         }
56408         //remove existing sort classes..
56409         
56410         var sc = this.sortClasses;
56411         var hds = this.el.select(this.headerSelector).removeClass(sc);
56412         
56413         for(var f in mstate) {
56414         
56415             var sortColumn = this.cm.findColumnIndex(f);
56416             
56417             if(sortColumn != -1){
56418                 var sortDir = mstate[f];        
56419                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
56420             }
56421         }
56422         
56423          
56424         
56425     },
56426
56427
56428     handleHeaderClick : function(g, index,e){
56429         
56430         Roo.log("header click");
56431         
56432         if (Roo.isTouch) {
56433             // touch events on header are handled by context
56434             this.handleHdCtx(g,index,e);
56435             return;
56436         }
56437         
56438         
56439         if(this.headersDisabled){
56440             return;
56441         }
56442         var dm = g.dataSource, cm = g.colModel;
56443         if(!cm.isSortable(index)){
56444             return;
56445         }
56446         g.stopEditing();
56447         
56448         if (dm.multiSort) {
56449             // update the sortOrder
56450             var so = [];
56451             for(var i = 0; i < cm.config.length; i++ ) {
56452                 
56453                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
56454                     continue; // dont' bother, it's not in sort list or being set.
56455                 }
56456                 
56457                 so.push(cm.config[i].dataIndex);
56458             };
56459             dm.sortOrder = so;
56460         }
56461         
56462         
56463         dm.sort(cm.getDataIndex(index));
56464     },
56465
56466
56467     destroy : function(){
56468         if(this.colMenu){
56469             this.colMenu.removeAll();
56470             Roo.menu.MenuMgr.unregister(this.colMenu);
56471             this.colMenu.getEl().remove();
56472             delete this.colMenu;
56473         }
56474         if(this.hmenu){
56475             this.hmenu.removeAll();
56476             Roo.menu.MenuMgr.unregister(this.hmenu);
56477             this.hmenu.getEl().remove();
56478             delete this.hmenu;
56479         }
56480         if(this.grid.enableColumnMove){
56481             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56482             if(dds){
56483                 for(var dd in dds){
56484                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
56485                         var elid = dds[dd].dragElId;
56486                         dds[dd].unreg();
56487                         Roo.get(elid).remove();
56488                     } else if(dds[dd].config.isTarget){
56489                         dds[dd].proxyTop.remove();
56490                         dds[dd].proxyBottom.remove();
56491                         dds[dd].unreg();
56492                     }
56493                     if(Roo.dd.DDM.locationCache[dd]){
56494                         delete Roo.dd.DDM.locationCache[dd];
56495                     }
56496                 }
56497                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
56498             }
56499         }
56500         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
56501         this.bind(null, null);
56502         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
56503     },
56504
56505     handleLockChange : function(){
56506         this.refresh(true);
56507     },
56508
56509     onDenyColumnLock : function(){
56510
56511     },
56512
56513     onDenyColumnHide : function(){
56514
56515     },
56516
56517     handleHdMenuClick : function(item){
56518         var index = this.hdCtxIndex;
56519         var cm = this.cm, ds = this.ds;
56520         switch(item.id){
56521             case "asc":
56522                 ds.sort(cm.getDataIndex(index), "ASC");
56523                 break;
56524             case "desc":
56525                 ds.sort(cm.getDataIndex(index), "DESC");
56526                 break;
56527             case "lock":
56528                 var lc = cm.getLockedCount();
56529                 if(cm.getColumnCount(true) <= lc+1){
56530                     this.onDenyColumnLock();
56531                     return;
56532                 }
56533                 if(lc != index){
56534                     cm.setLocked(index, true, true);
56535                     cm.moveColumn(index, lc);
56536                     this.grid.fireEvent("columnmove", index, lc);
56537                 }else{
56538                     cm.setLocked(index, true);
56539                 }
56540             break;
56541             case "unlock":
56542                 var lc = cm.getLockedCount();
56543                 if((lc-1) != index){
56544                     cm.setLocked(index, false, true);
56545                     cm.moveColumn(index, lc-1);
56546                     this.grid.fireEvent("columnmove", index, lc-1);
56547                 }else{
56548                     cm.setLocked(index, false);
56549                 }
56550             break;
56551             case 'wider': // used to expand cols on touch..
56552             case 'narrow':
56553                 var cw = cm.getColumnWidth(index);
56554                 cw += (item.id == 'wider' ? 1 : -1) * 50;
56555                 cw = Math.max(0, cw);
56556                 cw = Math.min(cw,4000);
56557                 cm.setColumnWidth(index, cw);
56558                 break;
56559                 
56560             default:
56561                 index = cm.getIndexById(item.id.substr(4));
56562                 if(index != -1){
56563                     if(item.checked && cm.getColumnCount(true) <= 1){
56564                         this.onDenyColumnHide();
56565                         return false;
56566                     }
56567                     cm.setHidden(index, item.checked);
56568                 }
56569         }
56570         return true;
56571     },
56572
56573     beforeColMenuShow : function(){
56574         var cm = this.cm,  colCount = cm.getColumnCount();
56575         this.colMenu.removeAll();
56576         for(var i = 0; i < colCount; i++){
56577             this.colMenu.add(new Roo.menu.CheckItem({
56578                 id: "col-"+cm.getColumnId(i),
56579                 text: cm.getColumnHeader(i),
56580                 checked: !cm.isHidden(i),
56581                 hideOnClick:false
56582             }));
56583         }
56584     },
56585
56586     handleHdCtx : function(g, index, e){
56587         e.stopEvent();
56588         var hd = this.getHeaderCell(index);
56589         this.hdCtxIndex = index;
56590         var ms = this.hmenu.items, cm = this.cm;
56591         ms.get("asc").setDisabled(!cm.isSortable(index));
56592         ms.get("desc").setDisabled(!cm.isSortable(index));
56593         if(this.grid.enableColLock !== false){
56594             ms.get("lock").setDisabled(cm.isLocked(index));
56595             ms.get("unlock").setDisabled(!cm.isLocked(index));
56596         }
56597         this.hmenu.show(hd, "tl-bl");
56598     },
56599
56600     handleHdOver : function(e){
56601         var hd = this.findHeaderCell(e.getTarget());
56602         if(hd && !this.headersDisabled){
56603             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
56604                this.fly(hd).addClass("x-grid-hd-over");
56605             }
56606         }
56607     },
56608
56609     handleHdOut : function(e){
56610         var hd = this.findHeaderCell(e.getTarget());
56611         if(hd){
56612             this.fly(hd).removeClass("x-grid-hd-over");
56613         }
56614     },
56615
56616     handleSplitDblClick : function(e, t){
56617         var i = this.getCellIndex(t);
56618         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
56619             this.autoSizeColumn(i, true);
56620             this.layout();
56621         }
56622     },
56623
56624     render : function(){
56625
56626         var cm = this.cm;
56627         var colCount = cm.getColumnCount();
56628
56629         if(this.grid.monitorWindowResize === true){
56630             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
56631         }
56632         var header = this.renderHeaders();
56633         var body = this.templates.body.apply({rows:""});
56634         var html = this.templates.master.apply({
56635             lockedBody: body,
56636             body: body,
56637             lockedHeader: header[0],
56638             header: header[1]
56639         });
56640
56641         //this.updateColumns();
56642
56643         this.grid.getGridEl().dom.innerHTML = html;
56644
56645         this.initElements();
56646         
56647         // a kludge to fix the random scolling effect in webkit
56648         this.el.on("scroll", function() {
56649             this.el.dom.scrollTop=0; // hopefully not recursive..
56650         },this);
56651
56652         this.scroller.on("scroll", this.handleScroll, this);
56653         this.lockedBody.on("mousewheel", this.handleWheel, this);
56654         this.mainBody.on("mousewheel", this.handleWheel, this);
56655
56656         this.mainHd.on("mouseover", this.handleHdOver, this);
56657         this.mainHd.on("mouseout", this.handleHdOut, this);
56658         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
56659                 {delegate: "."+this.splitClass});
56660
56661         this.lockedHd.on("mouseover", this.handleHdOver, this);
56662         this.lockedHd.on("mouseout", this.handleHdOut, this);
56663         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
56664                 {delegate: "."+this.splitClass});
56665
56666         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
56667             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56668         }
56669
56670         this.updateSplitters();
56671
56672         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
56673             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56674             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
56675         }
56676
56677         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
56678             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
56679             this.hmenu.add(
56680                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
56681                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
56682             );
56683             if(this.grid.enableColLock !== false){
56684                 this.hmenu.add('-',
56685                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
56686                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
56687                 );
56688             }
56689             if (Roo.isTouch) {
56690                  this.hmenu.add('-',
56691                     {id:"wider", text: this.columnsWiderText},
56692                     {id:"narrow", text: this.columnsNarrowText }
56693                 );
56694                 
56695                  
56696             }
56697             
56698             if(this.grid.enableColumnHide !== false){
56699
56700                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
56701                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
56702                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
56703
56704                 this.hmenu.add('-',
56705                     {id:"columns", text: this.columnsText, menu: this.colMenu}
56706                 );
56707             }
56708             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
56709
56710             this.grid.on("headercontextmenu", this.handleHdCtx, this);
56711         }
56712
56713         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
56714             this.dd = new Roo.grid.GridDragZone(this.grid, {
56715                 ddGroup : this.grid.ddGroup || 'GridDD'
56716             });
56717             
56718         }
56719
56720         /*
56721         for(var i = 0; i < colCount; i++){
56722             if(cm.isHidden(i)){
56723                 this.hideColumn(i);
56724             }
56725             if(cm.config[i].align){
56726                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
56727                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
56728             }
56729         }*/
56730         
56731         this.updateHeaderSortState();
56732
56733         this.beforeInitialResize();
56734         this.layout(true);
56735
56736         // two part rendering gives faster view to the user
56737         this.renderPhase2.defer(1, this);
56738     },
56739
56740     renderPhase2 : function(){
56741         // render the rows now
56742         this.refresh();
56743         if(this.grid.autoSizeColumns){
56744             this.autoSizeColumns();
56745         }
56746     },
56747
56748     beforeInitialResize : function(){
56749
56750     },
56751
56752     onColumnSplitterMoved : function(i, w){
56753         this.userResized = true;
56754         var cm = this.grid.colModel;
56755         cm.setColumnWidth(i, w, true);
56756         var cid = cm.getColumnId(i);
56757         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
56758         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
56759         this.updateSplitters();
56760         this.layout();
56761         this.grid.fireEvent("columnresize", i, w);
56762     },
56763
56764     syncRowHeights : function(startIndex, endIndex){
56765         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
56766             startIndex = startIndex || 0;
56767             var mrows = this.getBodyTable().rows;
56768             var lrows = this.getLockedTable().rows;
56769             var len = mrows.length-1;
56770             endIndex = Math.min(endIndex || len, len);
56771             for(var i = startIndex; i <= endIndex; i++){
56772                 var m = mrows[i], l = lrows[i];
56773                 var h = Math.max(m.offsetHeight, l.offsetHeight);
56774                 m.style.height = l.style.height = h + "px";
56775             }
56776         }
56777     },
56778
56779     layout : function(initialRender, is2ndPass){
56780         var g = this.grid;
56781         var auto = g.autoHeight;
56782         var scrollOffset = 16;
56783         var c = g.getGridEl(), cm = this.cm,
56784                 expandCol = g.autoExpandColumn,
56785                 gv = this;
56786         //c.beginMeasure();
56787
56788         if(!c.dom.offsetWidth){ // display:none?
56789             if(initialRender){
56790                 this.lockedWrap.show();
56791                 this.mainWrap.show();
56792             }
56793             return;
56794         }
56795
56796         var hasLock = this.cm.isLocked(0);
56797
56798         var tbh = this.headerPanel.getHeight();
56799         var bbh = this.footerPanel.getHeight();
56800
56801         if(auto){
56802             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
56803             var newHeight = ch + c.getBorderWidth("tb");
56804             if(g.maxHeight){
56805                 newHeight = Math.min(g.maxHeight, newHeight);
56806             }
56807             c.setHeight(newHeight);
56808         }
56809
56810         if(g.autoWidth){
56811             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
56812         }
56813
56814         var s = this.scroller;
56815
56816         var csize = c.getSize(true);
56817
56818         this.el.setSize(csize.width, csize.height);
56819
56820         this.headerPanel.setWidth(csize.width);
56821         this.footerPanel.setWidth(csize.width);
56822
56823         var hdHeight = this.mainHd.getHeight();
56824         var vw = csize.width;
56825         var vh = csize.height - (tbh + bbh);
56826
56827         s.setSize(vw, vh);
56828
56829         var bt = this.getBodyTable();
56830         
56831         if(cm.getLockedCount() == cm.config.length){
56832             bt = this.getLockedTable();
56833         }
56834         
56835         var ltWidth = hasLock ?
56836                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
56837
56838         var scrollHeight = bt.offsetHeight;
56839         var scrollWidth = ltWidth + bt.offsetWidth;
56840         var vscroll = false, hscroll = false;
56841
56842         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
56843
56844         var lw = this.lockedWrap, mw = this.mainWrap;
56845         var lb = this.lockedBody, mb = this.mainBody;
56846
56847         setTimeout(function(){
56848             var t = s.dom.offsetTop;
56849             var w = s.dom.clientWidth,
56850                 h = s.dom.clientHeight;
56851
56852             lw.setTop(t);
56853             lw.setSize(ltWidth, h);
56854
56855             mw.setLeftTop(ltWidth, t);
56856             mw.setSize(w-ltWidth, h);
56857
56858             lb.setHeight(h-hdHeight);
56859             mb.setHeight(h-hdHeight);
56860
56861             if(is2ndPass !== true && !gv.userResized && expandCol){
56862                 // high speed resize without full column calculation
56863                 
56864                 var ci = cm.getIndexById(expandCol);
56865                 if (ci < 0) {
56866                     ci = cm.findColumnIndex(expandCol);
56867                 }
56868                 ci = Math.max(0, ci); // make sure it's got at least the first col.
56869                 var expandId = cm.getColumnId(ci);
56870                 var  tw = cm.getTotalWidth(false);
56871                 var currentWidth = cm.getColumnWidth(ci);
56872                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
56873                 if(currentWidth != cw){
56874                     cm.setColumnWidth(ci, cw, true);
56875                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
56876                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
56877                     gv.updateSplitters();
56878                     gv.layout(false, true);
56879                 }
56880             }
56881
56882             if(initialRender){
56883                 lw.show();
56884                 mw.show();
56885             }
56886             //c.endMeasure();
56887         }, 10);
56888     },
56889
56890     onWindowResize : function(){
56891         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
56892             return;
56893         }
56894         this.layout();
56895     },
56896
56897     appendFooter : function(parentEl){
56898         return null;
56899     },
56900
56901     sortAscText : "Sort Ascending",
56902     sortDescText : "Sort Descending",
56903     lockText : "Lock Column",
56904     unlockText : "Unlock Column",
56905     columnsText : "Columns",
56906  
56907     columnsWiderText : "Wider",
56908     columnsNarrowText : "Thinner"
56909 });
56910
56911
56912 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
56913     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
56914     this.proxy.el.addClass('x-grid3-col-dd');
56915 };
56916
56917 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
56918     handleMouseDown : function(e){
56919
56920     },
56921
56922     callHandleMouseDown : function(e){
56923         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
56924     }
56925 });
56926 /*
56927  * Based on:
56928  * Ext JS Library 1.1.1
56929  * Copyright(c) 2006-2007, Ext JS, LLC.
56930  *
56931  * Originally Released Under LGPL - original licence link has changed is not relivant.
56932  *
56933  * Fork - LGPL
56934  * <script type="text/javascript">
56935  */
56936  
56937 // private
56938 // This is a support class used internally by the Grid components
56939 Roo.grid.SplitDragZone = function(grid, hd, hd2){
56940     this.grid = grid;
56941     this.view = grid.getView();
56942     this.proxy = this.view.resizeProxy;
56943     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
56944         "gridSplitters" + this.grid.getGridEl().id, {
56945         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
56946     });
56947     this.setHandleElId(Roo.id(hd));
56948     this.setOuterHandleElId(Roo.id(hd2));
56949     this.scroll = false;
56950 };
56951 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
56952     fly: Roo.Element.fly,
56953
56954     b4StartDrag : function(x, y){
56955         this.view.headersDisabled = true;
56956         this.proxy.setHeight(this.view.mainWrap.getHeight());
56957         var w = this.cm.getColumnWidth(this.cellIndex);
56958         var minw = Math.max(w-this.grid.minColumnWidth, 0);
56959         this.resetConstraints();
56960         this.setXConstraint(minw, 1000);
56961         this.setYConstraint(0, 0);
56962         this.minX = x - minw;
56963         this.maxX = x + 1000;
56964         this.startPos = x;
56965         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
56966     },
56967
56968
56969     handleMouseDown : function(e){
56970         ev = Roo.EventObject.setEvent(e);
56971         var t = this.fly(ev.getTarget());
56972         if(t.hasClass("x-grid-split")){
56973             this.cellIndex = this.view.getCellIndex(t.dom);
56974             this.split = t.dom;
56975             this.cm = this.grid.colModel;
56976             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
56977                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
56978             }
56979         }
56980     },
56981
56982     endDrag : function(e){
56983         this.view.headersDisabled = false;
56984         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
56985         var diff = endX - this.startPos;
56986         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
56987     },
56988
56989     autoOffset : function(){
56990         this.setDelta(0,0);
56991     }
56992 });/*
56993  * Based on:
56994  * Ext JS Library 1.1.1
56995  * Copyright(c) 2006-2007, Ext JS, LLC.
56996  *
56997  * Originally Released Under LGPL - original licence link has changed is not relivant.
56998  *
56999  * Fork - LGPL
57000  * <script type="text/javascript">
57001  */
57002  
57003 // private
57004 // This is a support class used internally by the Grid components
57005 Roo.grid.GridDragZone = function(grid, config){
57006     this.view = grid.getView();
57007     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57008     if(this.view.lockedBody){
57009         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57010         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57011     }
57012     this.scroll = false;
57013     this.grid = grid;
57014     this.ddel = document.createElement('div');
57015     this.ddel.className = 'x-grid-dd-wrap';
57016 };
57017
57018 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57019     ddGroup : "GridDD",
57020
57021     getDragData : function(e){
57022         var t = Roo.lib.Event.getTarget(e);
57023         var rowIndex = this.view.findRowIndex(t);
57024         var sm = this.grid.selModel;
57025             
57026         //Roo.log(rowIndex);
57027         
57028         if (sm.getSelectedCell) {
57029             // cell selection..
57030             if (!sm.getSelectedCell()) {
57031                 return false;
57032             }
57033             if (rowIndex != sm.getSelectedCell()[0]) {
57034                 return false;
57035             }
57036         
57037         }
57038         
57039         if(rowIndex !== false){
57040             
57041             // if editorgrid.. 
57042             
57043             
57044             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57045                
57046             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57047               //  
57048             //}
57049             if (e.hasModifier()){
57050                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57051             }
57052             
57053             Roo.log("getDragData");
57054             
57055             return {
57056                 grid: this.grid,
57057                 ddel: this.ddel,
57058                 rowIndex: rowIndex,
57059                 selections:sm.getSelections ? sm.getSelections() : (
57060                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57061                 )
57062             };
57063         }
57064         return false;
57065     },
57066
57067     onInitDrag : function(e){
57068         var data = this.dragData;
57069         this.ddel.innerHTML = this.grid.getDragDropText();
57070         this.proxy.update(this.ddel);
57071         // fire start drag?
57072     },
57073
57074     afterRepair : function(){
57075         this.dragging = false;
57076     },
57077
57078     getRepairXY : function(e, data){
57079         return false;
57080     },
57081
57082     onEndDrag : function(data, e){
57083         // fire end drag?
57084     },
57085
57086     onValidDrop : function(dd, e, id){
57087         // fire drag drop?
57088         this.hideProxy();
57089     },
57090
57091     beforeInvalidDrop : function(e, id){
57092
57093     }
57094 });/*
57095  * Based on:
57096  * Ext JS Library 1.1.1
57097  * Copyright(c) 2006-2007, Ext JS, LLC.
57098  *
57099  * Originally Released Under LGPL - original licence link has changed is not relivant.
57100  *
57101  * Fork - LGPL
57102  * <script type="text/javascript">
57103  */
57104  
57105
57106 /**
57107  * @class Roo.grid.ColumnModel
57108  * @extends Roo.util.Observable
57109  * This is the default implementation of a ColumnModel used by the Grid. It defines
57110  * the columns in the grid.
57111  * <br>Usage:<br>
57112  <pre><code>
57113  var colModel = new Roo.grid.ColumnModel([
57114         {header: "Ticker", width: 60, sortable: true, locked: true},
57115         {header: "Company Name", width: 150, sortable: true},
57116         {header: "Market Cap.", width: 100, sortable: true},
57117         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57118         {header: "Employees", width: 100, sortable: true, resizable: false}
57119  ]);
57120  </code></pre>
57121  * <p>
57122  
57123  * The config options listed for this class are options which may appear in each
57124  * individual column definition.
57125  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57126  * @constructor
57127  * @param {Object} config An Array of column config objects. See this class's
57128  * config objects for details.
57129 */
57130 Roo.grid.ColumnModel = function(config){
57131         /**
57132      * The config passed into the constructor
57133      */
57134     this.config = config;
57135     this.lookup = {};
57136
57137     // if no id, create one
57138     // if the column does not have a dataIndex mapping,
57139     // map it to the order it is in the config
57140     for(var i = 0, len = config.length; i < len; i++){
57141         var c = config[i];
57142         if(typeof c.dataIndex == "undefined"){
57143             c.dataIndex = i;
57144         }
57145         if(typeof c.renderer == "string"){
57146             c.renderer = Roo.util.Format[c.renderer];
57147         }
57148         if(typeof c.id == "undefined"){
57149             c.id = Roo.id();
57150         }
57151         if(c.editor && c.editor.xtype){
57152             c.editor  = Roo.factory(c.editor, Roo.grid);
57153         }
57154         if(c.editor && c.editor.isFormField){
57155             c.editor = new Roo.grid.GridEditor(c.editor);
57156         }
57157         this.lookup[c.id] = c;
57158     }
57159
57160     /**
57161      * The width of columns which have no width specified (defaults to 100)
57162      * @type Number
57163      */
57164     this.defaultWidth = 100;
57165
57166     /**
57167      * Default sortable of columns which have no sortable specified (defaults to false)
57168      * @type Boolean
57169      */
57170     this.defaultSortable = false;
57171
57172     this.addEvents({
57173         /**
57174              * @event widthchange
57175              * Fires when the width of a column changes.
57176              * @param {ColumnModel} this
57177              * @param {Number} columnIndex The column index
57178              * @param {Number} newWidth The new width
57179              */
57180             "widthchange": true,
57181         /**
57182              * @event headerchange
57183              * Fires when the text of a header changes.
57184              * @param {ColumnModel} this
57185              * @param {Number} columnIndex The column index
57186              * @param {Number} newText The new header text
57187              */
57188             "headerchange": true,
57189         /**
57190              * @event hiddenchange
57191              * Fires when a column is hidden or "unhidden".
57192              * @param {ColumnModel} this
57193              * @param {Number} columnIndex The column index
57194              * @param {Boolean} hidden true if hidden, false otherwise
57195              */
57196             "hiddenchange": true,
57197             /**
57198          * @event columnmoved
57199          * Fires when a column is moved.
57200          * @param {ColumnModel} this
57201          * @param {Number} oldIndex
57202          * @param {Number} newIndex
57203          */
57204         "columnmoved" : true,
57205         /**
57206          * @event columlockchange
57207          * Fires when a column's locked state is changed
57208          * @param {ColumnModel} this
57209          * @param {Number} colIndex
57210          * @param {Boolean} locked true if locked
57211          */
57212         "columnlockchange" : true
57213     });
57214     Roo.grid.ColumnModel.superclass.constructor.call(this);
57215 };
57216 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57217     /**
57218      * @cfg {String} header The header text to display in the Grid view.
57219      */
57220     /**
57221      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57222      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57223      * specified, the column's index is used as an index into the Record's data Array.
57224      */
57225     /**
57226      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57227      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57228      */
57229     /**
57230      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57231      * Defaults to the value of the {@link #defaultSortable} property.
57232      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57233      */
57234     /**
57235      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57236      */
57237     /**
57238      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57239      */
57240     /**
57241      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57242      */
57243     /**
57244      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57245      */
57246     /**
57247      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57248      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57249      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57250      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57251      */
57252        /**
57253      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57254      */
57255     /**
57256      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57257      */
57258     /**
57259      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57260      */
57261     /**
57262      * @cfg {String} cursor (Optional)
57263      */
57264     /**
57265      * @cfg {String} tooltip (Optional)
57266      */
57267     /**
57268      * @cfg {Number} xs (Optional)
57269      */
57270     /**
57271      * @cfg {Number} sm (Optional)
57272      */
57273     /**
57274      * @cfg {Number} md (Optional)
57275      */
57276     /**
57277      * @cfg {Number} lg (Optional)
57278      */
57279     /**
57280      * Returns the id of the column at the specified index.
57281      * @param {Number} index The column index
57282      * @return {String} the id
57283      */
57284     getColumnId : function(index){
57285         return this.config[index].id;
57286     },
57287
57288     /**
57289      * Returns the column for a specified id.
57290      * @param {String} id The column id
57291      * @return {Object} the column
57292      */
57293     getColumnById : function(id){
57294         return this.lookup[id];
57295     },
57296
57297     
57298     /**
57299      * Returns the column for a specified dataIndex.
57300      * @param {String} dataIndex The column dataIndex
57301      * @return {Object|Boolean} the column or false if not found
57302      */
57303     getColumnByDataIndex: function(dataIndex){
57304         var index = this.findColumnIndex(dataIndex);
57305         return index > -1 ? this.config[index] : false;
57306     },
57307     
57308     /**
57309      * Returns the index for a specified column id.
57310      * @param {String} id The column id
57311      * @return {Number} the index, or -1 if not found
57312      */
57313     getIndexById : function(id){
57314         for(var i = 0, len = this.config.length; i < len; i++){
57315             if(this.config[i].id == id){
57316                 return i;
57317             }
57318         }
57319         return -1;
57320     },
57321     
57322     /**
57323      * Returns the index for a specified column dataIndex.
57324      * @param {String} dataIndex The column dataIndex
57325      * @return {Number} the index, or -1 if not found
57326      */
57327     
57328     findColumnIndex : function(dataIndex){
57329         for(var i = 0, len = this.config.length; i < len; i++){
57330             if(this.config[i].dataIndex == dataIndex){
57331                 return i;
57332             }
57333         }
57334         return -1;
57335     },
57336     
57337     
57338     moveColumn : function(oldIndex, newIndex){
57339         var c = this.config[oldIndex];
57340         this.config.splice(oldIndex, 1);
57341         this.config.splice(newIndex, 0, c);
57342         this.dataMap = null;
57343         this.fireEvent("columnmoved", this, oldIndex, newIndex);
57344     },
57345
57346     isLocked : function(colIndex){
57347         return this.config[colIndex].locked === true;
57348     },
57349
57350     setLocked : function(colIndex, value, suppressEvent){
57351         if(this.isLocked(colIndex) == value){
57352             return;
57353         }
57354         this.config[colIndex].locked = value;
57355         if(!suppressEvent){
57356             this.fireEvent("columnlockchange", this, colIndex, value);
57357         }
57358     },
57359
57360     getTotalLockedWidth : function(){
57361         var totalWidth = 0;
57362         for(var i = 0; i < this.config.length; i++){
57363             if(this.isLocked(i) && !this.isHidden(i)){
57364                 this.totalWidth += this.getColumnWidth(i);
57365             }
57366         }
57367         return totalWidth;
57368     },
57369
57370     getLockedCount : function(){
57371         for(var i = 0, len = this.config.length; i < len; i++){
57372             if(!this.isLocked(i)){
57373                 return i;
57374             }
57375         }
57376         
57377         return this.config.length;
57378     },
57379
57380     /**
57381      * Returns the number of columns.
57382      * @return {Number}
57383      */
57384     getColumnCount : function(visibleOnly){
57385         if(visibleOnly === true){
57386             var c = 0;
57387             for(var i = 0, len = this.config.length; i < len; i++){
57388                 if(!this.isHidden(i)){
57389                     c++;
57390                 }
57391             }
57392             return c;
57393         }
57394         return this.config.length;
57395     },
57396
57397     /**
57398      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
57399      * @param {Function} fn
57400      * @param {Object} scope (optional)
57401      * @return {Array} result
57402      */
57403     getColumnsBy : function(fn, scope){
57404         var r = [];
57405         for(var i = 0, len = this.config.length; i < len; i++){
57406             var c = this.config[i];
57407             if(fn.call(scope||this, c, i) === true){
57408                 r[r.length] = c;
57409             }
57410         }
57411         return r;
57412     },
57413
57414     /**
57415      * Returns true if the specified column is sortable.
57416      * @param {Number} col The column index
57417      * @return {Boolean}
57418      */
57419     isSortable : function(col){
57420         if(typeof this.config[col].sortable == "undefined"){
57421             return this.defaultSortable;
57422         }
57423         return this.config[col].sortable;
57424     },
57425
57426     /**
57427      * Returns the rendering (formatting) function defined for the column.
57428      * @param {Number} col The column index.
57429      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
57430      */
57431     getRenderer : function(col){
57432         if(!this.config[col].renderer){
57433             return Roo.grid.ColumnModel.defaultRenderer;
57434         }
57435         return this.config[col].renderer;
57436     },
57437
57438     /**
57439      * Sets the rendering (formatting) function for a column.
57440      * @param {Number} col The column index
57441      * @param {Function} fn The function to use to process the cell's raw data
57442      * to return HTML markup for the grid view. The render function is called with
57443      * the following parameters:<ul>
57444      * <li>Data value.</li>
57445      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
57446      * <li>css A CSS style string to apply to the table cell.</li>
57447      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
57448      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
57449      * <li>Row index</li>
57450      * <li>Column index</li>
57451      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
57452      */
57453     setRenderer : function(col, fn){
57454         this.config[col].renderer = fn;
57455     },
57456
57457     /**
57458      * Returns the width for the specified column.
57459      * @param {Number} col The column index
57460      * @return {Number}
57461      */
57462     getColumnWidth : function(col){
57463         return this.config[col].width * 1 || this.defaultWidth;
57464     },
57465
57466     /**
57467      * Sets the width for a column.
57468      * @param {Number} col The column index
57469      * @param {Number} width The new width
57470      */
57471     setColumnWidth : function(col, width, suppressEvent){
57472         this.config[col].width = width;
57473         this.totalWidth = null;
57474         if(!suppressEvent){
57475              this.fireEvent("widthchange", this, col, width);
57476         }
57477     },
57478
57479     /**
57480      * Returns the total width of all columns.
57481      * @param {Boolean} includeHidden True to include hidden column widths
57482      * @return {Number}
57483      */
57484     getTotalWidth : function(includeHidden){
57485         if(!this.totalWidth){
57486             this.totalWidth = 0;
57487             for(var i = 0, len = this.config.length; i < len; i++){
57488                 if(includeHidden || !this.isHidden(i)){
57489                     this.totalWidth += this.getColumnWidth(i);
57490                 }
57491             }
57492         }
57493         return this.totalWidth;
57494     },
57495
57496     /**
57497      * Returns the header for the specified column.
57498      * @param {Number} col The column index
57499      * @return {String}
57500      */
57501     getColumnHeader : function(col){
57502         return this.config[col].header;
57503     },
57504
57505     /**
57506      * Sets the header for a column.
57507      * @param {Number} col The column index
57508      * @param {String} header The new header
57509      */
57510     setColumnHeader : function(col, header){
57511         this.config[col].header = header;
57512         this.fireEvent("headerchange", this, col, header);
57513     },
57514
57515     /**
57516      * Returns the tooltip for the specified column.
57517      * @param {Number} col The column index
57518      * @return {String}
57519      */
57520     getColumnTooltip : function(col){
57521             return this.config[col].tooltip;
57522     },
57523     /**
57524      * Sets the tooltip for a column.
57525      * @param {Number} col The column index
57526      * @param {String} tooltip The new tooltip
57527      */
57528     setColumnTooltip : function(col, tooltip){
57529             this.config[col].tooltip = tooltip;
57530     },
57531
57532     /**
57533      * Returns the dataIndex for the specified column.
57534      * @param {Number} col The column index
57535      * @return {Number}
57536      */
57537     getDataIndex : function(col){
57538         return this.config[col].dataIndex;
57539     },
57540
57541     /**
57542      * Sets the dataIndex for a column.
57543      * @param {Number} col The column index
57544      * @param {Number} dataIndex The new dataIndex
57545      */
57546     setDataIndex : function(col, dataIndex){
57547         this.config[col].dataIndex = dataIndex;
57548     },
57549
57550     
57551     
57552     /**
57553      * Returns true if the cell is editable.
57554      * @param {Number} colIndex The column index
57555      * @param {Number} rowIndex The row index - this is nto actually used..?
57556      * @return {Boolean}
57557      */
57558     isCellEditable : function(colIndex, rowIndex){
57559         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
57560     },
57561
57562     /**
57563      * Returns the editor defined for the cell/column.
57564      * return false or null to disable editing.
57565      * @param {Number} colIndex The column index
57566      * @param {Number} rowIndex The row index
57567      * @return {Object}
57568      */
57569     getCellEditor : function(colIndex, rowIndex){
57570         return this.config[colIndex].editor;
57571     },
57572
57573     /**
57574      * Sets if a column is editable.
57575      * @param {Number} col The column index
57576      * @param {Boolean} editable True if the column is editable
57577      */
57578     setEditable : function(col, editable){
57579         this.config[col].editable = editable;
57580     },
57581
57582
57583     /**
57584      * Returns true if the column is hidden.
57585      * @param {Number} colIndex The column index
57586      * @return {Boolean}
57587      */
57588     isHidden : function(colIndex){
57589         return this.config[colIndex].hidden;
57590     },
57591
57592
57593     /**
57594      * Returns true if the column width cannot be changed
57595      */
57596     isFixed : function(colIndex){
57597         return this.config[colIndex].fixed;
57598     },
57599
57600     /**
57601      * Returns true if the column can be resized
57602      * @return {Boolean}
57603      */
57604     isResizable : function(colIndex){
57605         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
57606     },
57607     /**
57608      * Sets if a column is hidden.
57609      * @param {Number} colIndex The column index
57610      * @param {Boolean} hidden True if the column is hidden
57611      */
57612     setHidden : function(colIndex, hidden){
57613         this.config[colIndex].hidden = hidden;
57614         this.totalWidth = null;
57615         this.fireEvent("hiddenchange", this, colIndex, hidden);
57616     },
57617
57618     /**
57619      * Sets the editor for a column.
57620      * @param {Number} col The column index
57621      * @param {Object} editor The editor object
57622      */
57623     setEditor : function(col, editor){
57624         this.config[col].editor = editor;
57625     }
57626 });
57627
57628 Roo.grid.ColumnModel.defaultRenderer = function(value)
57629 {
57630     if(typeof value == "object") {
57631         return value;
57632     }
57633         if(typeof value == "string" && value.length < 1){
57634             return "&#160;";
57635         }
57636     
57637         return String.format("{0}", value);
57638 };
57639
57640 // Alias for backwards compatibility
57641 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
57642 /*
57643  * Based on:
57644  * Ext JS Library 1.1.1
57645  * Copyright(c) 2006-2007, Ext JS, LLC.
57646  *
57647  * Originally Released Under LGPL - original licence link has changed is not relivant.
57648  *
57649  * Fork - LGPL
57650  * <script type="text/javascript">
57651  */
57652
57653 /**
57654  * @class Roo.grid.AbstractSelectionModel
57655  * @extends Roo.util.Observable
57656  * Abstract base class for grid SelectionModels.  It provides the interface that should be
57657  * implemented by descendant classes.  This class should not be directly instantiated.
57658  * @constructor
57659  */
57660 Roo.grid.AbstractSelectionModel = function(){
57661     this.locked = false;
57662     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
57663 };
57664
57665 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
57666     /** @ignore Called by the grid automatically. Do not call directly. */
57667     init : function(grid){
57668         this.grid = grid;
57669         this.initEvents();
57670     },
57671
57672     /**
57673      * Locks the selections.
57674      */
57675     lock : function(){
57676         this.locked = true;
57677     },
57678
57679     /**
57680      * Unlocks the selections.
57681      */
57682     unlock : function(){
57683         this.locked = false;
57684     },
57685
57686     /**
57687      * Returns true if the selections are locked.
57688      * @return {Boolean}
57689      */
57690     isLocked : function(){
57691         return this.locked;
57692     }
57693 });/*
57694  * Based on:
57695  * Ext JS Library 1.1.1
57696  * Copyright(c) 2006-2007, Ext JS, LLC.
57697  *
57698  * Originally Released Under LGPL - original licence link has changed is not relivant.
57699  *
57700  * Fork - LGPL
57701  * <script type="text/javascript">
57702  */
57703 /**
57704  * @extends Roo.grid.AbstractSelectionModel
57705  * @class Roo.grid.RowSelectionModel
57706  * The default SelectionModel used by {@link Roo.grid.Grid}.
57707  * It supports multiple selections and keyboard selection/navigation. 
57708  * @constructor
57709  * @param {Object} config
57710  */
57711 Roo.grid.RowSelectionModel = function(config){
57712     Roo.apply(this, config);
57713     this.selections = new Roo.util.MixedCollection(false, function(o){
57714         return o.id;
57715     });
57716
57717     this.last = false;
57718     this.lastActive = false;
57719
57720     this.addEvents({
57721         /**
57722              * @event selectionchange
57723              * Fires when the selection changes
57724              * @param {SelectionModel} this
57725              */
57726             "selectionchange" : true,
57727         /**
57728              * @event afterselectionchange
57729              * Fires after the selection changes (eg. by key press or clicking)
57730              * @param {SelectionModel} this
57731              */
57732             "afterselectionchange" : true,
57733         /**
57734              * @event beforerowselect
57735              * Fires when a row is selected being selected, return false to cancel.
57736              * @param {SelectionModel} this
57737              * @param {Number} rowIndex The selected index
57738              * @param {Boolean} keepExisting False if other selections will be cleared
57739              */
57740             "beforerowselect" : true,
57741         /**
57742              * @event rowselect
57743              * Fires when a row is selected.
57744              * @param {SelectionModel} this
57745              * @param {Number} rowIndex The selected index
57746              * @param {Roo.data.Record} r The record
57747              */
57748             "rowselect" : true,
57749         /**
57750              * @event rowdeselect
57751              * Fires when a row is deselected.
57752              * @param {SelectionModel} this
57753              * @param {Number} rowIndex The selected index
57754              */
57755         "rowdeselect" : true
57756     });
57757     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
57758     this.locked = false;
57759 };
57760
57761 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
57762     /**
57763      * @cfg {Boolean} singleSelect
57764      * True to allow selection of only one row at a time (defaults to false)
57765      */
57766     singleSelect : false,
57767
57768     // private
57769     initEvents : function(){
57770
57771         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
57772             this.grid.on("mousedown", this.handleMouseDown, this);
57773         }else{ // allow click to work like normal
57774             this.grid.on("rowclick", this.handleDragableRowClick, this);
57775         }
57776
57777         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
57778             "up" : function(e){
57779                 if(!e.shiftKey){
57780                     this.selectPrevious(e.shiftKey);
57781                 }else if(this.last !== false && this.lastActive !== false){
57782                     var last = this.last;
57783                     this.selectRange(this.last,  this.lastActive-1);
57784                     this.grid.getView().focusRow(this.lastActive);
57785                     if(last !== false){
57786                         this.last = last;
57787                     }
57788                 }else{
57789                     this.selectFirstRow();
57790                 }
57791                 this.fireEvent("afterselectionchange", this);
57792             },
57793             "down" : function(e){
57794                 if(!e.shiftKey){
57795                     this.selectNext(e.shiftKey);
57796                 }else if(this.last !== false && this.lastActive !== false){
57797                     var last = this.last;
57798                     this.selectRange(this.last,  this.lastActive+1);
57799                     this.grid.getView().focusRow(this.lastActive);
57800                     if(last !== false){
57801                         this.last = last;
57802                     }
57803                 }else{
57804                     this.selectFirstRow();
57805                 }
57806                 this.fireEvent("afterselectionchange", this);
57807             },
57808             scope: this
57809         });
57810
57811         var view = this.grid.view;
57812         view.on("refresh", this.onRefresh, this);
57813         view.on("rowupdated", this.onRowUpdated, this);
57814         view.on("rowremoved", this.onRemove, this);
57815     },
57816
57817     // private
57818     onRefresh : function(){
57819         var ds = this.grid.dataSource, i, v = this.grid.view;
57820         var s = this.selections;
57821         s.each(function(r){
57822             if((i = ds.indexOfId(r.id)) != -1){
57823                 v.onRowSelect(i);
57824                 s.add(ds.getAt(i)); // updating the selection relate data
57825             }else{
57826                 s.remove(r);
57827             }
57828         });
57829     },
57830
57831     // private
57832     onRemove : function(v, index, r){
57833         this.selections.remove(r);
57834     },
57835
57836     // private
57837     onRowUpdated : function(v, index, r){
57838         if(this.isSelected(r)){
57839             v.onRowSelect(index);
57840         }
57841     },
57842
57843     /**
57844      * Select records.
57845      * @param {Array} records The records to select
57846      * @param {Boolean} keepExisting (optional) True to keep existing selections
57847      */
57848     selectRecords : function(records, keepExisting){
57849         if(!keepExisting){
57850             this.clearSelections();
57851         }
57852         var ds = this.grid.dataSource;
57853         for(var i = 0, len = records.length; i < len; i++){
57854             this.selectRow(ds.indexOf(records[i]), true);
57855         }
57856     },
57857
57858     /**
57859      * Gets the number of selected rows.
57860      * @return {Number}
57861      */
57862     getCount : function(){
57863         return this.selections.length;
57864     },
57865
57866     /**
57867      * Selects the first row in the grid.
57868      */
57869     selectFirstRow : function(){
57870         this.selectRow(0);
57871     },
57872
57873     /**
57874      * Select the last row.
57875      * @param {Boolean} keepExisting (optional) True to keep existing selections
57876      */
57877     selectLastRow : function(keepExisting){
57878         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
57879     },
57880
57881     /**
57882      * Selects the row immediately following the last selected row.
57883      * @param {Boolean} keepExisting (optional) True to keep existing selections
57884      */
57885     selectNext : function(keepExisting){
57886         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
57887             this.selectRow(this.last+1, keepExisting);
57888             this.grid.getView().focusRow(this.last);
57889         }
57890     },
57891
57892     /**
57893      * Selects the row that precedes the last selected row.
57894      * @param {Boolean} keepExisting (optional) True to keep existing selections
57895      */
57896     selectPrevious : function(keepExisting){
57897         if(this.last){
57898             this.selectRow(this.last-1, keepExisting);
57899             this.grid.getView().focusRow(this.last);
57900         }
57901     },
57902
57903     /**
57904      * Returns the selected records
57905      * @return {Array} Array of selected records
57906      */
57907     getSelections : function(){
57908         return [].concat(this.selections.items);
57909     },
57910
57911     /**
57912      * Returns the first selected record.
57913      * @return {Record}
57914      */
57915     getSelected : function(){
57916         return this.selections.itemAt(0);
57917     },
57918
57919
57920     /**
57921      * Clears all selections.
57922      */
57923     clearSelections : function(fast){
57924         if(this.locked) {
57925             return;
57926         }
57927         if(fast !== true){
57928             var ds = this.grid.dataSource;
57929             var s = this.selections;
57930             s.each(function(r){
57931                 this.deselectRow(ds.indexOfId(r.id));
57932             }, this);
57933             s.clear();
57934         }else{
57935             this.selections.clear();
57936         }
57937         this.last = false;
57938     },
57939
57940
57941     /**
57942      * Selects all rows.
57943      */
57944     selectAll : function(){
57945         if(this.locked) {
57946             return;
57947         }
57948         this.selections.clear();
57949         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
57950             this.selectRow(i, true);
57951         }
57952     },
57953
57954     /**
57955      * Returns True if there is a selection.
57956      * @return {Boolean}
57957      */
57958     hasSelection : function(){
57959         return this.selections.length > 0;
57960     },
57961
57962     /**
57963      * Returns True if the specified row is selected.
57964      * @param {Number/Record} record The record or index of the record to check
57965      * @return {Boolean}
57966      */
57967     isSelected : function(index){
57968         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
57969         return (r && this.selections.key(r.id) ? true : false);
57970     },
57971
57972     /**
57973      * Returns True if the specified record id is selected.
57974      * @param {String} id The id of record to check
57975      * @return {Boolean}
57976      */
57977     isIdSelected : function(id){
57978         return (this.selections.key(id) ? true : false);
57979     },
57980
57981     // private
57982     handleMouseDown : function(e, t){
57983         var view = this.grid.getView(), rowIndex;
57984         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
57985             return;
57986         };
57987         if(e.shiftKey && this.last !== false){
57988             var last = this.last;
57989             this.selectRange(last, rowIndex, e.ctrlKey);
57990             this.last = last; // reset the last
57991             view.focusRow(rowIndex);
57992         }else{
57993             var isSelected = this.isSelected(rowIndex);
57994             if(e.button !== 0 && isSelected){
57995                 view.focusRow(rowIndex);
57996             }else if(e.ctrlKey && isSelected){
57997                 this.deselectRow(rowIndex);
57998             }else if(!isSelected){
57999                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58000                 view.focusRow(rowIndex);
58001             }
58002         }
58003         this.fireEvent("afterselectionchange", this);
58004     },
58005     // private
58006     handleDragableRowClick :  function(grid, rowIndex, e) 
58007     {
58008         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58009             this.selectRow(rowIndex, false);
58010             grid.view.focusRow(rowIndex);
58011              this.fireEvent("afterselectionchange", this);
58012         }
58013     },
58014     
58015     /**
58016      * Selects multiple rows.
58017      * @param {Array} rows Array of the indexes of the row to select
58018      * @param {Boolean} keepExisting (optional) True to keep existing selections
58019      */
58020     selectRows : function(rows, keepExisting){
58021         if(!keepExisting){
58022             this.clearSelections();
58023         }
58024         for(var i = 0, len = rows.length; i < len; i++){
58025             this.selectRow(rows[i], true);
58026         }
58027     },
58028
58029     /**
58030      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58031      * @param {Number} startRow The index of the first row in the range
58032      * @param {Number} endRow The index of the last row in the range
58033      * @param {Boolean} keepExisting (optional) True to retain existing selections
58034      */
58035     selectRange : function(startRow, endRow, keepExisting){
58036         if(this.locked) {
58037             return;
58038         }
58039         if(!keepExisting){
58040             this.clearSelections();
58041         }
58042         if(startRow <= endRow){
58043             for(var i = startRow; i <= endRow; i++){
58044                 this.selectRow(i, true);
58045             }
58046         }else{
58047             for(var i = startRow; i >= endRow; i--){
58048                 this.selectRow(i, true);
58049             }
58050         }
58051     },
58052
58053     /**
58054      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58055      * @param {Number} startRow The index of the first row in the range
58056      * @param {Number} endRow The index of the last row in the range
58057      */
58058     deselectRange : function(startRow, endRow, preventViewNotify){
58059         if(this.locked) {
58060             return;
58061         }
58062         for(var i = startRow; i <= endRow; i++){
58063             this.deselectRow(i, preventViewNotify);
58064         }
58065     },
58066
58067     /**
58068      * Selects a row.
58069      * @param {Number} row The index of the row to select
58070      * @param {Boolean} keepExisting (optional) True to keep existing selections
58071      */
58072     selectRow : function(index, keepExisting, preventViewNotify){
58073         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58074             return;
58075         }
58076         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58077             if(!keepExisting || this.singleSelect){
58078                 this.clearSelections();
58079             }
58080             var r = this.grid.dataSource.getAt(index);
58081             this.selections.add(r);
58082             this.last = this.lastActive = index;
58083             if(!preventViewNotify){
58084                 this.grid.getView().onRowSelect(index);
58085             }
58086             this.fireEvent("rowselect", this, index, r);
58087             this.fireEvent("selectionchange", this);
58088         }
58089     },
58090
58091     /**
58092      * Deselects a row.
58093      * @param {Number} row The index of the row to deselect
58094      */
58095     deselectRow : function(index, preventViewNotify){
58096         if(this.locked) {
58097             return;
58098         }
58099         if(this.last == index){
58100             this.last = false;
58101         }
58102         if(this.lastActive == index){
58103             this.lastActive = false;
58104         }
58105         var r = this.grid.dataSource.getAt(index);
58106         this.selections.remove(r);
58107         if(!preventViewNotify){
58108             this.grid.getView().onRowDeselect(index);
58109         }
58110         this.fireEvent("rowdeselect", this, index);
58111         this.fireEvent("selectionchange", this);
58112     },
58113
58114     // private
58115     restoreLast : function(){
58116         if(this._last){
58117             this.last = this._last;
58118         }
58119     },
58120
58121     // private
58122     acceptsNav : function(row, col, cm){
58123         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58124     },
58125
58126     // private
58127     onEditorKey : function(field, e){
58128         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58129         if(k == e.TAB){
58130             e.stopEvent();
58131             ed.completeEdit();
58132             if(e.shiftKey){
58133                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58134             }else{
58135                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58136             }
58137         }else if(k == e.ENTER && !e.ctrlKey){
58138             e.stopEvent();
58139             ed.completeEdit();
58140             if(e.shiftKey){
58141                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58142             }else{
58143                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58144             }
58145         }else if(k == e.ESC){
58146             ed.cancelEdit();
58147         }
58148         if(newCell){
58149             g.startEditing(newCell[0], newCell[1]);
58150         }
58151     }
58152 });/*
58153  * Based on:
58154  * Ext JS Library 1.1.1
58155  * Copyright(c) 2006-2007, Ext JS, LLC.
58156  *
58157  * Originally Released Under LGPL - original licence link has changed is not relivant.
58158  *
58159  * Fork - LGPL
58160  * <script type="text/javascript">
58161  */
58162 /**
58163  * @class Roo.grid.CellSelectionModel
58164  * @extends Roo.grid.AbstractSelectionModel
58165  * This class provides the basic implementation for cell selection in a grid.
58166  * @constructor
58167  * @param {Object} config The object containing the configuration of this model.
58168  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58169  */
58170 Roo.grid.CellSelectionModel = function(config){
58171     Roo.apply(this, config);
58172
58173     this.selection = null;
58174
58175     this.addEvents({
58176         /**
58177              * @event beforerowselect
58178              * Fires before a cell is selected.
58179              * @param {SelectionModel} this
58180              * @param {Number} rowIndex The selected row index
58181              * @param {Number} colIndex The selected cell index
58182              */
58183             "beforecellselect" : true,
58184         /**
58185              * @event cellselect
58186              * Fires when a cell is selected.
58187              * @param {SelectionModel} this
58188              * @param {Number} rowIndex The selected row index
58189              * @param {Number} colIndex The selected cell index
58190              */
58191             "cellselect" : true,
58192         /**
58193              * @event selectionchange
58194              * Fires when the active selection changes.
58195              * @param {SelectionModel} this
58196              * @param {Object} selection null for no selection or an object (o) with two properties
58197                 <ul>
58198                 <li>o.record: the record object for the row the selection is in</li>
58199                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58200                 </ul>
58201              */
58202             "selectionchange" : true,
58203         /**
58204              * @event tabend
58205              * Fires when the tab (or enter) was pressed on the last editable cell
58206              * You can use this to trigger add new row.
58207              * @param {SelectionModel} this
58208              */
58209             "tabend" : true,
58210          /**
58211              * @event beforeeditnext
58212              * Fires before the next editable sell is made active
58213              * You can use this to skip to another cell or fire the tabend
58214              *    if you set cell to false
58215              * @param {Object} eventdata object : { cell : [ row, col ] } 
58216              */
58217             "beforeeditnext" : true
58218     });
58219     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58220 };
58221
58222 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58223     
58224     enter_is_tab: false,
58225
58226     /** @ignore */
58227     initEvents : function(){
58228         this.grid.on("mousedown", this.handleMouseDown, this);
58229         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58230         var view = this.grid.view;
58231         view.on("refresh", this.onViewChange, this);
58232         view.on("rowupdated", this.onRowUpdated, this);
58233         view.on("beforerowremoved", this.clearSelections, this);
58234         view.on("beforerowsinserted", this.clearSelections, this);
58235         if(this.grid.isEditor){
58236             this.grid.on("beforeedit", this.beforeEdit,  this);
58237         }
58238     },
58239
58240         //private
58241     beforeEdit : function(e){
58242         this.select(e.row, e.column, false, true, e.record);
58243     },
58244
58245         //private
58246     onRowUpdated : function(v, index, r){
58247         if(this.selection && this.selection.record == r){
58248             v.onCellSelect(index, this.selection.cell[1]);
58249         }
58250     },
58251
58252         //private
58253     onViewChange : function(){
58254         this.clearSelections(true);
58255     },
58256
58257         /**
58258          * Returns the currently selected cell,.
58259          * @return {Array} The selected cell (row, column) or null if none selected.
58260          */
58261     getSelectedCell : function(){
58262         return this.selection ? this.selection.cell : null;
58263     },
58264
58265     /**
58266      * Clears all selections.
58267      * @param {Boolean} true to prevent the gridview from being notified about the change.
58268      */
58269     clearSelections : function(preventNotify){
58270         var s = this.selection;
58271         if(s){
58272             if(preventNotify !== true){
58273                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
58274             }
58275             this.selection = null;
58276             this.fireEvent("selectionchange", this, null);
58277         }
58278     },
58279
58280     /**
58281      * Returns true if there is a selection.
58282      * @return {Boolean}
58283      */
58284     hasSelection : function(){
58285         return this.selection ? true : false;
58286     },
58287
58288     /** @ignore */
58289     handleMouseDown : function(e, t){
58290         var v = this.grid.getView();
58291         if(this.isLocked()){
58292             return;
58293         };
58294         var row = v.findRowIndex(t);
58295         var cell = v.findCellIndex(t);
58296         if(row !== false && cell !== false){
58297             this.select(row, cell);
58298         }
58299     },
58300
58301     /**
58302      * Selects a cell.
58303      * @param {Number} rowIndex
58304      * @param {Number} collIndex
58305      */
58306     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
58307         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
58308             this.clearSelections();
58309             r = r || this.grid.dataSource.getAt(rowIndex);
58310             this.selection = {
58311                 record : r,
58312                 cell : [rowIndex, colIndex]
58313             };
58314             if(!preventViewNotify){
58315                 var v = this.grid.getView();
58316                 v.onCellSelect(rowIndex, colIndex);
58317                 if(preventFocus !== true){
58318                     v.focusCell(rowIndex, colIndex);
58319                 }
58320             }
58321             this.fireEvent("cellselect", this, rowIndex, colIndex);
58322             this.fireEvent("selectionchange", this, this.selection);
58323         }
58324     },
58325
58326         //private
58327     isSelectable : function(rowIndex, colIndex, cm){
58328         return !cm.isHidden(colIndex);
58329     },
58330
58331     /** @ignore */
58332     handleKeyDown : function(e){
58333         //Roo.log('Cell Sel Model handleKeyDown');
58334         if(!e.isNavKeyPress()){
58335             return;
58336         }
58337         var g = this.grid, s = this.selection;
58338         if(!s){
58339             e.stopEvent();
58340             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
58341             if(cell){
58342                 this.select(cell[0], cell[1]);
58343             }
58344             return;
58345         }
58346         var sm = this;
58347         var walk = function(row, col, step){
58348             return g.walkCells(row, col, step, sm.isSelectable,  sm);
58349         };
58350         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
58351         var newCell;
58352
58353       
58354
58355         switch(k){
58356             case e.TAB:
58357                 // handled by onEditorKey
58358                 if (g.isEditor && g.editing) {
58359                     return;
58360                 }
58361                 if(e.shiftKey) {
58362                     newCell = walk(r, c-1, -1);
58363                 } else {
58364                     newCell = walk(r, c+1, 1);
58365                 }
58366                 break;
58367             
58368             case e.DOWN:
58369                newCell = walk(r+1, c, 1);
58370                 break;
58371             
58372             case e.UP:
58373                 newCell = walk(r-1, c, -1);
58374                 break;
58375             
58376             case e.RIGHT:
58377                 newCell = walk(r, c+1, 1);
58378                 break;
58379             
58380             case e.LEFT:
58381                 newCell = walk(r, c-1, -1);
58382                 break;
58383             
58384             case e.ENTER:
58385                 
58386                 if(g.isEditor && !g.editing){
58387                    g.startEditing(r, c);
58388                    e.stopEvent();
58389                    return;
58390                 }
58391                 
58392                 
58393              break;
58394         };
58395         if(newCell){
58396             this.select(newCell[0], newCell[1]);
58397             e.stopEvent();
58398             
58399         }
58400     },
58401
58402     acceptsNav : function(row, col, cm){
58403         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58404     },
58405     /**
58406      * Selects a cell.
58407      * @param {Number} field (not used) - as it's normally used as a listener
58408      * @param {Number} e - event - fake it by using
58409      *
58410      * var e = Roo.EventObjectImpl.prototype;
58411      * e.keyCode = e.TAB
58412      *
58413      * 
58414      */
58415     onEditorKey : function(field, e){
58416         
58417         var k = e.getKey(),
58418             newCell,
58419             g = this.grid,
58420             ed = g.activeEditor,
58421             forward = false;
58422         ///Roo.log('onEditorKey' + k);
58423         
58424         
58425         if (this.enter_is_tab && k == e.ENTER) {
58426             k = e.TAB;
58427         }
58428         
58429         if(k == e.TAB){
58430             if(e.shiftKey){
58431                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58432             }else{
58433                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58434                 forward = true;
58435             }
58436             
58437             e.stopEvent();
58438             
58439         } else if(k == e.ENTER &&  !e.ctrlKey){
58440             ed.completeEdit();
58441             e.stopEvent();
58442             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58443         
58444                 } else if(k == e.ESC){
58445             ed.cancelEdit();
58446         }
58447                 
58448         if (newCell) {
58449             var ecall = { cell : newCell, forward : forward };
58450             this.fireEvent('beforeeditnext', ecall );
58451             newCell = ecall.cell;
58452                         forward = ecall.forward;
58453         }
58454                 
58455         if(newCell){
58456             //Roo.log('next cell after edit');
58457             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
58458         } else if (forward) {
58459             // tabbed past last
58460             this.fireEvent.defer(100, this, ['tabend',this]);
58461         }
58462     }
58463 });/*
58464  * Based on:
58465  * Ext JS Library 1.1.1
58466  * Copyright(c) 2006-2007, Ext JS, LLC.
58467  *
58468  * Originally Released Under LGPL - original licence link has changed is not relivant.
58469  *
58470  * Fork - LGPL
58471  * <script type="text/javascript">
58472  */
58473  
58474 /**
58475  * @class Roo.grid.EditorGrid
58476  * @extends Roo.grid.Grid
58477  * Class for creating and editable grid.
58478  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
58479  * The container MUST have some type of size defined for the grid to fill. The container will be 
58480  * automatically set to position relative if it isn't already.
58481  * @param {Object} dataSource The data model to bind to
58482  * @param {Object} colModel The column model with info about this grid's columns
58483  */
58484 Roo.grid.EditorGrid = function(container, config){
58485     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
58486     this.getGridEl().addClass("xedit-grid");
58487
58488     if(!this.selModel){
58489         this.selModel = new Roo.grid.CellSelectionModel();
58490     }
58491
58492     this.activeEditor = null;
58493
58494         this.addEvents({
58495             /**
58496              * @event beforeedit
58497              * Fires before cell editing is triggered. The edit event object has the following properties <br />
58498              * <ul style="padding:5px;padding-left:16px;">
58499              * <li>grid - This grid</li>
58500              * <li>record - The record being edited</li>
58501              * <li>field - The field name being edited</li>
58502              * <li>value - The value for the field being edited.</li>
58503              * <li>row - The grid row index</li>
58504              * <li>column - The grid column index</li>
58505              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58506              * </ul>
58507              * @param {Object} e An edit event (see above for description)
58508              */
58509             "beforeedit" : true,
58510             /**
58511              * @event afteredit
58512              * Fires after a cell is edited. <br />
58513              * <ul style="padding:5px;padding-left:16px;">
58514              * <li>grid - This grid</li>
58515              * <li>record - The record being edited</li>
58516              * <li>field - The field name being edited</li>
58517              * <li>value - The value being set</li>
58518              * <li>originalValue - The original value for the field, before the edit.</li>
58519              * <li>row - The grid row index</li>
58520              * <li>column - The grid column index</li>
58521              * </ul>
58522              * @param {Object} e An edit event (see above for description)
58523              */
58524             "afteredit" : true,
58525             /**
58526              * @event validateedit
58527              * Fires after a cell is edited, but before the value is set in the record. 
58528          * You can use this to modify the value being set in the field, Return false
58529              * to cancel the change. The edit event object has the following properties <br />
58530              * <ul style="padding:5px;padding-left:16px;">
58531          * <li>editor - This editor</li>
58532              * <li>grid - This grid</li>
58533              * <li>record - The record being edited</li>
58534              * <li>field - The field name being edited</li>
58535              * <li>value - The value being set</li>
58536              * <li>originalValue - The original value for the field, before the edit.</li>
58537              * <li>row - The grid row index</li>
58538              * <li>column - The grid column index</li>
58539              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
58540              * </ul>
58541              * @param {Object} e An edit event (see above for description)
58542              */
58543             "validateedit" : true
58544         });
58545     this.on("bodyscroll", this.stopEditing,  this);
58546     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
58547 };
58548
58549 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
58550     /**
58551      * @cfg {Number} clicksToEdit
58552      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
58553      */
58554     clicksToEdit: 2,
58555
58556     // private
58557     isEditor : true,
58558     // private
58559     trackMouseOver: false, // causes very odd FF errors
58560
58561     onCellDblClick : function(g, row, col){
58562         this.startEditing(row, col);
58563     },
58564
58565     onEditComplete : function(ed, value, startValue){
58566         this.editing = false;
58567         this.activeEditor = null;
58568         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
58569         var r = ed.record;
58570         var field = this.colModel.getDataIndex(ed.col);
58571         var e = {
58572             grid: this,
58573             record: r,
58574             field: field,
58575             originalValue: startValue,
58576             value: value,
58577             row: ed.row,
58578             column: ed.col,
58579             cancel:false,
58580             editor: ed
58581         };
58582         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
58583         cell.show();
58584           
58585         if(String(value) !== String(startValue)){
58586             
58587             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
58588                 r.set(field, e.value);
58589                 // if we are dealing with a combo box..
58590                 // then we also set the 'name' colum to be the displayField
58591                 if (ed.field.displayField && ed.field.name) {
58592                     r.set(ed.field.name, ed.field.el.dom.value);
58593                 }
58594                 
58595                 delete e.cancel; //?? why!!!
58596                 this.fireEvent("afteredit", e);
58597             }
58598         } else {
58599             this.fireEvent("afteredit", e); // always fire it!
58600         }
58601         this.view.focusCell(ed.row, ed.col);
58602     },
58603
58604     /**
58605      * Starts editing the specified for the specified row/column
58606      * @param {Number} rowIndex
58607      * @param {Number} colIndex
58608      */
58609     startEditing : function(row, col){
58610         this.stopEditing();
58611         if(this.colModel.isCellEditable(col, row)){
58612             this.view.ensureVisible(row, col, true);
58613           
58614             var r = this.dataSource.getAt(row);
58615             var field = this.colModel.getDataIndex(col);
58616             var cell = Roo.get(this.view.getCell(row,col));
58617             var e = {
58618                 grid: this,
58619                 record: r,
58620                 field: field,
58621                 value: r.data[field],
58622                 row: row,
58623                 column: col,
58624                 cancel:false 
58625             };
58626             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
58627                 this.editing = true;
58628                 var ed = this.colModel.getCellEditor(col, row);
58629                 
58630                 if (!ed) {
58631                     return;
58632                 }
58633                 if(!ed.rendered){
58634                     ed.render(ed.parentEl || document.body);
58635                 }
58636                 ed.field.reset();
58637                
58638                 cell.hide();
58639                 
58640                 (function(){ // complex but required for focus issues in safari, ie and opera
58641                     ed.row = row;
58642                     ed.col = col;
58643                     ed.record = r;
58644                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
58645                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
58646                     this.activeEditor = ed;
58647                     var v = r.data[field];
58648                     ed.startEdit(this.view.getCell(row, col), v);
58649                     // combo's with 'displayField and name set
58650                     if (ed.field.displayField && ed.field.name) {
58651                         ed.field.el.dom.value = r.data[ed.field.name];
58652                     }
58653                     
58654                     
58655                 }).defer(50, this);
58656             }
58657         }
58658     },
58659         
58660     /**
58661      * Stops any active editing
58662      */
58663     stopEditing : function(){
58664         if(this.activeEditor){
58665             this.activeEditor.completeEdit();
58666         }
58667         this.activeEditor = null;
58668     },
58669         
58670          /**
58671      * Called to get grid's drag proxy text, by default returns this.ddText.
58672      * @return {String}
58673      */
58674     getDragDropText : function(){
58675         var count = this.selModel.getSelectedCell() ? 1 : 0;
58676         return String.format(this.ddText, count, count == 1 ? '' : 's');
58677     }
58678         
58679 });/*
58680  * Based on:
58681  * Ext JS Library 1.1.1
58682  * Copyright(c) 2006-2007, Ext JS, LLC.
58683  *
58684  * Originally Released Under LGPL - original licence link has changed is not relivant.
58685  *
58686  * Fork - LGPL
58687  * <script type="text/javascript">
58688  */
58689
58690 // private - not really -- you end up using it !
58691 // This is a support class used internally by the Grid components
58692
58693 /**
58694  * @class Roo.grid.GridEditor
58695  * @extends Roo.Editor
58696  * Class for creating and editable grid elements.
58697  * @param {Object} config any settings (must include field)
58698  */
58699 Roo.grid.GridEditor = function(field, config){
58700     if (!config && field.field) {
58701         config = field;
58702         field = Roo.factory(config.field, Roo.form);
58703     }
58704     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
58705     field.monitorTab = false;
58706 };
58707
58708 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
58709     
58710     /**
58711      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
58712      */
58713     
58714     alignment: "tl-tl",
58715     autoSize: "width",
58716     hideEl : false,
58717     cls: "x-small-editor x-grid-editor",
58718     shim:false,
58719     shadow:"frame"
58720 });/*
58721  * Based on:
58722  * Ext JS Library 1.1.1
58723  * Copyright(c) 2006-2007, Ext JS, LLC.
58724  *
58725  * Originally Released Under LGPL - original licence link has changed is not relivant.
58726  *
58727  * Fork - LGPL
58728  * <script type="text/javascript">
58729  */
58730   
58731
58732   
58733 Roo.grid.PropertyRecord = Roo.data.Record.create([
58734     {name:'name',type:'string'},  'value'
58735 ]);
58736
58737
58738 Roo.grid.PropertyStore = function(grid, source){
58739     this.grid = grid;
58740     this.store = new Roo.data.Store({
58741         recordType : Roo.grid.PropertyRecord
58742     });
58743     this.store.on('update', this.onUpdate,  this);
58744     if(source){
58745         this.setSource(source);
58746     }
58747     Roo.grid.PropertyStore.superclass.constructor.call(this);
58748 };
58749
58750
58751
58752 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
58753     setSource : function(o){
58754         this.source = o;
58755         this.store.removeAll();
58756         var data = [];
58757         for(var k in o){
58758             if(this.isEditableValue(o[k])){
58759                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
58760             }
58761         }
58762         this.store.loadRecords({records: data}, {}, true);
58763     },
58764
58765     onUpdate : function(ds, record, type){
58766         if(type == Roo.data.Record.EDIT){
58767             var v = record.data['value'];
58768             var oldValue = record.modified['value'];
58769             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
58770                 this.source[record.id] = v;
58771                 record.commit();
58772                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
58773             }else{
58774                 record.reject();
58775             }
58776         }
58777     },
58778
58779     getProperty : function(row){
58780        return this.store.getAt(row);
58781     },
58782
58783     isEditableValue: function(val){
58784         if(val && val instanceof Date){
58785             return true;
58786         }else if(typeof val == 'object' || typeof val == 'function'){
58787             return false;
58788         }
58789         return true;
58790     },
58791
58792     setValue : function(prop, value){
58793         this.source[prop] = value;
58794         this.store.getById(prop).set('value', value);
58795     },
58796
58797     getSource : function(){
58798         return this.source;
58799     }
58800 });
58801
58802 Roo.grid.PropertyColumnModel = function(grid, store){
58803     this.grid = grid;
58804     var g = Roo.grid;
58805     g.PropertyColumnModel.superclass.constructor.call(this, [
58806         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
58807         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
58808     ]);
58809     this.store = store;
58810     this.bselect = Roo.DomHelper.append(document.body, {
58811         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
58812             {tag: 'option', value: 'true', html: 'true'},
58813             {tag: 'option', value: 'false', html: 'false'}
58814         ]
58815     });
58816     Roo.id(this.bselect);
58817     var f = Roo.form;
58818     this.editors = {
58819         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
58820         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
58821         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
58822         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
58823         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
58824     };
58825     this.renderCellDelegate = this.renderCell.createDelegate(this);
58826     this.renderPropDelegate = this.renderProp.createDelegate(this);
58827 };
58828
58829 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
58830     
58831     
58832     nameText : 'Name',
58833     valueText : 'Value',
58834     
58835     dateFormat : 'm/j/Y',
58836     
58837     
58838     renderDate : function(dateVal){
58839         return dateVal.dateFormat(this.dateFormat);
58840     },
58841
58842     renderBool : function(bVal){
58843         return bVal ? 'true' : 'false';
58844     },
58845
58846     isCellEditable : function(colIndex, rowIndex){
58847         return colIndex == 1;
58848     },
58849
58850     getRenderer : function(col){
58851         return col == 1 ?
58852             this.renderCellDelegate : this.renderPropDelegate;
58853     },
58854
58855     renderProp : function(v){
58856         return this.getPropertyName(v);
58857     },
58858
58859     renderCell : function(val){
58860         var rv = val;
58861         if(val instanceof Date){
58862             rv = this.renderDate(val);
58863         }else if(typeof val == 'boolean'){
58864             rv = this.renderBool(val);
58865         }
58866         return Roo.util.Format.htmlEncode(rv);
58867     },
58868
58869     getPropertyName : function(name){
58870         var pn = this.grid.propertyNames;
58871         return pn && pn[name] ? pn[name] : name;
58872     },
58873
58874     getCellEditor : function(colIndex, rowIndex){
58875         var p = this.store.getProperty(rowIndex);
58876         var n = p.data['name'], val = p.data['value'];
58877         
58878         if(typeof(this.grid.customEditors[n]) == 'string'){
58879             return this.editors[this.grid.customEditors[n]];
58880         }
58881         if(typeof(this.grid.customEditors[n]) != 'undefined'){
58882             return this.grid.customEditors[n];
58883         }
58884         if(val instanceof Date){
58885             return this.editors['date'];
58886         }else if(typeof val == 'number'){
58887             return this.editors['number'];
58888         }else if(typeof val == 'boolean'){
58889             return this.editors['boolean'];
58890         }else{
58891             return this.editors['string'];
58892         }
58893     }
58894 });
58895
58896 /**
58897  * @class Roo.grid.PropertyGrid
58898  * @extends Roo.grid.EditorGrid
58899  * This class represents the  interface of a component based property grid control.
58900  * <br><br>Usage:<pre><code>
58901  var grid = new Roo.grid.PropertyGrid("my-container-id", {
58902       
58903  });
58904  // set any options
58905  grid.render();
58906  * </code></pre>
58907   
58908  * @constructor
58909  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
58910  * The container MUST have some type of size defined for the grid to fill. The container will be
58911  * automatically set to position relative if it isn't already.
58912  * @param {Object} config A config object that sets properties on this grid.
58913  */
58914 Roo.grid.PropertyGrid = function(container, config){
58915     config = config || {};
58916     var store = new Roo.grid.PropertyStore(this);
58917     this.store = store;
58918     var cm = new Roo.grid.PropertyColumnModel(this, store);
58919     store.store.sort('name', 'ASC');
58920     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
58921         ds: store.store,
58922         cm: cm,
58923         enableColLock:false,
58924         enableColumnMove:false,
58925         stripeRows:false,
58926         trackMouseOver: false,
58927         clicksToEdit:1
58928     }, config));
58929     this.getGridEl().addClass('x-props-grid');
58930     this.lastEditRow = null;
58931     this.on('columnresize', this.onColumnResize, this);
58932     this.addEvents({
58933          /**
58934              * @event beforepropertychange
58935              * Fires before a property changes (return false to stop?)
58936              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
58937              * @param {String} id Record Id
58938              * @param {String} newval New Value
58939          * @param {String} oldval Old Value
58940              */
58941         "beforepropertychange": true,
58942         /**
58943              * @event propertychange
58944              * Fires after a property changes
58945              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
58946              * @param {String} id Record Id
58947              * @param {String} newval New Value
58948          * @param {String} oldval Old Value
58949              */
58950         "propertychange": true
58951     });
58952     this.customEditors = this.customEditors || {};
58953 };
58954 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
58955     
58956      /**
58957      * @cfg {Object} customEditors map of colnames=> custom editors.
58958      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
58959      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
58960      * false disables editing of the field.
58961          */
58962     
58963       /**
58964      * @cfg {Object} propertyNames map of property Names to their displayed value
58965          */
58966     
58967     render : function(){
58968         Roo.grid.PropertyGrid.superclass.render.call(this);
58969         this.autoSize.defer(100, this);
58970     },
58971
58972     autoSize : function(){
58973         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
58974         if(this.view){
58975             this.view.fitColumns();
58976         }
58977     },
58978
58979     onColumnResize : function(){
58980         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
58981         this.autoSize();
58982     },
58983     /**
58984      * Sets the data for the Grid
58985      * accepts a Key => Value object of all the elements avaiable.
58986      * @param {Object} data  to appear in grid.
58987      */
58988     setSource : function(source){
58989         this.store.setSource(source);
58990         //this.autoSize();
58991     },
58992     /**
58993      * Gets all the data from the grid.
58994      * @return {Object} data  data stored in grid
58995      */
58996     getSource : function(){
58997         return this.store.getSource();
58998     }
58999 });/*
59000   
59001  * Licence LGPL
59002  
59003  */
59004  
59005 /**
59006  * @class Roo.grid.Calendar
59007  * @extends Roo.util.Grid
59008  * This class extends the Grid to provide a calendar widget
59009  * <br><br>Usage:<pre><code>
59010  var grid = new Roo.grid.Calendar("my-container-id", {
59011      ds: myDataStore,
59012      cm: myColModel,
59013      selModel: mySelectionModel,
59014      autoSizeColumns: true,
59015      monitorWindowResize: false,
59016      trackMouseOver: true
59017      eventstore : real data store..
59018  });
59019  // set any options
59020  grid.render();
59021   
59022   * @constructor
59023  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59024  * The container MUST have some type of size defined for the grid to fill. The container will be
59025  * automatically set to position relative if it isn't already.
59026  * @param {Object} config A config object that sets properties on this grid.
59027  */
59028 Roo.grid.Calendar = function(container, config){
59029         // initialize the container
59030         this.container = Roo.get(container);
59031         this.container.update("");
59032         this.container.setStyle("overflow", "hidden");
59033     this.container.addClass('x-grid-container');
59034
59035     this.id = this.container.id;
59036
59037     Roo.apply(this, config);
59038     // check and correct shorthanded configs
59039     
59040     var rows = [];
59041     var d =1;
59042     for (var r = 0;r < 6;r++) {
59043         
59044         rows[r]=[];
59045         for (var c =0;c < 7;c++) {
59046             rows[r][c]= '';
59047         }
59048     }
59049     if (this.eventStore) {
59050         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59051         this.eventStore.on('load',this.onLoad, this);
59052         this.eventStore.on('beforeload',this.clearEvents, this);
59053          
59054     }
59055     
59056     this.dataSource = new Roo.data.Store({
59057             proxy: new Roo.data.MemoryProxy(rows),
59058             reader: new Roo.data.ArrayReader({}, [
59059                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59060     });
59061
59062     this.dataSource.load();
59063     this.ds = this.dataSource;
59064     this.ds.xmodule = this.xmodule || false;
59065     
59066     
59067     var cellRender = function(v,x,r)
59068     {
59069         return String.format(
59070             '<div class="fc-day  fc-widget-content"><div>' +
59071                 '<div class="fc-event-container"></div>' +
59072                 '<div class="fc-day-number">{0}</div>'+
59073                 
59074                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59075             '</div></div>', v);
59076     
59077     }
59078     
59079     
59080     this.colModel = new Roo.grid.ColumnModel( [
59081         {
59082             xtype: 'ColumnModel',
59083             xns: Roo.grid,
59084             dataIndex : 'weekday0',
59085             header : 'Sunday',
59086             renderer : cellRender
59087         },
59088         {
59089             xtype: 'ColumnModel',
59090             xns: Roo.grid,
59091             dataIndex : 'weekday1',
59092             header : 'Monday',
59093             renderer : cellRender
59094         },
59095         {
59096             xtype: 'ColumnModel',
59097             xns: Roo.grid,
59098             dataIndex : 'weekday2',
59099             header : 'Tuesday',
59100             renderer : cellRender
59101         },
59102         {
59103             xtype: 'ColumnModel',
59104             xns: Roo.grid,
59105             dataIndex : 'weekday3',
59106             header : 'Wednesday',
59107             renderer : cellRender
59108         },
59109         {
59110             xtype: 'ColumnModel',
59111             xns: Roo.grid,
59112             dataIndex : 'weekday4',
59113             header : 'Thursday',
59114             renderer : cellRender
59115         },
59116         {
59117             xtype: 'ColumnModel',
59118             xns: Roo.grid,
59119             dataIndex : 'weekday5',
59120             header : 'Friday',
59121             renderer : cellRender
59122         },
59123         {
59124             xtype: 'ColumnModel',
59125             xns: Roo.grid,
59126             dataIndex : 'weekday6',
59127             header : 'Saturday',
59128             renderer : cellRender
59129         }
59130     ]);
59131     this.cm = this.colModel;
59132     this.cm.xmodule = this.xmodule || false;
59133  
59134         
59135           
59136     //this.selModel = new Roo.grid.CellSelectionModel();
59137     //this.sm = this.selModel;
59138     //this.selModel.init(this);
59139     
59140     
59141     if(this.width){
59142         this.container.setWidth(this.width);
59143     }
59144
59145     if(this.height){
59146         this.container.setHeight(this.height);
59147     }
59148     /** @private */
59149         this.addEvents({
59150         // raw events
59151         /**
59152          * @event click
59153          * The raw click event for the entire grid.
59154          * @param {Roo.EventObject} e
59155          */
59156         "click" : true,
59157         /**
59158          * @event dblclick
59159          * The raw dblclick event for the entire grid.
59160          * @param {Roo.EventObject} e
59161          */
59162         "dblclick" : true,
59163         /**
59164          * @event contextmenu
59165          * The raw contextmenu event for the entire grid.
59166          * @param {Roo.EventObject} e
59167          */
59168         "contextmenu" : true,
59169         /**
59170          * @event mousedown
59171          * The raw mousedown event for the entire grid.
59172          * @param {Roo.EventObject} e
59173          */
59174         "mousedown" : true,
59175         /**
59176          * @event mouseup
59177          * The raw mouseup event for the entire grid.
59178          * @param {Roo.EventObject} e
59179          */
59180         "mouseup" : true,
59181         /**
59182          * @event mouseover
59183          * The raw mouseover event for the entire grid.
59184          * @param {Roo.EventObject} e
59185          */
59186         "mouseover" : true,
59187         /**
59188          * @event mouseout
59189          * The raw mouseout event for the entire grid.
59190          * @param {Roo.EventObject} e
59191          */
59192         "mouseout" : true,
59193         /**
59194          * @event keypress
59195          * The raw keypress event for the entire grid.
59196          * @param {Roo.EventObject} e
59197          */
59198         "keypress" : true,
59199         /**
59200          * @event keydown
59201          * The raw keydown event for the entire grid.
59202          * @param {Roo.EventObject} e
59203          */
59204         "keydown" : true,
59205
59206         // custom events
59207
59208         /**
59209          * @event cellclick
59210          * Fires when a cell is clicked
59211          * @param {Grid} this
59212          * @param {Number} rowIndex
59213          * @param {Number} columnIndex
59214          * @param {Roo.EventObject} e
59215          */
59216         "cellclick" : true,
59217         /**
59218          * @event celldblclick
59219          * Fires when a cell is double clicked
59220          * @param {Grid} this
59221          * @param {Number} rowIndex
59222          * @param {Number} columnIndex
59223          * @param {Roo.EventObject} e
59224          */
59225         "celldblclick" : true,
59226         /**
59227          * @event rowclick
59228          * Fires when a row is clicked
59229          * @param {Grid} this
59230          * @param {Number} rowIndex
59231          * @param {Roo.EventObject} e
59232          */
59233         "rowclick" : true,
59234         /**
59235          * @event rowdblclick
59236          * Fires when a row is double clicked
59237          * @param {Grid} this
59238          * @param {Number} rowIndex
59239          * @param {Roo.EventObject} e
59240          */
59241         "rowdblclick" : true,
59242         /**
59243          * @event headerclick
59244          * Fires when a header is clicked
59245          * @param {Grid} this
59246          * @param {Number} columnIndex
59247          * @param {Roo.EventObject} e
59248          */
59249         "headerclick" : true,
59250         /**
59251          * @event headerdblclick
59252          * Fires when a header cell is double clicked
59253          * @param {Grid} this
59254          * @param {Number} columnIndex
59255          * @param {Roo.EventObject} e
59256          */
59257         "headerdblclick" : true,
59258         /**
59259          * @event rowcontextmenu
59260          * Fires when a row is right clicked
59261          * @param {Grid} this
59262          * @param {Number} rowIndex
59263          * @param {Roo.EventObject} e
59264          */
59265         "rowcontextmenu" : true,
59266         /**
59267          * @event cellcontextmenu
59268          * Fires when a cell is right clicked
59269          * @param {Grid} this
59270          * @param {Number} rowIndex
59271          * @param {Number} cellIndex
59272          * @param {Roo.EventObject} e
59273          */
59274          "cellcontextmenu" : true,
59275         /**
59276          * @event headercontextmenu
59277          * Fires when a header is right clicked
59278          * @param {Grid} this
59279          * @param {Number} columnIndex
59280          * @param {Roo.EventObject} e
59281          */
59282         "headercontextmenu" : true,
59283         /**
59284          * @event bodyscroll
59285          * Fires when the body element is scrolled
59286          * @param {Number} scrollLeft
59287          * @param {Number} scrollTop
59288          */
59289         "bodyscroll" : true,
59290         /**
59291          * @event columnresize
59292          * Fires when the user resizes a column
59293          * @param {Number} columnIndex
59294          * @param {Number} newSize
59295          */
59296         "columnresize" : true,
59297         /**
59298          * @event columnmove
59299          * Fires when the user moves a column
59300          * @param {Number} oldIndex
59301          * @param {Number} newIndex
59302          */
59303         "columnmove" : true,
59304         /**
59305          * @event startdrag
59306          * Fires when row(s) start being dragged
59307          * @param {Grid} this
59308          * @param {Roo.GridDD} dd The drag drop object
59309          * @param {event} e The raw browser event
59310          */
59311         "startdrag" : true,
59312         /**
59313          * @event enddrag
59314          * Fires when a drag operation is complete
59315          * @param {Grid} this
59316          * @param {Roo.GridDD} dd The drag drop object
59317          * @param {event} e The raw browser event
59318          */
59319         "enddrag" : true,
59320         /**
59321          * @event dragdrop
59322          * Fires when dragged row(s) are dropped on a valid DD target
59323          * @param {Grid} this
59324          * @param {Roo.GridDD} dd The drag drop object
59325          * @param {String} targetId The target drag drop object
59326          * @param {event} e The raw browser event
59327          */
59328         "dragdrop" : true,
59329         /**
59330          * @event dragover
59331          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
59332          * @param {Grid} this
59333          * @param {Roo.GridDD} dd The drag drop object
59334          * @param {String} targetId The target drag drop object
59335          * @param {event} e The raw browser event
59336          */
59337         "dragover" : true,
59338         /**
59339          * @event dragenter
59340          *  Fires when the dragged row(s) first cross another DD target while being dragged
59341          * @param {Grid} this
59342          * @param {Roo.GridDD} dd The drag drop object
59343          * @param {String} targetId The target drag drop object
59344          * @param {event} e The raw browser event
59345          */
59346         "dragenter" : true,
59347         /**
59348          * @event dragout
59349          * Fires when the dragged row(s) leave another DD target while being dragged
59350          * @param {Grid} this
59351          * @param {Roo.GridDD} dd The drag drop object
59352          * @param {String} targetId The target drag drop object
59353          * @param {event} e The raw browser event
59354          */
59355         "dragout" : true,
59356         /**
59357          * @event rowclass
59358          * Fires when a row is rendered, so you can change add a style to it.
59359          * @param {GridView} gridview   The grid view
59360          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
59361          */
59362         'rowclass' : true,
59363
59364         /**
59365          * @event render
59366          * Fires when the grid is rendered
59367          * @param {Grid} grid
59368          */
59369         'render' : true,
59370             /**
59371              * @event select
59372              * Fires when a date is selected
59373              * @param {DatePicker} this
59374              * @param {Date} date The selected date
59375              */
59376         'select': true,
59377         /**
59378              * @event monthchange
59379              * Fires when the displayed month changes 
59380              * @param {DatePicker} this
59381              * @param {Date} date The selected month
59382              */
59383         'monthchange': true,
59384         /**
59385              * @event evententer
59386              * Fires when mouse over an event
59387              * @param {Calendar} this
59388              * @param {event} Event
59389              */
59390         'evententer': true,
59391         /**
59392              * @event eventleave
59393              * Fires when the mouse leaves an
59394              * @param {Calendar} this
59395              * @param {event}
59396              */
59397         'eventleave': true,
59398         /**
59399              * @event eventclick
59400              * Fires when the mouse click an
59401              * @param {Calendar} this
59402              * @param {event}
59403              */
59404         'eventclick': true,
59405         /**
59406              * @event eventrender
59407              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
59408              * @param {Calendar} this
59409              * @param {data} data to be modified
59410              */
59411         'eventrender': true
59412         
59413     });
59414
59415     Roo.grid.Grid.superclass.constructor.call(this);
59416     this.on('render', function() {
59417         this.view.el.addClass('x-grid-cal'); 
59418         
59419         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
59420
59421     },this);
59422     
59423     if (!Roo.grid.Calendar.style) {
59424         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
59425             
59426             
59427             '.x-grid-cal .x-grid-col' :  {
59428                 height: 'auto !important',
59429                 'vertical-align': 'top'
59430             },
59431             '.x-grid-cal  .fc-event-hori' : {
59432                 height: '14px'
59433             }
59434              
59435             
59436         }, Roo.id());
59437     }
59438
59439     
59440     
59441 };
59442 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
59443     /**
59444      * @cfg {Store} eventStore The store that loads events.
59445      */
59446     eventStore : 25,
59447
59448      
59449     activeDate : false,
59450     startDay : 0,
59451     autoWidth : true,
59452     monitorWindowResize : false,
59453
59454     
59455     resizeColumns : function() {
59456         var col = (this.view.el.getWidth() / 7) - 3;
59457         // loop through cols, and setWidth
59458         for(var i =0 ; i < 7 ; i++){
59459             this.cm.setColumnWidth(i, col);
59460         }
59461     },
59462      setDate :function(date) {
59463         
59464         Roo.log('setDate?');
59465         
59466         this.resizeColumns();
59467         var vd = this.activeDate;
59468         this.activeDate = date;
59469 //        if(vd && this.el){
59470 //            var t = date.getTime();
59471 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
59472 //                Roo.log('using add remove');
59473 //                
59474 //                this.fireEvent('monthchange', this, date);
59475 //                
59476 //                this.cells.removeClass("fc-state-highlight");
59477 //                this.cells.each(function(c){
59478 //                   if(c.dateValue == t){
59479 //                       c.addClass("fc-state-highlight");
59480 //                       setTimeout(function(){
59481 //                            try{c.dom.firstChild.focus();}catch(e){}
59482 //                       }, 50);
59483 //                       return false;
59484 //                   }
59485 //                   return true;
59486 //                });
59487 //                return;
59488 //            }
59489 //        }
59490         
59491         var days = date.getDaysInMonth();
59492         
59493         var firstOfMonth = date.getFirstDateOfMonth();
59494         var startingPos = firstOfMonth.getDay()-this.startDay;
59495         
59496         if(startingPos < this.startDay){
59497             startingPos += 7;
59498         }
59499         
59500         var pm = date.add(Date.MONTH, -1);
59501         var prevStart = pm.getDaysInMonth()-startingPos;
59502 //        
59503         
59504         
59505         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59506         
59507         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
59508         //this.cells.addClassOnOver('fc-state-hover');
59509         
59510         var cells = this.cells.elements;
59511         var textEls = this.textNodes;
59512         
59513         //Roo.each(cells, function(cell){
59514         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
59515         //});
59516         
59517         days += startingPos;
59518
59519         // convert everything to numbers so it's fast
59520         var day = 86400000;
59521         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
59522         //Roo.log(d);
59523         //Roo.log(pm);
59524         //Roo.log(prevStart);
59525         
59526         var today = new Date().clearTime().getTime();
59527         var sel = date.clearTime().getTime();
59528         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
59529         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
59530         var ddMatch = this.disabledDatesRE;
59531         var ddText = this.disabledDatesText;
59532         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
59533         var ddaysText = this.disabledDaysText;
59534         var format = this.format;
59535         
59536         var setCellClass = function(cal, cell){
59537             
59538             //Roo.log('set Cell Class');
59539             cell.title = "";
59540             var t = d.getTime();
59541             
59542             //Roo.log(d);
59543             
59544             
59545             cell.dateValue = t;
59546             if(t == today){
59547                 cell.className += " fc-today";
59548                 cell.className += " fc-state-highlight";
59549                 cell.title = cal.todayText;
59550             }
59551             if(t == sel){
59552                 // disable highlight in other month..
59553                 cell.className += " fc-state-highlight";
59554                 
59555             }
59556             // disabling
59557             if(t < min) {
59558                 //cell.className = " fc-state-disabled";
59559                 cell.title = cal.minText;
59560                 return;
59561             }
59562             if(t > max) {
59563                 //cell.className = " fc-state-disabled";
59564                 cell.title = cal.maxText;
59565                 return;
59566             }
59567             if(ddays){
59568                 if(ddays.indexOf(d.getDay()) != -1){
59569                     // cell.title = ddaysText;
59570                    // cell.className = " fc-state-disabled";
59571                 }
59572             }
59573             if(ddMatch && format){
59574                 var fvalue = d.dateFormat(format);
59575                 if(ddMatch.test(fvalue)){
59576                     cell.title = ddText.replace("%0", fvalue);
59577                    cell.className = " fc-state-disabled";
59578                 }
59579             }
59580             
59581             if (!cell.initialClassName) {
59582                 cell.initialClassName = cell.dom.className;
59583             }
59584             
59585             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
59586         };
59587
59588         var i = 0;
59589         
59590         for(; i < startingPos; i++) {
59591             cells[i].dayName =  (++prevStart);
59592             Roo.log(textEls[i]);
59593             d.setDate(d.getDate()+1);
59594             
59595             //cells[i].className = "fc-past fc-other-month";
59596             setCellClass(this, cells[i]);
59597         }
59598         
59599         var intDay = 0;
59600         
59601         for(; i < days; i++){
59602             intDay = i - startingPos + 1;
59603             cells[i].dayName =  (intDay);
59604             d.setDate(d.getDate()+1);
59605             
59606             cells[i].className = ''; // "x-date-active";
59607             setCellClass(this, cells[i]);
59608         }
59609         var extraDays = 0;
59610         
59611         for(; i < 42; i++) {
59612             //textEls[i].innerHTML = (++extraDays);
59613             
59614             d.setDate(d.getDate()+1);
59615             cells[i].dayName = (++extraDays);
59616             cells[i].className = "fc-future fc-other-month";
59617             setCellClass(this, cells[i]);
59618         }
59619         
59620         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
59621         
59622         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
59623         
59624         // this will cause all the cells to mis
59625         var rows= [];
59626         var i =0;
59627         for (var r = 0;r < 6;r++) {
59628             for (var c =0;c < 7;c++) {
59629                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
59630             }    
59631         }
59632         
59633         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
59634         for(i=0;i<cells.length;i++) {
59635             
59636             this.cells.elements[i].dayName = cells[i].dayName ;
59637             this.cells.elements[i].className = cells[i].className;
59638             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
59639             this.cells.elements[i].title = cells[i].title ;
59640             this.cells.elements[i].dateValue = cells[i].dateValue ;
59641         }
59642         
59643         
59644         
59645         
59646         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
59647         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
59648         
59649         ////if(totalRows != 6){
59650             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
59651            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
59652        // }
59653         
59654         this.fireEvent('monthchange', this, date);
59655         
59656         
59657     },
59658  /**
59659      * Returns the grid's SelectionModel.
59660      * @return {SelectionModel}
59661      */
59662     getSelectionModel : function(){
59663         if(!this.selModel){
59664             this.selModel = new Roo.grid.CellSelectionModel();
59665         }
59666         return this.selModel;
59667     },
59668
59669     load: function() {
59670         this.eventStore.load()
59671         
59672         
59673         
59674     },
59675     
59676     findCell : function(dt) {
59677         dt = dt.clearTime().getTime();
59678         var ret = false;
59679         this.cells.each(function(c){
59680             //Roo.log("check " +c.dateValue + '?=' + dt);
59681             if(c.dateValue == dt){
59682                 ret = c;
59683                 return false;
59684             }
59685             return true;
59686         });
59687         
59688         return ret;
59689     },
59690     
59691     findCells : function(rec) {
59692         var s = rec.data.start_dt.clone().clearTime().getTime();
59693        // Roo.log(s);
59694         var e= rec.data.end_dt.clone().clearTime().getTime();
59695        // Roo.log(e);
59696         var ret = [];
59697         this.cells.each(function(c){
59698              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
59699             
59700             if(c.dateValue > e){
59701                 return ;
59702             }
59703             if(c.dateValue < s){
59704                 return ;
59705             }
59706             ret.push(c);
59707         });
59708         
59709         return ret;    
59710     },
59711     
59712     findBestRow: function(cells)
59713     {
59714         var ret = 0;
59715         
59716         for (var i =0 ; i < cells.length;i++) {
59717             ret  = Math.max(cells[i].rows || 0,ret);
59718         }
59719         return ret;
59720         
59721     },
59722     
59723     
59724     addItem : function(rec)
59725     {
59726         // look for vertical location slot in
59727         var cells = this.findCells(rec);
59728         
59729         rec.row = this.findBestRow(cells);
59730         
59731         // work out the location.
59732         
59733         var crow = false;
59734         var rows = [];
59735         for(var i =0; i < cells.length; i++) {
59736             if (!crow) {
59737                 crow = {
59738                     start : cells[i],
59739                     end :  cells[i]
59740                 };
59741                 continue;
59742             }
59743             if (crow.start.getY() == cells[i].getY()) {
59744                 // on same row.
59745                 crow.end = cells[i];
59746                 continue;
59747             }
59748             // different row.
59749             rows.push(crow);
59750             crow = {
59751                 start: cells[i],
59752                 end : cells[i]
59753             };
59754             
59755         }
59756         
59757         rows.push(crow);
59758         rec.els = [];
59759         rec.rows = rows;
59760         rec.cells = cells;
59761         for (var i = 0; i < cells.length;i++) {
59762             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
59763             
59764         }
59765         
59766         
59767     },
59768     
59769     clearEvents: function() {
59770         
59771         if (!this.eventStore.getCount()) {
59772             return;
59773         }
59774         // reset number of rows in cells.
59775         Roo.each(this.cells.elements, function(c){
59776             c.rows = 0;
59777         });
59778         
59779         this.eventStore.each(function(e) {
59780             this.clearEvent(e);
59781         },this);
59782         
59783     },
59784     
59785     clearEvent : function(ev)
59786     {
59787         if (ev.els) {
59788             Roo.each(ev.els, function(el) {
59789                 el.un('mouseenter' ,this.onEventEnter, this);
59790                 el.un('mouseleave' ,this.onEventLeave, this);
59791                 el.remove();
59792             },this);
59793             ev.els = [];
59794         }
59795     },
59796     
59797     
59798     renderEvent : function(ev,ctr) {
59799         if (!ctr) {
59800              ctr = this.view.el.select('.fc-event-container',true).first();
59801         }
59802         
59803          
59804         this.clearEvent(ev);
59805             //code
59806        
59807         
59808         
59809         ev.els = [];
59810         var cells = ev.cells;
59811         var rows = ev.rows;
59812         this.fireEvent('eventrender', this, ev);
59813         
59814         for(var i =0; i < rows.length; i++) {
59815             
59816             cls = '';
59817             if (i == 0) {
59818                 cls += ' fc-event-start';
59819             }
59820             if ((i+1) == rows.length) {
59821                 cls += ' fc-event-end';
59822             }
59823             
59824             //Roo.log(ev.data);
59825             // how many rows should it span..
59826             var cg = this.eventTmpl.append(ctr,Roo.apply({
59827                 fccls : cls
59828                 
59829             }, ev.data) , true);
59830             
59831             
59832             cg.on('mouseenter' ,this.onEventEnter, this, ev);
59833             cg.on('mouseleave' ,this.onEventLeave, this, ev);
59834             cg.on('click', this.onEventClick, this, ev);
59835             
59836             ev.els.push(cg);
59837             
59838             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
59839             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
59840             //Roo.log(cg);
59841              
59842             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
59843             cg.setWidth(ebox.right - sbox.x -2);
59844         }
59845     },
59846     
59847     renderEvents: function()
59848     {   
59849         // first make sure there is enough space..
59850         
59851         if (!this.eventTmpl) {
59852             this.eventTmpl = new Roo.Template(
59853                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
59854                     '<div class="fc-event-inner">' +
59855                         '<span class="fc-event-time">{time}</span>' +
59856                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
59857                     '</div>' +
59858                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
59859                 '</div>'
59860             );
59861                 
59862         }
59863                
59864         
59865         
59866         this.cells.each(function(c) {
59867             //Roo.log(c.select('.fc-day-content div',true).first());
59868             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
59869         });
59870         
59871         var ctr = this.view.el.select('.fc-event-container',true).first();
59872         
59873         var cls;
59874         this.eventStore.each(function(ev){
59875             
59876             this.renderEvent(ev);
59877              
59878              
59879         }, this);
59880         this.view.layout();
59881         
59882     },
59883     
59884     onEventEnter: function (e, el,event,d) {
59885         this.fireEvent('evententer', this, el, event);
59886     },
59887     
59888     onEventLeave: function (e, el,event,d) {
59889         this.fireEvent('eventleave', this, el, event);
59890     },
59891     
59892     onEventClick: function (e, el,event,d) {
59893         this.fireEvent('eventclick', this, el, event);
59894     },
59895     
59896     onMonthChange: function () {
59897         this.store.load();
59898     },
59899     
59900     onLoad: function () {
59901         
59902         //Roo.log('calendar onload');
59903 //         
59904         if(this.eventStore.getCount() > 0){
59905             
59906            
59907             
59908             this.eventStore.each(function(d){
59909                 
59910                 
59911                 // FIXME..
59912                 var add =   d.data;
59913                 if (typeof(add.end_dt) == 'undefined')  {
59914                     Roo.log("Missing End time in calendar data: ");
59915                     Roo.log(d);
59916                     return;
59917                 }
59918                 if (typeof(add.start_dt) == 'undefined')  {
59919                     Roo.log("Missing Start time in calendar data: ");
59920                     Roo.log(d);
59921                     return;
59922                 }
59923                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
59924                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
59925                 add.id = add.id || d.id;
59926                 add.title = add.title || '??';
59927                 
59928                 this.addItem(d);
59929                 
59930              
59931             },this);
59932         }
59933         
59934         this.renderEvents();
59935     }
59936     
59937
59938 });
59939 /*
59940  grid : {
59941                 xtype: 'Grid',
59942                 xns: Roo.grid,
59943                 listeners : {
59944                     render : function ()
59945                     {
59946                         _this.grid = this;
59947                         
59948                         if (!this.view.el.hasClass('course-timesheet')) {
59949                             this.view.el.addClass('course-timesheet');
59950                         }
59951                         if (this.tsStyle) {
59952                             this.ds.load({});
59953                             return; 
59954                         }
59955                         Roo.log('width');
59956                         Roo.log(_this.grid.view.el.getWidth());
59957                         
59958                         
59959                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
59960                             '.course-timesheet .x-grid-row' : {
59961                                 height: '80px'
59962                             },
59963                             '.x-grid-row td' : {
59964                                 'vertical-align' : 0
59965                             },
59966                             '.course-edit-link' : {
59967                                 'color' : 'blue',
59968                                 'text-overflow' : 'ellipsis',
59969                                 'overflow' : 'hidden',
59970                                 'white-space' : 'nowrap',
59971                                 'cursor' : 'pointer'
59972                             },
59973                             '.sub-link' : {
59974                                 'color' : 'green'
59975                             },
59976                             '.de-act-sup-link' : {
59977                                 'color' : 'purple',
59978                                 'text-decoration' : 'line-through'
59979                             },
59980                             '.de-act-link' : {
59981                                 'color' : 'red',
59982                                 'text-decoration' : 'line-through'
59983                             },
59984                             '.course-timesheet .course-highlight' : {
59985                                 'border-top-style': 'dashed !important',
59986                                 'border-bottom-bottom': 'dashed !important'
59987                             },
59988                             '.course-timesheet .course-item' : {
59989                                 'font-family'   : 'tahoma, arial, helvetica',
59990                                 'font-size'     : '11px',
59991                                 'overflow'      : 'hidden',
59992                                 'padding-left'  : '10px',
59993                                 'padding-right' : '10px',
59994                                 'padding-top' : '10px' 
59995                             }
59996                             
59997                         }, Roo.id());
59998                                 this.ds.load({});
59999                     }
60000                 },
60001                 autoWidth : true,
60002                 monitorWindowResize : false,
60003                 cellrenderer : function(v,x,r)
60004                 {
60005                     return v;
60006                 },
60007                 sm : {
60008                     xtype: 'CellSelectionModel',
60009                     xns: Roo.grid
60010                 },
60011                 dataSource : {
60012                     xtype: 'Store',
60013                     xns: Roo.data,
60014                     listeners : {
60015                         beforeload : function (_self, options)
60016                         {
60017                             options.params = options.params || {};
60018                             options.params._month = _this.monthField.getValue();
60019                             options.params.limit = 9999;
60020                             options.params['sort'] = 'when_dt';    
60021                             options.params['dir'] = 'ASC';    
60022                             this.proxy.loadResponse = this.loadResponse;
60023                             Roo.log("load?");
60024                             //this.addColumns();
60025                         },
60026                         load : function (_self, records, options)
60027                         {
60028                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60029                                 // if you click on the translation.. you can edit it...
60030                                 var el = Roo.get(this);
60031                                 var id = el.dom.getAttribute('data-id');
60032                                 var d = el.dom.getAttribute('data-date');
60033                                 var t = el.dom.getAttribute('data-time');
60034                                 //var id = this.child('span').dom.textContent;
60035                                 
60036                                 //Roo.log(this);
60037                                 Pman.Dialog.CourseCalendar.show({
60038                                     id : id,
60039                                     when_d : d,
60040                                     when_t : t,
60041                                     productitem_active : id ? 1 : 0
60042                                 }, function() {
60043                                     _this.grid.ds.load({});
60044                                 });
60045                            
60046                            });
60047                            
60048                            _this.panel.fireEvent('resize', [ '', '' ]);
60049                         }
60050                     },
60051                     loadResponse : function(o, success, response){
60052                             // this is overridden on before load..
60053                             
60054                             Roo.log("our code?");       
60055                             //Roo.log(success);
60056                             //Roo.log(response)
60057                             delete this.activeRequest;
60058                             if(!success){
60059                                 this.fireEvent("loadexception", this, o, response);
60060                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60061                                 return;
60062                             }
60063                             var result;
60064                             try {
60065                                 result = o.reader.read(response);
60066                             }catch(e){
60067                                 Roo.log("load exception?");
60068                                 this.fireEvent("loadexception", this, o, response, e);
60069                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60070                                 return;
60071                             }
60072                             Roo.log("ready...");        
60073                             // loop through result.records;
60074                             // and set this.tdate[date] = [] << array of records..
60075                             _this.tdata  = {};
60076                             Roo.each(result.records, function(r){
60077                                 //Roo.log(r.data);
60078                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60079                                     _this.tdata[r.data.when_dt.format('j')] = [];
60080                                 }
60081                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60082                             });
60083                             
60084                             //Roo.log(_this.tdata);
60085                             
60086                             result.records = [];
60087                             result.totalRecords = 6;
60088                     
60089                             // let's generate some duumy records for the rows.
60090                             //var st = _this.dateField.getValue();
60091                             
60092                             // work out monday..
60093                             //st = st.add(Date.DAY, -1 * st.format('w'));
60094                             
60095                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60096                             
60097                             var firstOfMonth = date.getFirstDayOfMonth();
60098                             var days = date.getDaysInMonth();
60099                             var d = 1;
60100                             var firstAdded = false;
60101                             for (var i = 0; i < result.totalRecords ; i++) {
60102                                 //var d= st.add(Date.DAY, i);
60103                                 var row = {};
60104                                 var added = 0;
60105                                 for(var w = 0 ; w < 7 ; w++){
60106                                     if(!firstAdded && firstOfMonth != w){
60107                                         continue;
60108                                     }
60109                                     if(d > days){
60110                                         continue;
60111                                     }
60112                                     firstAdded = true;
60113                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60114                                     row['weekday'+w] = String.format(
60115                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60116                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60117                                                     d,
60118                                                     date.format('Y-m-')+dd
60119                                                 );
60120                                     added++;
60121                                     if(typeof(_this.tdata[d]) != 'undefined'){
60122                                         Roo.each(_this.tdata[d], function(r){
60123                                             var is_sub = '';
60124                                             var deactive = '';
60125                                             var id = r.id;
60126                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60127                                             if(r.parent_id*1>0){
60128                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60129                                                 id = r.parent_id;
60130                                             }
60131                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60132                                                 deactive = 'de-act-link';
60133                                             }
60134                                             
60135                                             row['weekday'+w] += String.format(
60136                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60137                                                     id, //0
60138                                                     r.product_id_name, //1
60139                                                     r.when_dt.format('h:ia'), //2
60140                                                     is_sub, //3
60141                                                     deactive, //4
60142                                                     desc // 5
60143                                             );
60144                                         });
60145                                     }
60146                                     d++;
60147                                 }
60148                                 
60149                                 // only do this if something added..
60150                                 if(added > 0){ 
60151                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60152                                 }
60153                                 
60154                                 
60155                                 // push it twice. (second one with an hour..
60156                                 
60157                             }
60158                             //Roo.log(result);
60159                             this.fireEvent("load", this, o, o.request.arg);
60160                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60161                         },
60162                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60163                     proxy : {
60164                         xtype: 'HttpProxy',
60165                         xns: Roo.data,
60166                         method : 'GET',
60167                         url : baseURL + '/Roo/Shop_course.php'
60168                     },
60169                     reader : {
60170                         xtype: 'JsonReader',
60171                         xns: Roo.data,
60172                         id : 'id',
60173                         fields : [
60174                             {
60175                                 'name': 'id',
60176                                 'type': 'int'
60177                             },
60178                             {
60179                                 'name': 'when_dt',
60180                                 'type': 'string'
60181                             },
60182                             {
60183                                 'name': 'end_dt',
60184                                 'type': 'string'
60185                             },
60186                             {
60187                                 'name': 'parent_id',
60188                                 'type': 'int'
60189                             },
60190                             {
60191                                 'name': 'product_id',
60192                                 'type': 'int'
60193                             },
60194                             {
60195                                 'name': 'productitem_id',
60196                                 'type': 'int'
60197                             },
60198                             {
60199                                 'name': 'guid',
60200                                 'type': 'int'
60201                             }
60202                         ]
60203                     }
60204                 },
60205                 toolbar : {
60206                     xtype: 'Toolbar',
60207                     xns: Roo,
60208                     items : [
60209                         {
60210                             xtype: 'Button',
60211                             xns: Roo.Toolbar,
60212                             listeners : {
60213                                 click : function (_self, e)
60214                                 {
60215                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60216                                     sd.setMonth(sd.getMonth()-1);
60217                                     _this.monthField.setValue(sd.format('Y-m-d'));
60218                                     _this.grid.ds.load({});
60219                                 }
60220                             },
60221                             text : "Back"
60222                         },
60223                         {
60224                             xtype: 'Separator',
60225                             xns: Roo.Toolbar
60226                         },
60227                         {
60228                             xtype: 'MonthField',
60229                             xns: Roo.form,
60230                             listeners : {
60231                                 render : function (_self)
60232                                 {
60233                                     _this.monthField = _self;
60234                                    // _this.monthField.set  today
60235                                 },
60236                                 select : function (combo, date)
60237                                 {
60238                                     _this.grid.ds.load({});
60239                                 }
60240                             },
60241                             value : (function() { return new Date(); })()
60242                         },
60243                         {
60244                             xtype: 'Separator',
60245                             xns: Roo.Toolbar
60246                         },
60247                         {
60248                             xtype: 'TextItem',
60249                             xns: Roo.Toolbar,
60250                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60251                         },
60252                         {
60253                             xtype: 'Fill',
60254                             xns: Roo.Toolbar
60255                         },
60256                         {
60257                             xtype: 'Button',
60258                             xns: Roo.Toolbar,
60259                             listeners : {
60260                                 click : function (_self, e)
60261                                 {
60262                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60263                                     sd.setMonth(sd.getMonth()+1);
60264                                     _this.monthField.setValue(sd.format('Y-m-d'));
60265                                     _this.grid.ds.load({});
60266                                 }
60267                             },
60268                             text : "Next"
60269                         }
60270                     ]
60271                 },
60272                  
60273             }
60274         };
60275         
60276         *//*
60277  * Based on:
60278  * Ext JS Library 1.1.1
60279  * Copyright(c) 2006-2007, Ext JS, LLC.
60280  *
60281  * Originally Released Under LGPL - original licence link has changed is not relivant.
60282  *
60283  * Fork - LGPL
60284  * <script type="text/javascript">
60285  */
60286  
60287 /**
60288  * @class Roo.LoadMask
60289  * A simple utility class for generically masking elements while loading data.  If the element being masked has
60290  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
60291  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
60292  * element's UpdateManager load indicator and will be destroyed after the initial load.
60293  * @constructor
60294  * Create a new LoadMask
60295  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
60296  * @param {Object} config The config object
60297  */
60298 Roo.LoadMask = function(el, config){
60299     this.el = Roo.get(el);
60300     Roo.apply(this, config);
60301     if(this.store){
60302         this.store.on('beforeload', this.onBeforeLoad, this);
60303         this.store.on('load', this.onLoad, this);
60304         this.store.on('loadexception', this.onLoadException, this);
60305         this.removeMask = false;
60306     }else{
60307         var um = this.el.getUpdateManager();
60308         um.showLoadIndicator = false; // disable the default indicator
60309         um.on('beforeupdate', this.onBeforeLoad, this);
60310         um.on('update', this.onLoad, this);
60311         um.on('failure', this.onLoad, this);
60312         this.removeMask = true;
60313     }
60314 };
60315
60316 Roo.LoadMask.prototype = {
60317     /**
60318      * @cfg {Boolean} removeMask
60319      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
60320      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
60321      */
60322     /**
60323      * @cfg {String} msg
60324      * The text to display in a centered loading message box (defaults to 'Loading...')
60325      */
60326     msg : 'Loading...',
60327     /**
60328      * @cfg {String} msgCls
60329      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
60330      */
60331     msgCls : 'x-mask-loading',
60332
60333     /**
60334      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
60335      * @type Boolean
60336      */
60337     disabled: false,
60338
60339     /**
60340      * Disables the mask to prevent it from being displayed
60341      */
60342     disable : function(){
60343        this.disabled = true;
60344     },
60345
60346     /**
60347      * Enables the mask so that it can be displayed
60348      */
60349     enable : function(){
60350         this.disabled = false;
60351     },
60352     
60353     onLoadException : function()
60354     {
60355         Roo.log(arguments);
60356         
60357         if (typeof(arguments[3]) != 'undefined') {
60358             Roo.MessageBox.alert("Error loading",arguments[3]);
60359         } 
60360         /*
60361         try {
60362             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
60363                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
60364             }   
60365         } catch(e) {
60366             
60367         }
60368         */
60369     
60370         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60371     },
60372     // private
60373     onLoad : function()
60374     {
60375         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
60376     },
60377
60378     // private
60379     onBeforeLoad : function(){
60380         if(!this.disabled){
60381             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
60382         }
60383     },
60384
60385     // private
60386     destroy : function(){
60387         if(this.store){
60388             this.store.un('beforeload', this.onBeforeLoad, this);
60389             this.store.un('load', this.onLoad, this);
60390             this.store.un('loadexception', this.onLoadException, this);
60391         }else{
60392             var um = this.el.getUpdateManager();
60393             um.un('beforeupdate', this.onBeforeLoad, this);
60394             um.un('update', this.onLoad, this);
60395             um.un('failure', this.onLoad, this);
60396         }
60397     }
60398 };/*
60399  * Based on:
60400  * Ext JS Library 1.1.1
60401  * Copyright(c) 2006-2007, Ext JS, LLC.
60402  *
60403  * Originally Released Under LGPL - original licence link has changed is not relivant.
60404  *
60405  * Fork - LGPL
60406  * <script type="text/javascript">
60407  */
60408
60409
60410 /**
60411  * @class Roo.XTemplate
60412  * @extends Roo.Template
60413  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
60414 <pre><code>
60415 var t = new Roo.XTemplate(
60416         '&lt;select name="{name}"&gt;',
60417                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
60418         '&lt;/select&gt;'
60419 );
60420  
60421 // then append, applying the master template values
60422  </code></pre>
60423  *
60424  * Supported features:
60425  *
60426  *  Tags:
60427
60428 <pre><code>
60429       {a_variable} - output encoded.
60430       {a_variable.format:("Y-m-d")} - call a method on the variable
60431       {a_variable:raw} - unencoded output
60432       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
60433       {a_variable:this.method_on_template(...)} - call a method on the template object.
60434  
60435 </code></pre>
60436  *  The tpl tag:
60437 <pre><code>
60438         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
60439         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
60440         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
60441         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
60442   
60443         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
60444         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
60445 </code></pre>
60446  *      
60447  */
60448 Roo.XTemplate = function()
60449 {
60450     Roo.XTemplate.superclass.constructor.apply(this, arguments);
60451     if (this.html) {
60452         this.compile();
60453     }
60454 };
60455
60456
60457 Roo.extend(Roo.XTemplate, Roo.Template, {
60458
60459     /**
60460      * The various sub templates
60461      */
60462     tpls : false,
60463     /**
60464      *
60465      * basic tag replacing syntax
60466      * WORD:WORD()
60467      *
60468      * // you can fake an object call by doing this
60469      *  x.t:(test,tesT) 
60470      * 
60471      */
60472     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
60473
60474     /**
60475      * compile the template
60476      *
60477      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
60478      *
60479      */
60480     compile: function()
60481     {
60482         var s = this.html;
60483      
60484         s = ['<tpl>', s, '</tpl>'].join('');
60485     
60486         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
60487             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
60488             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
60489             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
60490             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
60491             m,
60492             id     = 0,
60493             tpls   = [];
60494     
60495         while(true == !!(m = s.match(re))){
60496             var forMatch   = m[0].match(nameRe),
60497                 ifMatch   = m[0].match(ifRe),
60498                 execMatch   = m[0].match(execRe),
60499                 namedMatch   = m[0].match(namedRe),
60500                 
60501                 exp  = null, 
60502                 fn   = null,
60503                 exec = null,
60504                 name = forMatch && forMatch[1] ? forMatch[1] : '';
60505                 
60506             if (ifMatch) {
60507                 // if - puts fn into test..
60508                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
60509                 if(exp){
60510                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
60511                 }
60512             }
60513             
60514             if (execMatch) {
60515                 // exec - calls a function... returns empty if true is  returned.
60516                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
60517                 if(exp){
60518                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
60519                 }
60520             }
60521             
60522             
60523             if (name) {
60524                 // for = 
60525                 switch(name){
60526                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
60527                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
60528                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
60529                 }
60530             }
60531             var uid = namedMatch ? namedMatch[1] : id;
60532             
60533             
60534             tpls.push({
60535                 id:     namedMatch ? namedMatch[1] : id,
60536                 target: name,
60537                 exec:   exec,
60538                 test:   fn,
60539                 body:   m[1] || ''
60540             });
60541             if (namedMatch) {
60542                 s = s.replace(m[0], '');
60543             } else { 
60544                 s = s.replace(m[0], '{xtpl'+ id + '}');
60545             }
60546             ++id;
60547         }
60548         this.tpls = [];
60549         for(var i = tpls.length-1; i >= 0; --i){
60550             this.compileTpl(tpls[i]);
60551             this.tpls[tpls[i].id] = tpls[i];
60552         }
60553         this.master = tpls[tpls.length-1];
60554         return this;
60555     },
60556     /**
60557      * same as applyTemplate, except it's done to one of the subTemplates
60558      * when using named templates, you can do:
60559      *
60560      * var str = pl.applySubTemplate('your-name', values);
60561      *
60562      * 
60563      * @param {Number} id of the template
60564      * @param {Object} values to apply to template
60565      * @param {Object} parent (normaly the instance of this object)
60566      */
60567     applySubTemplate : function(id, values, parent)
60568     {
60569         
60570         
60571         var t = this.tpls[id];
60572         
60573         
60574         try { 
60575             if(t.test && !t.test.call(this, values, parent)){
60576                 return '';
60577             }
60578         } catch(e) {
60579             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
60580             Roo.log(e.toString());
60581             Roo.log(t.test);
60582             return ''
60583         }
60584         try { 
60585             
60586             if(t.exec && t.exec.call(this, values, parent)){
60587                 return '';
60588             }
60589         } catch(e) {
60590             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
60591             Roo.log(e.toString());
60592             Roo.log(t.exec);
60593             return ''
60594         }
60595         try {
60596             var vs = t.target ? t.target.call(this, values, parent) : values;
60597             parent = t.target ? values : parent;
60598             if(t.target && vs instanceof Array){
60599                 var buf = [];
60600                 for(var i = 0, len = vs.length; i < len; i++){
60601                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
60602                 }
60603                 return buf.join('');
60604             }
60605             return t.compiled.call(this, vs, parent);
60606         } catch (e) {
60607             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
60608             Roo.log(e.toString());
60609             Roo.log(t.compiled);
60610             return '';
60611         }
60612     },
60613
60614     compileTpl : function(tpl)
60615     {
60616         var fm = Roo.util.Format;
60617         var useF = this.disableFormats !== true;
60618         var sep = Roo.isGecko ? "+" : ",";
60619         var undef = function(str) {
60620             Roo.log("Property not found :"  + str);
60621             return '';
60622         };
60623         
60624         var fn = function(m, name, format, args)
60625         {
60626             //Roo.log(arguments);
60627             args = args ? args.replace(/\\'/g,"'") : args;
60628             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
60629             if (typeof(format) == 'undefined') {
60630                 format= 'htmlEncode';
60631             }
60632             if (format == 'raw' ) {
60633                 format = false;
60634             }
60635             
60636             if(name.substr(0, 4) == 'xtpl'){
60637                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
60638             }
60639             
60640             // build an array of options to determine if value is undefined..
60641             
60642             // basically get 'xxxx.yyyy' then do
60643             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
60644             //    (function () { Roo.log("Property not found"); return ''; })() :
60645             //    ......
60646             
60647             var udef_ar = [];
60648             var lookfor = '';
60649             Roo.each(name.split('.'), function(st) {
60650                 lookfor += (lookfor.length ? '.': '') + st;
60651                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
60652             });
60653             
60654             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
60655             
60656             
60657             if(format && useF){
60658                 
60659                 args = args ? ',' + args : "";
60660                  
60661                 if(format.substr(0, 5) != "this."){
60662                     format = "fm." + format + '(';
60663                 }else{
60664                     format = 'this.call("'+ format.substr(5) + '", ';
60665                     args = ", values";
60666                 }
60667                 
60668                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
60669             }
60670              
60671             if (args.length) {
60672                 // called with xxyx.yuu:(test,test)
60673                 // change to ()
60674                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
60675             }
60676             // raw.. - :raw modifier..
60677             return "'"+ sep + udef_st  + name + ")"+sep+"'";
60678             
60679         };
60680         var body;
60681         // branched to use + in gecko and [].join() in others
60682         if(Roo.isGecko){
60683             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
60684                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
60685                     "';};};";
60686         }else{
60687             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
60688             body.push(tpl.body.replace(/(\r\n|\n)/g,
60689                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
60690             body.push("'].join('');};};");
60691             body = body.join('');
60692         }
60693         
60694         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
60695        
60696         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
60697         eval(body);
60698         
60699         return this;
60700     },
60701
60702     applyTemplate : function(values){
60703         return this.master.compiled.call(this, values, {});
60704         //var s = this.subs;
60705     },
60706
60707     apply : function(){
60708         return this.applyTemplate.apply(this, arguments);
60709     }
60710
60711  });
60712
60713 Roo.XTemplate.from = function(el){
60714     el = Roo.getDom(el);
60715     return new Roo.XTemplate(el.value || el.innerHTML);
60716 };