roojs-core.js
[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         Roo.log(["applyTemplate", values]);
4671         try {
4672            
4673             if(this.compiled){
4674                 return this.compiled(values);
4675             }
4676             var useF = this.disableFormats !== true;
4677             var fm = Roo.util.Format, tpl = this;
4678             var fn = function(m, name, format, args){
4679                 if(format && useF){
4680                     if(format.substr(0, 5) == "this."){
4681                         return tpl.call(format.substr(5), values[name], values);
4682                     }else{
4683                         if(args){
4684                             // quoted values are required for strings in compiled templates, 
4685                             // but for non compiled we need to strip them
4686                             // quoted reversed for jsmin
4687                             var re = /^\s*['"](.*)["']\s*$/;
4688                             args = args.split(',');
4689                             for(var i = 0, len = args.length; i < len; i++){
4690                                 args[i] = args[i].replace(re, "$1");
4691                             }
4692                             args = [values[name]].concat(args);
4693                         }else{
4694                             args = [values[name]];
4695                         }
4696                         return fm[format].apply(fm, args);
4697                     }
4698                 }else{
4699                     return values[name] !== undefined ? values[name] : "";
4700                 }
4701             };
4702             return this.html.replace(this.re, fn);
4703         } catch (e) {
4704             Roo.log(e);
4705             throw e;
4706         }
4707          
4708     },
4709     
4710     loading : false,
4711       
4712     load : function ()
4713     {
4714          
4715         if (this.loading) {
4716             return;
4717         }
4718         var _t = this;
4719         
4720         this.loading = true;
4721         this.compiled = false;
4722         
4723         var cx = new Roo.data.Connection();
4724         cx.request({
4725             url : this.url,
4726             method : 'GET',
4727             success : function (response) {
4728                 _t.loading = false;
4729                 _t.html = response.responseText;
4730                 _t.url = false;
4731                 _t.compile();
4732              },
4733             failure : function(response) {
4734                 Roo.log("Template failed to load from " + _t.url);
4735                 _t.loading = false;
4736             }
4737         });
4738     },
4739
4740     /**
4741      * Sets the HTML used as the template and optionally compiles it.
4742      * @param {String} html
4743      * @param {Boolean} compile (optional) True to compile the template (defaults to undefined)
4744      * @return {Roo.Template} this
4745      */
4746     set : function(html, compile){
4747         this.html = html;
4748         this.compiled = null;
4749         if(compile){
4750             this.compile();
4751         }
4752         return this;
4753     },
4754     
4755     /**
4756      * True to disable format functions (defaults to false)
4757      * @type Boolean
4758      */
4759     disableFormats : false,
4760     
4761     /**
4762     * The regular expression used to match template variables 
4763     * @type RegExp
4764     * @property 
4765     */
4766     re : /\{([\w-]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
4767     
4768     /**
4769      * Compiles the template into an internal function, eliminating the RegEx overhead.
4770      * @return {Roo.Template} this
4771      */
4772     compile : function(){
4773         var fm = Roo.util.Format;
4774         var useF = this.disableFormats !== true;
4775         var sep = Roo.isGecko ? "+" : ",";
4776         var fn = function(m, name, format, args){
4777             if(format && useF){
4778                 args = args ? ',' + args : "";
4779                 if(format.substr(0, 5) != "this."){
4780                     format = "fm." + format + '(';
4781                 }else{
4782                     format = 'this.call("'+ format.substr(5) + '", ';
4783                     args = ", values";
4784                 }
4785             }else{
4786                 args= ''; format = "(values['" + name + "'] == undefined ? '' : ";
4787             }
4788             return "'"+ sep + format + "values['" + name + "']" + args + ")"+sep+"'";
4789         };
4790         var body;
4791         // branched to use + in gecko and [].join() in others
4792         if(Roo.isGecko){
4793             body = "this.compiled = function(values){ return '" +
4794                    this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
4795                     "';};";
4796         }else{
4797             body = ["this.compiled = function(values){ return ['"];
4798             body.push(this.html.replace(/\\/g, '\\\\').replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn));
4799             body.push("'].join('');};");
4800             body = body.join('');
4801         }
4802         /**
4803          * eval:var:values
4804          * eval:var:fm
4805          */
4806         eval(body);
4807         return this;
4808     },
4809     
4810     // private function used to call members
4811     call : function(fnName, value, allValues){
4812         return this[fnName](value, allValues);
4813     },
4814     
4815     /**
4816      * Applies the supplied values to the template and inserts the new node(s) as the first child of el.
4817      * @param {String/HTMLElement/Roo.Element} el The context element
4818      * @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'})
4819      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4820      * @return {HTMLElement/Roo.Element} The new node or Element
4821      */
4822     insertFirst: function(el, values, returnElement){
4823         return this.doInsert('afterBegin', el, values, returnElement);
4824     },
4825
4826     /**
4827      * Applies the supplied values to the template and inserts the new node(s) before el.
4828      * @param {String/HTMLElement/Roo.Element} el The context element
4829      * @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'})
4830      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4831      * @return {HTMLElement/Roo.Element} The new node or Element
4832      */
4833     insertBefore: function(el, values, returnElement){
4834         return this.doInsert('beforeBegin', el, values, returnElement);
4835     },
4836
4837     /**
4838      * Applies the supplied values to the template and inserts the new node(s) after el.
4839      * @param {String/HTMLElement/Roo.Element} el The context element
4840      * @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'})
4841      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4842      * @return {HTMLElement/Roo.Element} The new node or Element
4843      */
4844     insertAfter : function(el, values, returnElement){
4845         return this.doInsert('afterEnd', el, values, returnElement);
4846     },
4847     
4848     /**
4849      * Applies the supplied values to the template and appends the new node(s) to el.
4850      * @param {String/HTMLElement/Roo.Element} el The context element
4851      * @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'})
4852      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4853      * @return {HTMLElement/Roo.Element} The new node or Element
4854      */
4855     append : function(el, values, returnElement){
4856         return this.doInsert('beforeEnd', el, values, returnElement);
4857     },
4858
4859     doInsert : function(where, el, values, returnEl){
4860         el = Roo.getDom(el);
4861         var newNode = Roo.DomHelper.insertHtml(where, el, this.applyTemplate(values));
4862         return returnEl ? Roo.get(newNode, true) : newNode;
4863     },
4864
4865     /**
4866      * Applies the supplied values to the template and overwrites the content of el with the new node(s).
4867      * @param {String/HTMLElement/Roo.Element} el The context element
4868      * @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'})
4869      * @param {Boolean} returnElement (optional) true to return a Roo.Element (defaults to undefined)
4870      * @return {HTMLElement/Roo.Element} The new node or Element
4871      */
4872     overwrite : function(el, values, returnElement){
4873         el = Roo.getDom(el);
4874         el.innerHTML = this.applyTemplate(values);
4875         return returnElement ? Roo.get(el.firstChild, true) : el.firstChild;
4876     }
4877 };
4878 /**
4879  * Alias for {@link #applyTemplate}
4880  * @method
4881  */
4882 Roo.Template.prototype.apply = Roo.Template.prototype.applyTemplate;
4883
4884 // backwards compat
4885 Roo.DomHelper.Template = Roo.Template;
4886
4887 /**
4888  * Creates a template from the passed element's value (<i>display:none</i> textarea, preferred) or innerHTML.
4889  * @param {String/HTMLElement} el A DOM element or its id
4890  * @returns {Roo.Template} The created template
4891  * @static
4892  */
4893 Roo.Template.from = function(el){
4894     el = Roo.getDom(el);
4895     return new Roo.Template(el.value || el.innerHTML);
4896 };/*
4897  * Based on:
4898  * Ext JS Library 1.1.1
4899  * Copyright(c) 2006-2007, Ext JS, LLC.
4900  *
4901  * Originally Released Under LGPL - original licence link has changed is not relivant.
4902  *
4903  * Fork - LGPL
4904  * <script type="text/javascript">
4905  */
4906  
4907
4908 /*
4909  * This is code is also distributed under MIT license for use
4910  * with jQuery and prototype JavaScript libraries.
4911  */
4912 /**
4913  * @class Roo.DomQuery
4914 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).
4915 <p>
4916 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>
4917
4918 <p>
4919 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.
4920 </p>
4921 <h4>Element Selectors:</h4>
4922 <ul class="list">
4923     <li> <b>*</b> any element</li>
4924     <li> <b>E</b> an element with the tag E</li>
4925     <li> <b>E F</b> All descendent elements of E that have the tag F</li>
4926     <li> <b>E > F</b> or <b>E/F</b> all direct children elements of E that have the tag F</li>
4927     <li> <b>E + F</b> all elements with the tag F that are immediately preceded by an element with the tag E</li>
4928     <li> <b>E ~ F</b> all elements with the tag F that are preceded by a sibling element with the tag E</li>
4929 </ul>
4930 <h4>Attribute Selectors:</h4>
4931 <p>The use of @ and quotes are optional. For example, div[@foo='bar'] is also a valid attribute selector.</p>
4932 <ul class="list">
4933     <li> <b>E[foo]</b> has an attribute "foo"</li>
4934     <li> <b>E[foo=bar]</b> has an attribute "foo" that equals "bar"</li>
4935     <li> <b>E[foo^=bar]</b> has an attribute "foo" that starts with "bar"</li>
4936     <li> <b>E[foo$=bar]</b> has an attribute "foo" that ends with "bar"</li>
4937     <li> <b>E[foo*=bar]</b> has an attribute "foo" that contains the substring "bar"</li>
4938     <li> <b>E[foo%=2]</b> has an attribute "foo" that is evenly divisible by 2</li>
4939     <li> <b>E[foo!=bar]</b> has an attribute "foo" that does not equal "bar"</li>
4940 </ul>
4941 <h4>Pseudo Classes:</h4>
4942 <ul class="list">
4943     <li> <b>E:first-child</b> E is the first child of its parent</li>
4944     <li> <b>E:last-child</b> E is the last child of its parent</li>
4945     <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>
4946     <li> <b>E:nth-child(odd)</b> E is an odd child of its parent</li>
4947     <li> <b>E:nth-child(even)</b> E is an even child of its parent</li>
4948     <li> <b>E:only-child</b> E is the only child of its parent</li>
4949     <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>
4950     <li> <b>E:first</b> the first E in the resultset</li>
4951     <li> <b>E:last</b> the last E in the resultset</li>
4952     <li> <b>E:nth(<i>n</i>)</b> the <i>n</i>th E in the resultset (1 based)</li>
4953     <li> <b>E:odd</b> shortcut for :nth-child(odd)</li>
4954     <li> <b>E:even</b> shortcut for :nth-child(even)</li>
4955     <li> <b>E:contains(foo)</b> E's innerHTML contains the substring "foo"</li>
4956     <li> <b>E:nodeValue(foo)</b> E contains a textNode with a nodeValue that equals "foo"</li>
4957     <li> <b>E:not(S)</b> an E element that does not match simple selector S</li>
4958     <li> <b>E:has(S)</b> an E element that has a descendent that matches simple selector S</li>
4959     <li> <b>E:next(S)</b> an E element whose next sibling matches simple selector S</li>
4960     <li> <b>E:prev(S)</b> an E element whose previous sibling matches simple selector S</li>
4961 </ul>
4962 <h4>CSS Value Selectors:</h4>
4963 <ul class="list">
4964     <li> <b>E{display=none}</b> css value "display" that equals "none"</li>
4965     <li> <b>E{display^=none}</b> css value "display" that starts with "none"</li>
4966     <li> <b>E{display$=none}</b> css value "display" that ends with "none"</li>
4967     <li> <b>E{display*=none}</b> css value "display" that contains the substring "none"</li>
4968     <li> <b>E{display%=2}</b> css value "display" that is evenly divisible by 2</li>
4969     <li> <b>E{display!=none}</b> css value "display" that does not equal "none"</li>
4970 </ul>
4971  * @singleton
4972  */
4973 Roo.DomQuery = function(){
4974     var cache = {}, simpleCache = {}, valueCache = {};
4975     var nonSpace = /\S/;
4976     var trimRe = /^\s+|\s+$/g;
4977     var tplRe = /\{(\d+)\}/g;
4978     var modeRe = /^(\s?[\/>+~]\s?|\s|$)/;
4979     var tagTokenRe = /^(#)?([\w-\*]+)/;
4980     var nthRe = /(\d*)n\+?(\d*)/, nthRe2 = /\D/;
4981
4982     function child(p, index){
4983         var i = 0;
4984         var n = p.firstChild;
4985         while(n){
4986             if(n.nodeType == 1){
4987                if(++i == index){
4988                    return n;
4989                }
4990             }
4991             n = n.nextSibling;
4992         }
4993         return null;
4994     };
4995
4996     function next(n){
4997         while((n = n.nextSibling) && n.nodeType != 1);
4998         return n;
4999     };
5000
5001     function prev(n){
5002         while((n = n.previousSibling) && n.nodeType != 1);
5003         return n;
5004     };
5005
5006     function children(d){
5007         var n = d.firstChild, ni = -1;
5008             while(n){
5009                 var nx = n.nextSibling;
5010                 if(n.nodeType == 3 && !nonSpace.test(n.nodeValue)){
5011                     d.removeChild(n);
5012                 }else{
5013                     n.nodeIndex = ++ni;
5014                 }
5015                 n = nx;
5016             }
5017             return this;
5018         };
5019
5020     function byClassName(c, a, v){
5021         if(!v){
5022             return c;
5023         }
5024         var r = [], ri = -1, cn;
5025         for(var i = 0, ci; ci = c[i]; i++){
5026             if((' '+ci.className+' ').indexOf(v) != -1){
5027                 r[++ri] = ci;
5028             }
5029         }
5030         return r;
5031     };
5032
5033     function attrValue(n, attr){
5034         if(!n.tagName && typeof n.length != "undefined"){
5035             n = n[0];
5036         }
5037         if(!n){
5038             return null;
5039         }
5040         if(attr == "for"){
5041             return n.htmlFor;
5042         }
5043         if(attr == "class" || attr == "className"){
5044             return n.className;
5045         }
5046         return n.getAttribute(attr) || n[attr];
5047
5048     };
5049
5050     function getNodes(ns, mode, tagName){
5051         var result = [], ri = -1, cs;
5052         if(!ns){
5053             return result;
5054         }
5055         tagName = tagName || "*";
5056         if(typeof ns.getElementsByTagName != "undefined"){
5057             ns = [ns];
5058         }
5059         if(!mode){
5060             for(var i = 0, ni; ni = ns[i]; i++){
5061                 cs = ni.getElementsByTagName(tagName);
5062                 for(var j = 0, ci; ci = cs[j]; j++){
5063                     result[++ri] = ci;
5064                 }
5065             }
5066         }else if(mode == "/" || mode == ">"){
5067             var utag = tagName.toUpperCase();
5068             for(var i = 0, ni, cn; ni = ns[i]; i++){
5069                 cn = ni.children || ni.childNodes;
5070                 for(var j = 0, cj; cj = cn[j]; j++){
5071                     if(cj.nodeName == utag || cj.nodeName == tagName  || tagName == '*'){
5072                         result[++ri] = cj;
5073                     }
5074                 }
5075             }
5076         }else if(mode == "+"){
5077             var utag = tagName.toUpperCase();
5078             for(var i = 0, n; n = ns[i]; i++){
5079                 while((n = n.nextSibling) && n.nodeType != 1);
5080                 if(n && (n.nodeName == utag || n.nodeName == tagName || tagName == '*')){
5081                     result[++ri] = n;
5082                 }
5083             }
5084         }else if(mode == "~"){
5085             for(var i = 0, n; n = ns[i]; i++){
5086                 while((n = n.nextSibling) && (n.nodeType != 1 || (tagName == '*' || n.tagName.toLowerCase()!=tagName)));
5087                 if(n){
5088                     result[++ri] = n;
5089                 }
5090             }
5091         }
5092         return result;
5093     };
5094
5095     function concat(a, b){
5096         if(b.slice){
5097             return a.concat(b);
5098         }
5099         for(var i = 0, l = b.length; i < l; i++){
5100             a[a.length] = b[i];
5101         }
5102         return a;
5103     }
5104
5105     function byTag(cs, tagName){
5106         if(cs.tagName || cs == document){
5107             cs = [cs];
5108         }
5109         if(!tagName){
5110             return cs;
5111         }
5112         var r = [], ri = -1;
5113         tagName = tagName.toLowerCase();
5114         for(var i = 0, ci; ci = cs[i]; i++){
5115             if(ci.nodeType == 1 && ci.tagName.toLowerCase()==tagName){
5116                 r[++ri] = ci;
5117             }
5118         }
5119         return r;
5120     };
5121
5122     function byId(cs, attr, id){
5123         if(cs.tagName || cs == document){
5124             cs = [cs];
5125         }
5126         if(!id){
5127             return cs;
5128         }
5129         var r = [], ri = -1;
5130         for(var i = 0,ci; ci = cs[i]; i++){
5131             if(ci && ci.id == id){
5132                 r[++ri] = ci;
5133                 return r;
5134             }
5135         }
5136         return r;
5137     };
5138
5139     function byAttribute(cs, attr, value, op, custom){
5140         var r = [], ri = -1, st = custom=="{";
5141         var f = Roo.DomQuery.operators[op];
5142         for(var i = 0, ci; ci = cs[i]; i++){
5143             var a;
5144             if(st){
5145                 a = Roo.DomQuery.getStyle(ci, attr);
5146             }
5147             else if(attr == "class" || attr == "className"){
5148                 a = ci.className;
5149             }else if(attr == "for"){
5150                 a = ci.htmlFor;
5151             }else if(attr == "href"){
5152                 a = ci.getAttribute("href", 2);
5153             }else{
5154                 a = ci.getAttribute(attr);
5155             }
5156             if((f && f(a, value)) || (!f && a)){
5157                 r[++ri] = ci;
5158             }
5159         }
5160         return r;
5161     };
5162
5163     function byPseudo(cs, name, value){
5164         return Roo.DomQuery.pseudos[name](cs, value);
5165     };
5166
5167     // This is for IE MSXML which does not support expandos.
5168     // IE runs the same speed using setAttribute, however FF slows way down
5169     // and Safari completely fails so they need to continue to use expandos.
5170     var isIE = window.ActiveXObject ? true : false;
5171
5172     // this eval is stop the compressor from
5173     // renaming the variable to something shorter
5174     
5175     /** eval:var:batch */
5176     var batch = 30803; 
5177
5178     var key = 30803;
5179
5180     function nodupIEXml(cs){
5181         var d = ++key;
5182         cs[0].setAttribute("_nodup", d);
5183         var r = [cs[0]];
5184         for(var i = 1, len = cs.length; i < len; i++){
5185             var c = cs[i];
5186             if(!c.getAttribute("_nodup") != d){
5187                 c.setAttribute("_nodup", d);
5188                 r[r.length] = c;
5189             }
5190         }
5191         for(var i = 0, len = cs.length; i < len; i++){
5192             cs[i].removeAttribute("_nodup");
5193         }
5194         return r;
5195     }
5196
5197     function nodup(cs){
5198         if(!cs){
5199             return [];
5200         }
5201         var len = cs.length, c, i, r = cs, cj, ri = -1;
5202         if(!len || typeof cs.nodeType != "undefined" || len == 1){
5203             return cs;
5204         }
5205         if(isIE && typeof cs[0].selectSingleNode != "undefined"){
5206             return nodupIEXml(cs);
5207         }
5208         var d = ++key;
5209         cs[0]._nodup = d;
5210         for(i = 1; c = cs[i]; i++){
5211             if(c._nodup != d){
5212                 c._nodup = d;
5213             }else{
5214                 r = [];
5215                 for(var j = 0; j < i; j++){
5216                     r[++ri] = cs[j];
5217                 }
5218                 for(j = i+1; cj = cs[j]; j++){
5219                     if(cj._nodup != d){
5220                         cj._nodup = d;
5221                         r[++ri] = cj;
5222                     }
5223                 }
5224                 return r;
5225             }
5226         }
5227         return r;
5228     }
5229
5230     function quickDiffIEXml(c1, c2){
5231         var d = ++key;
5232         for(var i = 0, len = c1.length; i < len; i++){
5233             c1[i].setAttribute("_qdiff", d);
5234         }
5235         var r = [];
5236         for(var i = 0, len = c2.length; i < len; i++){
5237             if(c2[i].getAttribute("_qdiff") != d){
5238                 r[r.length] = c2[i];
5239             }
5240         }
5241         for(var i = 0, len = c1.length; i < len; i++){
5242            c1[i].removeAttribute("_qdiff");
5243         }
5244         return r;
5245     }
5246
5247     function quickDiff(c1, c2){
5248         var len1 = c1.length;
5249         if(!len1){
5250             return c2;
5251         }
5252         if(isIE && c1[0].selectSingleNode){
5253             return quickDiffIEXml(c1, c2);
5254         }
5255         var d = ++key;
5256         for(var i = 0; i < len1; i++){
5257             c1[i]._qdiff = d;
5258         }
5259         var r = [];
5260         for(var i = 0, len = c2.length; i < len; i++){
5261             if(c2[i]._qdiff != d){
5262                 r[r.length] = c2[i];
5263             }
5264         }
5265         return r;
5266     }
5267
5268     function quickId(ns, mode, root, id){
5269         if(ns == root){
5270            var d = root.ownerDocument || root;
5271            return d.getElementById(id);
5272         }
5273         ns = getNodes(ns, mode, "*");
5274         return byId(ns, null, id);
5275     }
5276
5277     return {
5278         getStyle : function(el, name){
5279             return Roo.fly(el).getStyle(name);
5280         },
5281         /**
5282          * Compiles a selector/xpath query into a reusable function. The returned function
5283          * takes one parameter "root" (optional), which is the context node from where the query should start.
5284          * @param {String} selector The selector/xpath query
5285          * @param {String} type (optional) Either "select" (the default) or "simple" for a simple selector match
5286          * @return {Function}
5287          */
5288         compile : function(path, type){
5289             type = type || "select";
5290             
5291             var fn = ["var f = function(root){\n var mode; ++batch; var n = root || document;\n"];
5292             var q = path, mode, lq;
5293             var tk = Roo.DomQuery.matchers;
5294             var tklen = tk.length;
5295             var mm;
5296
5297             // accept leading mode switch
5298             var lmode = q.match(modeRe);
5299             if(lmode && lmode[1]){
5300                 fn[fn.length] = 'mode="'+lmode[1].replace(trimRe, "")+'";';
5301                 q = q.replace(lmode[1], "");
5302             }
5303             // strip leading slashes
5304             while(path.substr(0, 1)=="/"){
5305                 path = path.substr(1);
5306             }
5307
5308             while(q && lq != q){
5309                 lq = q;
5310                 var tm = q.match(tagTokenRe);
5311                 if(type == "select"){
5312                     if(tm){
5313                         if(tm[1] == "#"){
5314                             fn[fn.length] = 'n = quickId(n, mode, root, "'+tm[2]+'");';
5315                         }else{
5316                             fn[fn.length] = 'n = getNodes(n, mode, "'+tm[2]+'");';
5317                         }
5318                         q = q.replace(tm[0], "");
5319                     }else if(q.substr(0, 1) != '@'){
5320                         fn[fn.length] = 'n = getNodes(n, mode, "*");';
5321                     }
5322                 }else{
5323                     if(tm){
5324                         if(tm[1] == "#"){
5325                             fn[fn.length] = 'n = byId(n, null, "'+tm[2]+'");';
5326                         }else{
5327                             fn[fn.length] = 'n = byTag(n, "'+tm[2]+'");';
5328                         }
5329                         q = q.replace(tm[0], "");
5330                     }
5331                 }
5332                 while(!(mm = q.match(modeRe))){
5333                     var matched = false;
5334                     for(var j = 0; j < tklen; j++){
5335                         var t = tk[j];
5336                         var m = q.match(t.re);
5337                         if(m){
5338                             fn[fn.length] = t.select.replace(tplRe, function(x, i){
5339                                                     return m[i];
5340                                                 });
5341                             q = q.replace(m[0], "");
5342                             matched = true;
5343                             break;
5344                         }
5345                     }
5346                     // prevent infinite loop on bad selector
5347                     if(!matched){
5348                         throw 'Error parsing selector, parsing failed at "' + q + '"';
5349                     }
5350                 }
5351                 if(mm[1]){
5352                     fn[fn.length] = 'mode="'+mm[1].replace(trimRe, "")+'";';
5353                     q = q.replace(mm[1], "");
5354                 }
5355             }
5356             fn[fn.length] = "return nodup(n);\n}";
5357             
5358              /** 
5359               * list of variables that need from compression as they are used by eval.
5360              *  eval:var:batch 
5361              *  eval:var:nodup
5362              *  eval:var:byTag
5363              *  eval:var:ById
5364              *  eval:var:getNodes
5365              *  eval:var:quickId
5366              *  eval:var:mode
5367              *  eval:var:root
5368              *  eval:var:n
5369              *  eval:var:byClassName
5370              *  eval:var:byPseudo
5371              *  eval:var:byAttribute
5372              *  eval:var:attrValue
5373              * 
5374              **/ 
5375             eval(fn.join(""));
5376             return f;
5377         },
5378
5379         /**
5380          * Selects a group of elements.
5381          * @param {String} selector The selector/xpath query (can be a comma separated list of selectors)
5382          * @param {Node} root (optional) The start of the query (defaults to document).
5383          * @return {Array}
5384          */
5385         select : function(path, root, type){
5386             if(!root || root == document){
5387                 root = document;
5388             }
5389             if(typeof root == "string"){
5390                 root = document.getElementById(root);
5391             }
5392             var paths = path.split(",");
5393             var results = [];
5394             for(var i = 0, len = paths.length; i < len; i++){
5395                 var p = paths[i].replace(trimRe, "");
5396                 if(!cache[p]){
5397                     cache[p] = Roo.DomQuery.compile(p);
5398                     if(!cache[p]){
5399                         throw p + " is not a valid selector";
5400                     }
5401                 }
5402                 var result = cache[p](root);
5403                 if(result && result != document){
5404                     results = results.concat(result);
5405                 }
5406             }
5407             if(paths.length > 1){
5408                 return nodup(results);
5409             }
5410             return results;
5411         },
5412
5413         /**
5414          * Selects a single element.
5415          * @param {String} selector The selector/xpath query
5416          * @param {Node} root (optional) The start of the query (defaults to document).
5417          * @return {Element}
5418          */
5419         selectNode : function(path, root){
5420             return Roo.DomQuery.select(path, root)[0];
5421         },
5422
5423         /**
5424          * Selects the value of a node, optionally replacing null with the defaultValue.
5425          * @param {String} selector The selector/xpath query
5426          * @param {Node} root (optional) The start of the query (defaults to document).
5427          * @param {String} defaultValue
5428          */
5429         selectValue : function(path, root, defaultValue){
5430             path = path.replace(trimRe, "");
5431             if(!valueCache[path]){
5432                 valueCache[path] = Roo.DomQuery.compile(path, "select");
5433             }
5434             var n = valueCache[path](root);
5435             n = n[0] ? n[0] : n;
5436             var v = (n && n.firstChild ? n.firstChild.nodeValue : null);
5437             return ((v === null||v === undefined||v==='') ? defaultValue : v);
5438         },
5439
5440         /**
5441          * Selects the value of a node, parsing integers and floats.
5442          * @param {String} selector The selector/xpath query
5443          * @param {Node} root (optional) The start of the query (defaults to document).
5444          * @param {Number} defaultValue
5445          * @return {Number}
5446          */
5447         selectNumber : function(path, root, defaultValue){
5448             var v = Roo.DomQuery.selectValue(path, root, defaultValue || 0);
5449             return parseFloat(v);
5450         },
5451
5452         /**
5453          * Returns true if the passed element(s) match the passed simple selector (e.g. div.some-class or span:first-child)
5454          * @param {String/HTMLElement/Array} el An element id, element or array of elements
5455          * @param {String} selector The simple selector to test
5456          * @return {Boolean}
5457          */
5458         is : function(el, ss){
5459             if(typeof el == "string"){
5460                 el = document.getElementById(el);
5461             }
5462             var isArray = (el instanceof Array);
5463             var result = Roo.DomQuery.filter(isArray ? el : [el], ss);
5464             return isArray ? (result.length == el.length) : (result.length > 0);
5465         },
5466
5467         /**
5468          * Filters an array of elements to only include matches of a simple selector (e.g. div.some-class or span:first-child)
5469          * @param {Array} el An array of elements to filter
5470          * @param {String} selector The simple selector to test
5471          * @param {Boolean} nonMatches If true, it returns the elements that DON'T match
5472          * the selector instead of the ones that match
5473          * @return {Array}
5474          */
5475         filter : function(els, ss, nonMatches){
5476             ss = ss.replace(trimRe, "");
5477             if(!simpleCache[ss]){
5478                 simpleCache[ss] = Roo.DomQuery.compile(ss, "simple");
5479             }
5480             var result = simpleCache[ss](els);
5481             return nonMatches ? quickDiff(result, els) : result;
5482         },
5483
5484         /**
5485          * Collection of matching regular expressions and code snippets.
5486          */
5487         matchers : [{
5488                 re: /^\.([\w-]+)/,
5489                 select: 'n = byClassName(n, null, " {1} ");'
5490             }, {
5491                 re: /^\:([\w-]+)(?:\(((?:[^\s>\/]*|.*?))\))?/,
5492                 select: 'n = byPseudo(n, "{1}", "{2}");'
5493             },{
5494                 re: /^(?:([\[\{])(?:@)?([\w-]+)\s?(?:(=|.=)\s?['"]?(.*?)["']?)?[\]\}])/,
5495                 select: 'n = byAttribute(n, "{2}", "{4}", "{3}", "{1}");'
5496             }, {
5497                 re: /^#([\w-]+)/,
5498                 select: 'n = byId(n, null, "{1}");'
5499             },{
5500                 re: /^@([\w-]+)/,
5501                 select: 'return {firstChild:{nodeValue:attrValue(n, "{1}")}};'
5502             }
5503         ],
5504
5505         /**
5506          * Collection of operator comparison functions. The default operators are =, !=, ^=, $=, *=, %=, |= and ~=.
5507          * 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;.
5508          */
5509         operators : {
5510             "=" : function(a, v){
5511                 return a == v;
5512             },
5513             "!=" : function(a, v){
5514                 return a != v;
5515             },
5516             "^=" : function(a, v){
5517                 return a && a.substr(0, v.length) == v;
5518             },
5519             "$=" : function(a, v){
5520                 return a && a.substr(a.length-v.length) == v;
5521             },
5522             "*=" : function(a, v){
5523                 return a && a.indexOf(v) !== -1;
5524             },
5525             "%=" : function(a, v){
5526                 return (a % v) == 0;
5527             },
5528             "|=" : function(a, v){
5529                 return a && (a == v || a.substr(0, v.length+1) == v+'-');
5530             },
5531             "~=" : function(a, v){
5532                 return a && (' '+a+' ').indexOf(' '+v+' ') != -1;
5533             }
5534         },
5535
5536         /**
5537          * Collection of "pseudo class" processors. Each processor is passed the current nodeset (array)
5538          * and the argument (if any) supplied in the selector.
5539          */
5540         pseudos : {
5541             "first-child" : function(c){
5542                 var r = [], ri = -1, n;
5543                 for(var i = 0, ci; ci = n = c[i]; i++){
5544                     while((n = n.previousSibling) && n.nodeType != 1);
5545                     if(!n){
5546                         r[++ri] = ci;
5547                     }
5548                 }
5549                 return r;
5550             },
5551
5552             "last-child" : function(c){
5553                 var r = [], ri = -1, n;
5554                 for(var i = 0, ci; ci = n = c[i]; i++){
5555                     while((n = n.nextSibling) && n.nodeType != 1);
5556                     if(!n){
5557                         r[++ri] = ci;
5558                     }
5559                 }
5560                 return r;
5561             },
5562
5563             "nth-child" : function(c, a) {
5564                 var r = [], ri = -1;
5565                 var m = nthRe.exec(a == "even" && "2n" || a == "odd" && "2n+1" || !nthRe2.test(a) && "n+" + a || a);
5566                 var f = (m[1] || 1) - 0, l = m[2] - 0;
5567                 for(var i = 0, n; n = c[i]; i++){
5568                     var pn = n.parentNode;
5569                     if (batch != pn._batch) {
5570                         var j = 0;
5571                         for(var cn = pn.firstChild; cn; cn = cn.nextSibling){
5572                             if(cn.nodeType == 1){
5573                                cn.nodeIndex = ++j;
5574                             }
5575                         }
5576                         pn._batch = batch;
5577                     }
5578                     if (f == 1) {
5579                         if (l == 0 || n.nodeIndex == l){
5580                             r[++ri] = n;
5581                         }
5582                     } else if ((n.nodeIndex + l) % f == 0){
5583                         r[++ri] = n;
5584                     }
5585                 }
5586
5587                 return r;
5588             },
5589
5590             "only-child" : function(c){
5591                 var r = [], ri = -1;;
5592                 for(var i = 0, ci; ci = c[i]; i++){
5593                     if(!prev(ci) && !next(ci)){
5594                         r[++ri] = ci;
5595                     }
5596                 }
5597                 return r;
5598             },
5599
5600             "empty" : function(c){
5601                 var r = [], ri = -1;
5602                 for(var i = 0, ci; ci = c[i]; i++){
5603                     var cns = ci.childNodes, j = 0, cn, empty = true;
5604                     while(cn = cns[j]){
5605                         ++j;
5606                         if(cn.nodeType == 1 || cn.nodeType == 3){
5607                             empty = false;
5608                             break;
5609                         }
5610                     }
5611                     if(empty){
5612                         r[++ri] = ci;
5613                     }
5614                 }
5615                 return r;
5616             },
5617
5618             "contains" : function(c, v){
5619                 var r = [], ri = -1;
5620                 for(var i = 0, ci; ci = c[i]; i++){
5621                     if((ci.textContent||ci.innerText||'').indexOf(v) != -1){
5622                         r[++ri] = ci;
5623                     }
5624                 }
5625                 return r;
5626             },
5627
5628             "nodeValue" : function(c, v){
5629                 var r = [], ri = -1;
5630                 for(var i = 0, ci; ci = c[i]; i++){
5631                     if(ci.firstChild && ci.firstChild.nodeValue == v){
5632                         r[++ri] = ci;
5633                     }
5634                 }
5635                 return r;
5636             },
5637
5638             "checked" : function(c){
5639                 var r = [], ri = -1;
5640                 for(var i = 0, ci; ci = c[i]; i++){
5641                     if(ci.checked == true){
5642                         r[++ri] = ci;
5643                     }
5644                 }
5645                 return r;
5646             },
5647
5648             "not" : function(c, ss){
5649                 return Roo.DomQuery.filter(c, ss, true);
5650             },
5651
5652             "odd" : function(c){
5653                 return this["nth-child"](c, "odd");
5654             },
5655
5656             "even" : function(c){
5657                 return this["nth-child"](c, "even");
5658             },
5659
5660             "nth" : function(c, a){
5661                 return c[a-1] || [];
5662             },
5663
5664             "first" : function(c){
5665                 return c[0] || [];
5666             },
5667
5668             "last" : function(c){
5669                 return c[c.length-1] || [];
5670             },
5671
5672             "has" : function(c, ss){
5673                 var s = Roo.DomQuery.select;
5674                 var r = [], ri = -1;
5675                 for(var i = 0, ci; ci = c[i]; i++){
5676                     if(s(ss, ci).length > 0){
5677                         r[++ri] = ci;
5678                     }
5679                 }
5680                 return r;
5681             },
5682
5683             "next" : function(c, ss){
5684                 var is = Roo.DomQuery.is;
5685                 var r = [], ri = -1;
5686                 for(var i = 0, ci; ci = c[i]; i++){
5687                     var n = next(ci);
5688                     if(n && is(n, ss)){
5689                         r[++ri] = ci;
5690                     }
5691                 }
5692                 return r;
5693             },
5694
5695             "prev" : function(c, ss){
5696                 var is = Roo.DomQuery.is;
5697                 var r = [], ri = -1;
5698                 for(var i = 0, ci; ci = c[i]; i++){
5699                     var n = prev(ci);
5700                     if(n && is(n, ss)){
5701                         r[++ri] = ci;
5702                     }
5703                 }
5704                 return r;
5705             }
5706         }
5707     };
5708 }();
5709
5710 /**
5711  * Selects an array of DOM nodes by CSS/XPath selector. Shorthand of {@link Roo.DomQuery#select}
5712  * @param {String} path The selector/xpath query
5713  * @param {Node} root (optional) The start of the query (defaults to document).
5714  * @return {Array}
5715  * @member Roo
5716  * @method query
5717  */
5718 Roo.query = Roo.DomQuery.select;
5719 /*
5720  * Based on:
5721  * Ext JS Library 1.1.1
5722  * Copyright(c) 2006-2007, Ext JS, LLC.
5723  *
5724  * Originally Released Under LGPL - original licence link has changed is not relivant.
5725  *
5726  * Fork - LGPL
5727  * <script type="text/javascript">
5728  */
5729
5730 /**
5731  * @class Roo.util.Observable
5732  * Base class that provides a common interface for publishing events. Subclasses are expected to
5733  * to have a property "events" with all the events defined.<br>
5734  * For example:
5735  * <pre><code>
5736  Employee = function(name){
5737     this.name = name;
5738     this.addEvents({
5739         "fired" : true,
5740         "quit" : true
5741     });
5742  }
5743  Roo.extend(Employee, Roo.util.Observable);
5744 </code></pre>
5745  * @param {Object} config properties to use (incuding events / listeners)
5746  */
5747
5748 Roo.util.Observable = function(cfg){
5749     
5750     cfg = cfg|| {};
5751     this.addEvents(cfg.events || {});
5752     if (cfg.events) {
5753         delete cfg.events; // make sure
5754     }
5755      
5756     Roo.apply(this, cfg);
5757     
5758     if(this.listeners){
5759         this.on(this.listeners);
5760         delete this.listeners;
5761     }
5762 };
5763 Roo.util.Observable.prototype = {
5764     /** 
5765  * @cfg {Object} listeners  list of events and functions to call for this object, 
5766  * For example :
5767  * <pre><code>
5768     listeners :  { 
5769        'click' : function(e) {
5770            ..... 
5771         } ,
5772         .... 
5773     } 
5774   </code></pre>
5775  */
5776     
5777     
5778     /**
5779      * Fires the specified event with the passed parameters (minus the event name).
5780      * @param {String} eventName
5781      * @param {Object...} args Variable number of parameters are passed to handlers
5782      * @return {Boolean} returns false if any of the handlers return false otherwise it returns true
5783      */
5784     fireEvent : function(){
5785         var ce = this.events[arguments[0].toLowerCase()];
5786         if(typeof ce == "object"){
5787             return ce.fire.apply(ce, Array.prototype.slice.call(arguments, 1));
5788         }else{
5789             return true;
5790         }
5791     },
5792
5793     // private
5794     filterOptRe : /^(?:scope|delay|buffer|single)$/,
5795
5796     /**
5797      * Appends an event handler to this component
5798      * @param {String}   eventName The type of event to listen for
5799      * @param {Function} handler The method the event invokes
5800      * @param {Object}   scope (optional) The scope in which to execute the handler
5801      * function. The handler function's "this" context.
5802      * @param {Object}   options (optional) An object containing handler configuration
5803      * properties. This may contain any of the following properties:<ul>
5804      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
5805      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
5806      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
5807      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
5808      * by the specified number of milliseconds. If the event fires again within that time, the original
5809      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
5810      * </ul><br>
5811      * <p>
5812      * <b>Combining Options</b><br>
5813      * Using the options argument, it is possible to combine different types of listeners:<br>
5814      * <br>
5815      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)
5816                 <pre><code>
5817                 el.on('click', this.onClick, this, {
5818                         single: true,
5819                 delay: 100,
5820                 forumId: 4
5821                 });
5822                 </code></pre>
5823      * <p>
5824      * <b>Attaching multiple handlers in 1 call</b><br>
5825      * The method also allows for a single argument to be passed which is a config object containing properties
5826      * which specify multiple handlers.
5827      * <pre><code>
5828                 el.on({
5829                         'click': {
5830                         fn: this.onClick,
5831                         scope: this,
5832                         delay: 100
5833                 }, 
5834                 'mouseover': {
5835                         fn: this.onMouseOver,
5836                         scope: this
5837                 },
5838                 'mouseout': {
5839                         fn: this.onMouseOut,
5840                         scope: this
5841                 }
5842                 });
5843                 </code></pre>
5844      * <p>
5845      * Or a shorthand syntax which passes the same scope object to all handlers:
5846         <pre><code>
5847                 el.on({
5848                         'click': this.onClick,
5849                 'mouseover': this.onMouseOver,
5850                 'mouseout': this.onMouseOut,
5851                 scope: this
5852                 });
5853                 </code></pre>
5854      */
5855     addListener : function(eventName, fn, scope, o){
5856         if(typeof eventName == "object"){
5857             o = eventName;
5858             for(var e in o){
5859                 if(this.filterOptRe.test(e)){
5860                     continue;
5861                 }
5862                 if(typeof o[e] == "function"){
5863                     // shared options
5864                     this.addListener(e, o[e], o.scope,  o);
5865                 }else{
5866                     // individual options
5867                     this.addListener(e, o[e].fn, o[e].scope, o[e]);
5868                 }
5869             }
5870             return;
5871         }
5872         o = (!o || typeof o == "boolean") ? {} : o;
5873         eventName = eventName.toLowerCase();
5874         var ce = this.events[eventName] || true;
5875         if(typeof ce == "boolean"){
5876             ce = new Roo.util.Event(this, eventName);
5877             this.events[eventName] = ce;
5878         }
5879         ce.addListener(fn, scope, o);
5880     },
5881
5882     /**
5883      * Removes a listener
5884      * @param {String}   eventName     The type of event to listen for
5885      * @param {Function} handler        The handler to remove
5886      * @param {Object}   scope  (optional) The scope (this object) for the handler
5887      */
5888     removeListener : function(eventName, fn, scope){
5889         var ce = this.events[eventName.toLowerCase()];
5890         if(typeof ce == "object"){
5891             ce.removeListener(fn, scope);
5892         }
5893     },
5894
5895     /**
5896      * Removes all listeners for this object
5897      */
5898     purgeListeners : function(){
5899         for(var evt in this.events){
5900             if(typeof this.events[evt] == "object"){
5901                  this.events[evt].clearListeners();
5902             }
5903         }
5904     },
5905
5906     relayEvents : function(o, events){
5907         var createHandler = function(ename){
5908             return function(){
5909                 return this.fireEvent.apply(this, Roo.combine(ename, Array.prototype.slice.call(arguments, 0)));
5910             };
5911         };
5912         for(var i = 0, len = events.length; i < len; i++){
5913             var ename = events[i];
5914             if(!this.events[ename]){ this.events[ename] = true; };
5915             o.on(ename, createHandler(ename), this);
5916         }
5917     },
5918
5919     /**
5920      * Used to define events on this Observable
5921      * @param {Object} object The object with the events defined
5922      */
5923     addEvents : function(o){
5924         if(!this.events){
5925             this.events = {};
5926         }
5927         Roo.applyIf(this.events, o);
5928     },
5929
5930     /**
5931      * Checks to see if this object has any listeners for a specified event
5932      * @param {String} eventName The name of the event to check for
5933      * @return {Boolean} True if the event is being listened for, else false
5934      */
5935     hasListener : function(eventName){
5936         var e = this.events[eventName];
5937         return typeof e == "object" && e.listeners.length > 0;
5938     }
5939 };
5940 /**
5941  * Appends an event handler to this element (shorthand for addListener)
5942  * @param {String}   eventName     The type of event to listen for
5943  * @param {Function} handler        The method the event invokes
5944  * @param {Object}   scope (optional) The scope in which to execute the handler
5945  * function. The handler function's "this" context.
5946  * @param {Object}   options  (optional)
5947  * @method
5948  */
5949 Roo.util.Observable.prototype.on = Roo.util.Observable.prototype.addListener;
5950 /**
5951  * Removes a listener (shorthand for removeListener)
5952  * @param {String}   eventName     The type of event to listen for
5953  * @param {Function} handler        The handler to remove
5954  * @param {Object}   scope  (optional) The scope (this object) for the handler
5955  * @method
5956  */
5957 Roo.util.Observable.prototype.un = Roo.util.Observable.prototype.removeListener;
5958
5959 /**
5960  * Starts capture on the specified Observable. All events will be passed
5961  * to the supplied function with the event name + standard signature of the event
5962  * <b>before</b> the event is fired. If the supplied function returns false,
5963  * the event will not fire.
5964  * @param {Observable} o The Observable to capture
5965  * @param {Function} fn The function to call
5966  * @param {Object} scope (optional) The scope (this object) for the fn
5967  * @static
5968  */
5969 Roo.util.Observable.capture = function(o, fn, scope){
5970     o.fireEvent = o.fireEvent.createInterceptor(fn, scope);
5971 };
5972
5973 /**
5974  * Removes <b>all</b> added captures from the Observable.
5975  * @param {Observable} o The Observable to release
5976  * @static
5977  */
5978 Roo.util.Observable.releaseCapture = function(o){
5979     o.fireEvent = Roo.util.Observable.prototype.fireEvent;
5980 };
5981
5982 (function(){
5983
5984     var createBuffered = function(h, o, scope){
5985         var task = new Roo.util.DelayedTask();
5986         return function(){
5987             task.delay(o.buffer, h, scope, Array.prototype.slice.call(arguments, 0));
5988         };
5989     };
5990
5991     var createSingle = function(h, e, fn, scope){
5992         return function(){
5993             e.removeListener(fn, scope);
5994             return h.apply(scope, arguments);
5995         };
5996     };
5997
5998     var createDelayed = function(h, o, scope){
5999         return function(){
6000             var args = Array.prototype.slice.call(arguments, 0);
6001             setTimeout(function(){
6002                 h.apply(scope, args);
6003             }, o.delay || 10);
6004         };
6005     };
6006
6007     Roo.util.Event = function(obj, name){
6008         this.name = name;
6009         this.obj = obj;
6010         this.listeners = [];
6011     };
6012
6013     Roo.util.Event.prototype = {
6014         addListener : function(fn, scope, options){
6015             var o = options || {};
6016             scope = scope || this.obj;
6017             if(!this.isListening(fn, scope)){
6018                 var l = {fn: fn, scope: scope, options: o};
6019                 var h = fn;
6020                 if(o.delay){
6021                     h = createDelayed(h, o, scope);
6022                 }
6023                 if(o.single){
6024                     h = createSingle(h, this, fn, scope);
6025                 }
6026                 if(o.buffer){
6027                     h = createBuffered(h, o, scope);
6028                 }
6029                 l.fireFn = h;
6030                 if(!this.firing){ // if we are currently firing this event, don't disturb the listener loop
6031                     this.listeners.push(l);
6032                 }else{
6033                     this.listeners = this.listeners.slice(0);
6034                     this.listeners.push(l);
6035                 }
6036             }
6037         },
6038
6039         findListener : function(fn, scope){
6040             scope = scope || this.obj;
6041             var ls = this.listeners;
6042             for(var i = 0, len = ls.length; i < len; i++){
6043                 var l = ls[i];
6044                 if(l.fn == fn && l.scope == scope){
6045                     return i;
6046                 }
6047             }
6048             return -1;
6049         },
6050
6051         isListening : function(fn, scope){
6052             return this.findListener(fn, scope) != -1;
6053         },
6054
6055         removeListener : function(fn, scope){
6056             var index;
6057             if((index = this.findListener(fn, scope)) != -1){
6058                 if(!this.firing){
6059                     this.listeners.splice(index, 1);
6060                 }else{
6061                     this.listeners = this.listeners.slice(0);
6062                     this.listeners.splice(index, 1);
6063                 }
6064                 return true;
6065             }
6066             return false;
6067         },
6068
6069         clearListeners : function(){
6070             this.listeners = [];
6071         },
6072
6073         fire : function(){
6074             var ls = this.listeners, scope, len = ls.length;
6075             if(len > 0){
6076                 this.firing = true;
6077                 
6078                 for(var i = 0; i < len; i++){
6079                     var args = Array.prototype.slice.call(arguments, 0);
6080                     var l = ls[i];
6081                     args.push(l.options);
6082                     if(l.fireFn.apply(l.scope||this.obj||window, args) === false){
6083                         this.firing = false;
6084                         return false;
6085                     }
6086                 }
6087                 this.firing = false;
6088             }
6089             return true;
6090         }
6091     };
6092 })();/*
6093  * RooJS Library 
6094  * Copyright(c) 2007-2017, Roo J Solutions Ltd
6095  *
6096  * Licence LGPL 
6097  *
6098  */
6099  
6100 /**
6101  * @class Roo.Document
6102  * @extends Roo.util.Observable
6103  * This is a convience class to wrap up the main document loading code.. , rather than adding Roo.onReady(......)
6104  * 
6105  * @param {Object} config the methods and properties of the 'base' class for the application.
6106  * 
6107  *  Generic Page handler - implement this to start your app..
6108  * 
6109  * eg.
6110  *  MyProject = new Roo.Document({
6111         events : {
6112             'load' : true // your events..
6113         },
6114         listeners : {
6115             'ready' : function() {
6116                 // fired on Roo.onReady()
6117             }
6118         }
6119  * 
6120  */
6121 Roo.Document = function(cfg) {
6122      
6123     this.addEvents({ 
6124         'ready' : true
6125     });
6126     Roo.util.Observable.call(this,cfg);
6127     
6128     var _this = this;
6129     
6130     Roo.onReady(function() {
6131         _this.fireEvent('ready');
6132     },null,false);
6133     
6134     
6135 }
6136
6137 Roo.extend(Roo.Document, Roo.util.Observable, {});/*
6138  * Based on:
6139  * Ext JS Library 1.1.1
6140  * Copyright(c) 2006-2007, Ext JS, LLC.
6141  *
6142  * Originally Released Under LGPL - original licence link has changed is not relivant.
6143  *
6144  * Fork - LGPL
6145  * <script type="text/javascript">
6146  */
6147
6148 /**
6149  * @class Roo.EventManager
6150  * Registers event handlers that want to receive a normalized EventObject instead of the standard browser event and provides 
6151  * several useful events directly.
6152  * See {@link Roo.EventObject} for more details on normalized event objects.
6153  * @singleton
6154  */
6155 Roo.EventManager = function(){
6156     var docReadyEvent, docReadyProcId, docReadyState = false;
6157     var resizeEvent, resizeTask, textEvent, textSize;
6158     var E = Roo.lib.Event;
6159     var D = Roo.lib.Dom;
6160
6161     
6162     
6163
6164     var fireDocReady = function(){
6165         if(!docReadyState){
6166             docReadyState = true;
6167             Roo.isReady = true;
6168             if(docReadyProcId){
6169                 clearInterval(docReadyProcId);
6170             }
6171             if(Roo.isGecko || Roo.isOpera) {
6172                 document.removeEventListener("DOMContentLoaded", fireDocReady, false);
6173             }
6174             if(Roo.isIE){
6175                 var defer = document.getElementById("ie-deferred-loader");
6176                 if(defer){
6177                     defer.onreadystatechange = null;
6178                     defer.parentNode.removeChild(defer);
6179                 }
6180             }
6181             if(docReadyEvent){
6182                 docReadyEvent.fire();
6183                 docReadyEvent.clearListeners();
6184             }
6185         }
6186     };
6187     
6188     var initDocReady = function(){
6189         docReadyEvent = new Roo.util.Event();
6190         if(Roo.isGecko || Roo.isOpera) {
6191             document.addEventListener("DOMContentLoaded", fireDocReady, false);
6192         }else if(Roo.isIE){
6193             document.write("<s"+'cript id="ie-deferred-loader" defer="defer" src="/'+'/:"></s'+"cript>");
6194             var defer = document.getElementById("ie-deferred-loader");
6195             defer.onreadystatechange = function(){
6196                 if(this.readyState == "complete"){
6197                     fireDocReady();
6198                 }
6199             };
6200         }else if(Roo.isSafari){ 
6201             docReadyProcId = setInterval(function(){
6202                 var rs = document.readyState;
6203                 if(rs == "complete") {
6204                     fireDocReady();     
6205                  }
6206             }, 10);
6207         }
6208         // no matter what, make sure it fires on load
6209         E.on(window, "load", fireDocReady);
6210     };
6211
6212     var createBuffered = function(h, o){
6213         var task = new Roo.util.DelayedTask(h);
6214         return function(e){
6215             // create new event object impl so new events don't wipe out properties
6216             e = new Roo.EventObjectImpl(e);
6217             task.delay(o.buffer, h, null, [e]);
6218         };
6219     };
6220
6221     var createSingle = function(h, el, ename, fn){
6222         return function(e){
6223             Roo.EventManager.removeListener(el, ename, fn);
6224             h(e);
6225         };
6226     };
6227
6228     var createDelayed = function(h, o){
6229         return function(e){
6230             // create new event object impl so new events don't wipe out properties
6231             e = new Roo.EventObjectImpl(e);
6232             setTimeout(function(){
6233                 h(e);
6234             }, o.delay || 10);
6235         };
6236     };
6237     var transitionEndVal = false;
6238     
6239     var transitionEnd = function()
6240     {
6241         if (transitionEndVal) {
6242             return transitionEndVal;
6243         }
6244         var el = document.createElement('div');
6245
6246         var transEndEventNames = {
6247             WebkitTransition : 'webkitTransitionEnd',
6248             MozTransition    : 'transitionend',
6249             OTransition      : 'oTransitionEnd otransitionend',
6250             transition       : 'transitionend'
6251         };
6252     
6253         for (var name in transEndEventNames) {
6254             if (el.style[name] !== undefined) {
6255                 transitionEndVal = transEndEventNames[name];
6256                 return  transitionEndVal ;
6257             }
6258         }
6259     }
6260     
6261
6262     var listen = function(element, ename, opt, fn, scope){
6263         var o = (!opt || typeof opt == "boolean") ? {} : opt;
6264         fn = fn || o.fn; scope = scope || o.scope;
6265         var el = Roo.getDom(element);
6266         
6267         
6268         if(!el){
6269             throw "Error listening for \"" + ename + '\". Element "' + element + '" doesn\'t exist.';
6270         }
6271         
6272         if (ename == 'transitionend') {
6273             ename = transitionEnd();
6274         }
6275         var h = function(e){
6276             e = Roo.EventObject.setEvent(e);
6277             var t;
6278             if(o.delegate){
6279                 t = e.getTarget(o.delegate, el);
6280                 if(!t){
6281                     return;
6282                 }
6283             }else{
6284                 t = e.target;
6285             }
6286             if(o.stopEvent === true){
6287                 e.stopEvent();
6288             }
6289             if(o.preventDefault === true){
6290                e.preventDefault();
6291             }
6292             if(o.stopPropagation === true){
6293                 e.stopPropagation();
6294             }
6295
6296             if(o.normalized === false){
6297                 e = e.browserEvent;
6298             }
6299
6300             fn.call(scope || el, e, t, o);
6301         };
6302         if(o.delay){
6303             h = createDelayed(h, o);
6304         }
6305         if(o.single){
6306             h = createSingle(h, el, ename, fn);
6307         }
6308         if(o.buffer){
6309             h = createBuffered(h, o);
6310         }
6311         
6312         fn._handlers = fn._handlers || [];
6313         
6314         
6315         fn._handlers.push([Roo.id(el), ename, h]);
6316         
6317         
6318          
6319         E.on(el, ename, h);
6320         if(ename == "mousewheel" && el.addEventListener){ // workaround for jQuery
6321             el.addEventListener("DOMMouseScroll", h, false);
6322             E.on(window, 'unload', function(){
6323                 el.removeEventListener("DOMMouseScroll", h, false);
6324             });
6325         }
6326         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6327             Roo.EventManager.stoppedMouseDownEvent.addListener(h);
6328         }
6329         return h;
6330     };
6331
6332     var stopListening = function(el, ename, fn){
6333         var id = Roo.id(el), hds = fn._handlers, hd = fn;
6334         if(hds){
6335             for(var i = 0, len = hds.length; i < len; i++){
6336                 var h = hds[i];
6337                 if(h[0] == id && h[1] == ename){
6338                     hd = h[2];
6339                     hds.splice(i, 1);
6340                     break;
6341                 }
6342             }
6343         }
6344         E.un(el, ename, hd);
6345         el = Roo.getDom(el);
6346         if(ename == "mousewheel" && el.addEventListener){
6347             el.removeEventListener("DOMMouseScroll", hd, false);
6348         }
6349         if(ename == "mousedown" && el == document){ // fix stopped mousedowns on the document
6350             Roo.EventManager.stoppedMouseDownEvent.removeListener(hd);
6351         }
6352     };
6353
6354     var propRe = /^(?:scope|delay|buffer|single|stopEvent|preventDefault|stopPropagation|normalized|args|delegate)$/;
6355     
6356     var pub = {
6357         
6358         
6359         /** 
6360          * Fix for doc tools
6361          * @scope Roo.EventManager
6362          */
6363         
6364         
6365         /** 
6366          * This is no longer needed and is deprecated. Places a simple wrapper around an event handler to override the browser event
6367          * object with a Roo.EventObject
6368          * @param {Function} fn        The method the event invokes
6369          * @param {Object}   scope    An object that becomes the scope of the handler
6370          * @param {boolean}  override If true, the obj passed in becomes
6371          *                             the execution scope of the listener
6372          * @return {Function} The wrapped function
6373          * @deprecated
6374          */
6375         wrap : function(fn, scope, override){
6376             return function(e){
6377                 Roo.EventObject.setEvent(e);
6378                 fn.call(override ? scope || window : window, Roo.EventObject, scope);
6379             };
6380         },
6381         
6382         /**
6383      * Appends an event handler to an element (shorthand for addListener)
6384      * @param {String/HTMLElement}   element        The html element or id to assign the
6385      * @param {String}   eventName The type of event to listen for
6386      * @param {Function} handler The method the event invokes
6387      * @param {Object}   scope (optional) The scope in which to execute the handler
6388      * function. The handler function's "this" context.
6389      * @param {Object}   options (optional) An object containing handler configuration
6390      * properties. This may contain any of the following properties:<ul>
6391      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6392      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6393      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6394      * <li>preventDefault {Boolean} True to prevent the default action</li>
6395      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6396      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6397      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6398      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6399      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6400      * by the specified number of milliseconds. If the event fires again within that time, the original
6401      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6402      * </ul><br>
6403      * <p>
6404      * <b>Combining Options</b><br>
6405      * Using the options argument, it is possible to combine different types of listeners:<br>
6406      * <br>
6407      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6408      * Code:<pre><code>
6409 el.on('click', this.onClick, this, {
6410     single: true,
6411     delay: 100,
6412     stopEvent : true,
6413     forumId: 4
6414 });</code></pre>
6415      * <p>
6416      * <b>Attaching multiple handlers in 1 call</b><br>
6417       * The method also allows for a single argument to be passed which is a config object containing properties
6418      * which specify multiple handlers.
6419      * <p>
6420      * Code:<pre><code>
6421 el.on({
6422     'click' : {
6423         fn: this.onClick
6424         scope: this,
6425         delay: 100
6426     },
6427     'mouseover' : {
6428         fn: this.onMouseOver
6429         scope: this
6430     },
6431     'mouseout' : {
6432         fn: this.onMouseOut
6433         scope: this
6434     }
6435 });</code></pre>
6436      * <p>
6437      * Or a shorthand syntax:<br>
6438      * Code:<pre><code>
6439 el.on({
6440     'click' : this.onClick,
6441     'mouseover' : this.onMouseOver,
6442     'mouseout' : this.onMouseOut
6443     scope: this
6444 });</code></pre>
6445      */
6446         addListener : function(element, eventName, fn, scope, options){
6447             if(typeof eventName == "object"){
6448                 var o = eventName;
6449                 for(var e in o){
6450                     if(propRe.test(e)){
6451                         continue;
6452                     }
6453                     if(typeof o[e] == "function"){
6454                         // shared options
6455                         listen(element, e, o, o[e], o.scope);
6456                     }else{
6457                         // individual options
6458                         listen(element, e, o[e]);
6459                     }
6460                 }
6461                 return;
6462             }
6463             return listen(element, eventName, options, fn, scope);
6464         },
6465         
6466         /**
6467          * Removes an event handler
6468          *
6469          * @param {String/HTMLElement}   element        The id or html element to remove the 
6470          *                             event from
6471          * @param {String}   eventName     The type of event
6472          * @param {Function} fn
6473          * @return {Boolean} True if a listener was actually removed
6474          */
6475         removeListener : function(element, eventName, fn){
6476             return stopListening(element, eventName, fn);
6477         },
6478         
6479         /**
6480          * Fires when the document is ready (before onload and before images are loaded). Can be 
6481          * accessed shorthanded Roo.onReady().
6482          * @param {Function} fn        The method the event invokes
6483          * @param {Object}   scope    An  object that becomes the scope of the handler
6484          * @param {boolean}  options
6485          */
6486         onDocumentReady : function(fn, scope, options){
6487             if(docReadyState){ // if it already fired
6488                 docReadyEvent.addListener(fn, scope, options);
6489                 docReadyEvent.fire();
6490                 docReadyEvent.clearListeners();
6491                 return;
6492             }
6493             if(!docReadyEvent){
6494                 initDocReady();
6495             }
6496             docReadyEvent.addListener(fn, scope, options);
6497         },
6498         
6499         /**
6500          * Fires when the window is resized and provides resize event buffering (50 milliseconds), passes new viewport width and height to handlers.
6501          * @param {Function} fn        The method the event invokes
6502          * @param {Object}   scope    An object that becomes the scope of the handler
6503          * @param {boolean}  options
6504          */
6505         onWindowResize : function(fn, scope, options){
6506             if(!resizeEvent){
6507                 resizeEvent = new Roo.util.Event();
6508                 resizeTask = new Roo.util.DelayedTask(function(){
6509                     resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6510                 });
6511                 E.on(window, "resize", function(){
6512                     if(Roo.isIE){
6513                         resizeTask.delay(50);
6514                     }else{
6515                         resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6516                     }
6517                 });
6518             }
6519             resizeEvent.addListener(fn, scope, options);
6520         },
6521
6522         /**
6523          * Fires when the user changes the active text size. Handler gets called with 2 params, the old size and the new size.
6524          * @param {Function} fn        The method the event invokes
6525          * @param {Object}   scope    An object that becomes the scope of the handler
6526          * @param {boolean}  options
6527          */
6528         onTextResize : function(fn, scope, options){
6529             if(!textEvent){
6530                 textEvent = new Roo.util.Event();
6531                 var textEl = new Roo.Element(document.createElement('div'));
6532                 textEl.dom.className = 'x-text-resize';
6533                 textEl.dom.innerHTML = 'X';
6534                 textEl.appendTo(document.body);
6535                 textSize = textEl.dom.offsetHeight;
6536                 setInterval(function(){
6537                     if(textEl.dom.offsetHeight != textSize){
6538                         textEvent.fire(textSize, textSize = textEl.dom.offsetHeight);
6539                     }
6540                 }, this.textResizeInterval);
6541             }
6542             textEvent.addListener(fn, scope, options);
6543         },
6544
6545         /**
6546          * Removes the passed window resize listener.
6547          * @param {Function} fn        The method the event invokes
6548          * @param {Object}   scope    The scope of handler
6549          */
6550         removeResizeListener : function(fn, scope){
6551             if(resizeEvent){
6552                 resizeEvent.removeListener(fn, scope);
6553             }
6554         },
6555
6556         // private
6557         fireResize : function(){
6558             if(resizeEvent){
6559                 resizeEvent.fire(D.getViewWidth(), D.getViewHeight());
6560             }   
6561         },
6562         /**
6563          * Url used for onDocumentReady with using SSL (defaults to Roo.SSL_SECURE_URL)
6564          */
6565         ieDeferSrc : false,
6566         /**
6567          * The frequency, in milliseconds, to check for text resize events (defaults to 50)
6568          */
6569         textResizeInterval : 50
6570     };
6571     
6572     /**
6573      * Fix for doc tools
6574      * @scopeAlias pub=Roo.EventManager
6575      */
6576     
6577      /**
6578      * Appends an event handler to an element (shorthand for addListener)
6579      * @param {String/HTMLElement}   element        The html element or id to assign the
6580      * @param {String}   eventName The type of event to listen for
6581      * @param {Function} handler The method the event invokes
6582      * @param {Object}   scope (optional) The scope in which to execute the handler
6583      * function. The handler function's "this" context.
6584      * @param {Object}   options (optional) An object containing handler configuration
6585      * properties. This may contain any of the following properties:<ul>
6586      * <li>scope {Object} The scope in which to execute the handler function. The handler function's "this" context.</li>
6587      * <li>delegate {String} A simple selector to filter the target or look for a descendant of the target</li>
6588      * <li>stopEvent {Boolean} True to stop the event. That is stop propagation, and prevent the default action.</li>
6589      * <li>preventDefault {Boolean} True to prevent the default action</li>
6590      * <li>stopPropagation {Boolean} True to prevent event propagation</li>
6591      * <li>normalized {Boolean} False to pass a browser event to the handler function instead of an Roo.EventObject</li>
6592      * <li>delay {Number} The number of milliseconds to delay the invocation of the handler after te event fires.</li>
6593      * <li>single {Boolean} True to add a handler to handle just the next firing of the event, and then remove itself.</li>
6594      * <li>buffer {Number} Causes the handler to be scheduled to run in an {@link Roo.util.DelayedTask} delayed
6595      * by the specified number of milliseconds. If the event fires again within that time, the original
6596      * handler is <em>not</em> invoked, but the new handler is scheduled in its place.</li>
6597      * </ul><br>
6598      * <p>
6599      * <b>Combining Options</b><br>
6600      * Using the options argument, it is possible to combine different types of listeners:<br>
6601      * <br>
6602      * A normalized, delayed, one-time listener that auto stops the event and passes a custom argument (forumId)<div style="margin: 5px 20px 20px;">
6603      * Code:<pre><code>
6604 el.on('click', this.onClick, this, {
6605     single: true,
6606     delay: 100,
6607     stopEvent : true,
6608     forumId: 4
6609 });</code></pre>
6610      * <p>
6611      * <b>Attaching multiple handlers in 1 call</b><br>
6612       * The method also allows for a single argument to be passed which is a config object containing properties
6613      * which specify multiple handlers.
6614      * <p>
6615      * Code:<pre><code>
6616 el.on({
6617     'click' : {
6618         fn: this.onClick
6619         scope: this,
6620         delay: 100
6621     },
6622     'mouseover' : {
6623         fn: this.onMouseOver
6624         scope: this
6625     },
6626     'mouseout' : {
6627         fn: this.onMouseOut
6628         scope: this
6629     }
6630 });</code></pre>
6631      * <p>
6632      * Or a shorthand syntax:<br>
6633      * Code:<pre><code>
6634 el.on({
6635     'click' : this.onClick,
6636     'mouseover' : this.onMouseOver,
6637     'mouseout' : this.onMouseOut
6638     scope: this
6639 });</code></pre>
6640      */
6641     pub.on = pub.addListener;
6642     pub.un = pub.removeListener;
6643
6644     pub.stoppedMouseDownEvent = new Roo.util.Event();
6645     return pub;
6646 }();
6647 /**
6648   * Fires when the document is ready (before onload and before images are loaded).  Shorthand of {@link Roo.EventManager#onDocumentReady}.
6649   * @param {Function} fn        The method the event invokes
6650   * @param {Object}   scope    An  object that becomes the scope of the handler
6651   * @param {boolean}  override If true, the obj passed in becomes
6652   *                             the execution scope of the listener
6653   * @member Roo
6654   * @method onReady
6655  */
6656 Roo.onReady = Roo.EventManager.onDocumentReady;
6657
6658 Roo.onReady(function(){
6659     var bd = Roo.get(document.body);
6660     if(!bd){ return; }
6661
6662     var cls = [
6663             Roo.isIE ? "roo-ie"
6664             : Roo.isIE11 ? "roo-ie11"
6665             : Roo.isEdge ? "roo-edge"
6666             : Roo.isGecko ? "roo-gecko"
6667             : Roo.isOpera ? "roo-opera"
6668             : Roo.isSafari ? "roo-safari" : ""];
6669
6670     if(Roo.isMac){
6671         cls.push("roo-mac");
6672     }
6673     if(Roo.isLinux){
6674         cls.push("roo-linux");
6675     }
6676     if(Roo.isIOS){
6677         cls.push("roo-ios");
6678     }
6679     if(Roo.isTouch){
6680         cls.push("roo-touch");
6681     }
6682     if(Roo.isBorderBox){
6683         cls.push('roo-border-box');
6684     }
6685     if(Roo.isStrict){ // add to the parent to allow for selectors like ".ext-strict .ext-ie"
6686         var p = bd.dom.parentNode;
6687         if(p){
6688             p.className += ' roo-strict';
6689         }
6690     }
6691     bd.addClass(cls.join(' '));
6692 });
6693
6694 /**
6695  * @class Roo.EventObject
6696  * EventObject exposes the Yahoo! UI Event functionality directly on the object
6697  * passed to your event handler. It exists mostly for convenience. It also fixes the annoying null checks automatically to cleanup your code 
6698  * Example:
6699  * <pre><code>
6700  function handleClick(e){ // e is not a standard event object, it is a Roo.EventObject
6701     e.preventDefault();
6702     var target = e.getTarget();
6703     ...
6704  }
6705  var myDiv = Roo.get("myDiv");
6706  myDiv.on("click", handleClick);
6707  //or
6708  Roo.EventManager.on("myDiv", 'click', handleClick);
6709  Roo.EventManager.addListener("myDiv", 'click', handleClick);
6710  </code></pre>
6711  * @singleton
6712  */
6713 Roo.EventObject = function(){
6714     
6715     var E = Roo.lib.Event;
6716     
6717     // safari keypress events for special keys return bad keycodes
6718     var safariKeys = {
6719         63234 : 37, // left
6720         63235 : 39, // right
6721         63232 : 38, // up
6722         63233 : 40, // down
6723         63276 : 33, // page up
6724         63277 : 34, // page down
6725         63272 : 46, // delete
6726         63273 : 36, // home
6727         63275 : 35  // end
6728     };
6729
6730     // normalize button clicks
6731     var btnMap = Roo.isIE ? {1:0,4:1,2:2} :
6732                 (Roo.isSafari ? {1:0,2:1,3:2} : {0:0,1:1,2:2});
6733
6734     Roo.EventObjectImpl = function(e){
6735         if(e){
6736             this.setEvent(e.browserEvent || e);
6737         }
6738     };
6739     Roo.EventObjectImpl.prototype = {
6740         /**
6741          * Used to fix doc tools.
6742          * @scope Roo.EventObject.prototype
6743          */
6744             
6745
6746         
6747         
6748         /** The normal browser event */
6749         browserEvent : null,
6750         /** The button pressed in a mouse event */
6751         button : -1,
6752         /** True if the shift key was down during the event */
6753         shiftKey : false,
6754         /** True if the control key was down during the event */
6755         ctrlKey : false,
6756         /** True if the alt key was down during the event */
6757         altKey : false,
6758
6759         /** Key constant 
6760         * @type Number */
6761         BACKSPACE : 8,
6762         /** Key constant 
6763         * @type Number */
6764         TAB : 9,
6765         /** Key constant 
6766         * @type Number */
6767         RETURN : 13,
6768         /** Key constant 
6769         * @type Number */
6770         ENTER : 13,
6771         /** Key constant 
6772         * @type Number */
6773         SHIFT : 16,
6774         /** Key constant 
6775         * @type Number */
6776         CONTROL : 17,
6777         /** Key constant 
6778         * @type Number */
6779         ESC : 27,
6780         /** Key constant 
6781         * @type Number */
6782         SPACE : 32,
6783         /** Key constant 
6784         * @type Number */
6785         PAGEUP : 33,
6786         /** Key constant 
6787         * @type Number */
6788         PAGEDOWN : 34,
6789         /** Key constant 
6790         * @type Number */
6791         END : 35,
6792         /** Key constant 
6793         * @type Number */
6794         HOME : 36,
6795         /** Key constant 
6796         * @type Number */
6797         LEFT : 37,
6798         /** Key constant 
6799         * @type Number */
6800         UP : 38,
6801         /** Key constant 
6802         * @type Number */
6803         RIGHT : 39,
6804         /** Key constant 
6805         * @type Number */
6806         DOWN : 40,
6807         /** Key constant 
6808         * @type Number */
6809         DELETE : 46,
6810         /** Key constant 
6811         * @type Number */
6812         F5 : 116,
6813
6814            /** @private */
6815         setEvent : function(e){
6816             if(e == this || (e && e.browserEvent)){ // already wrapped
6817                 return e;
6818             }
6819             this.browserEvent = e;
6820             if(e){
6821                 // normalize buttons
6822                 this.button = e.button ? btnMap[e.button] : (e.which ? e.which-1 : -1);
6823                 if(e.type == 'click' && this.button == -1){
6824                     this.button = 0;
6825                 }
6826                 this.type = e.type;
6827                 this.shiftKey = e.shiftKey;
6828                 // mac metaKey behaves like ctrlKey
6829                 this.ctrlKey = e.ctrlKey || e.metaKey;
6830                 this.altKey = e.altKey;
6831                 // in getKey these will be normalized for the mac
6832                 this.keyCode = e.keyCode;
6833                 // keyup warnings on firefox.
6834                 this.charCode = (e.type == 'keyup' || e.type == 'keydown') ? 0 : e.charCode;
6835                 // cache the target for the delayed and or buffered events
6836                 this.target = E.getTarget(e);
6837                 // same for XY
6838                 this.xy = E.getXY(e);
6839             }else{
6840                 this.button = -1;
6841                 this.shiftKey = false;
6842                 this.ctrlKey = false;
6843                 this.altKey = false;
6844                 this.keyCode = 0;
6845                 this.charCode =0;
6846                 this.target = null;
6847                 this.xy = [0, 0];
6848             }
6849             return this;
6850         },
6851
6852         /**
6853          * Stop the event (preventDefault and stopPropagation)
6854          */
6855         stopEvent : function(){
6856             if(this.browserEvent){
6857                 if(this.browserEvent.type == 'mousedown'){
6858                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6859                 }
6860                 E.stopEvent(this.browserEvent);
6861             }
6862         },
6863
6864         /**
6865          * Prevents the browsers default handling of the event.
6866          */
6867         preventDefault : function(){
6868             if(this.browserEvent){
6869                 E.preventDefault(this.browserEvent);
6870             }
6871         },
6872
6873         /** @private */
6874         isNavKeyPress : function(){
6875             var k = this.keyCode;
6876             k = Roo.isSafari ? (safariKeys[k] || k) : k;
6877             return (k >= 33 && k <= 40) || k == this.RETURN || k == this.TAB || k == this.ESC;
6878         },
6879
6880         isSpecialKey : function(){
6881             var k = this.keyCode;
6882             return (this.type == 'keypress' && this.ctrlKey) || k == 9 || k == 13  || k == 40 || k == 27 ||
6883             (k == 16) || (k == 17) ||
6884             (k >= 18 && k <= 20) ||
6885             (k >= 33 && k <= 35) ||
6886             (k >= 36 && k <= 39) ||
6887             (k >= 44 && k <= 45);
6888         },
6889         /**
6890          * Cancels bubbling of the event.
6891          */
6892         stopPropagation : function(){
6893             if(this.browserEvent){
6894                 if(this.type == 'mousedown'){
6895                     Roo.EventManager.stoppedMouseDownEvent.fire(this);
6896                 }
6897                 E.stopPropagation(this.browserEvent);
6898             }
6899         },
6900
6901         /**
6902          * Gets the key code for the event.
6903          * @return {Number}
6904          */
6905         getCharCode : function(){
6906             return this.charCode || this.keyCode;
6907         },
6908
6909         /**
6910          * Returns a normalized keyCode for the event.
6911          * @return {Number} The key code
6912          */
6913         getKey : function(){
6914             var k = this.keyCode || this.charCode;
6915             return Roo.isSafari ? (safariKeys[k] || k) : k;
6916         },
6917
6918         /**
6919          * Gets the x coordinate of the event.
6920          * @return {Number}
6921          */
6922         getPageX : function(){
6923             return this.xy[0];
6924         },
6925
6926         /**
6927          * Gets the y coordinate of the event.
6928          * @return {Number}
6929          */
6930         getPageY : function(){
6931             return this.xy[1];
6932         },
6933
6934         /**
6935          * Gets the time of the event.
6936          * @return {Number}
6937          */
6938         getTime : function(){
6939             if(this.browserEvent){
6940                 return E.getTime(this.browserEvent);
6941             }
6942             return null;
6943         },
6944
6945         /**
6946          * Gets the page coordinates of the event.
6947          * @return {Array} The xy values like [x, y]
6948          */
6949         getXY : function(){
6950             return this.xy;
6951         },
6952
6953         /**
6954          * Gets the target for the event.
6955          * @param {String} selector (optional) A simple selector to filter the target or look for an ancestor of the target
6956          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
6957                 search as a number or element (defaults to 10 || document.body)
6958          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
6959          * @return {HTMLelement}
6960          */
6961         getTarget : function(selector, maxDepth, returnEl){
6962             return selector ? Roo.fly(this.target).findParent(selector, maxDepth, returnEl) : this.target;
6963         },
6964         /**
6965          * Gets the related target.
6966          * @return {HTMLElement}
6967          */
6968         getRelatedTarget : function(){
6969             if(this.browserEvent){
6970                 return E.getRelatedTarget(this.browserEvent);
6971             }
6972             return null;
6973         },
6974
6975         /**
6976          * Normalizes mouse wheel delta across browsers
6977          * @return {Number} The delta
6978          */
6979         getWheelDelta : function(){
6980             var e = this.browserEvent;
6981             var delta = 0;
6982             if(e.wheelDelta){ /* IE/Opera. */
6983                 delta = e.wheelDelta/120;
6984             }else if(e.detail){ /* Mozilla case. */
6985                 delta = -e.detail/3;
6986             }
6987             return delta;
6988         },
6989
6990         /**
6991          * Returns true if the control, meta, shift or alt key was pressed during this event.
6992          * @return {Boolean}
6993          */
6994         hasModifier : function(){
6995             return !!((this.ctrlKey || this.altKey) || this.shiftKey);
6996         },
6997
6998         /**
6999          * Returns true if the target of this event equals el or is a child of el
7000          * @param {String/HTMLElement/Element} el
7001          * @param {Boolean} related (optional) true to test if the related target is within el instead of the target
7002          * @return {Boolean}
7003          */
7004         within : function(el, related){
7005             var t = this[related ? "getRelatedTarget" : "getTarget"]();
7006             return t && Roo.fly(el).contains(t);
7007         },
7008
7009         getPoint : function(){
7010             return new Roo.lib.Point(this.xy[0], this.xy[1]);
7011         }
7012     };
7013
7014     return new Roo.EventObjectImpl();
7015 }();
7016             
7017     /*
7018  * Based on:
7019  * Ext JS Library 1.1.1
7020  * Copyright(c) 2006-2007, Ext JS, LLC.
7021  *
7022  * Originally Released Under LGPL - original licence link has changed is not relivant.
7023  *
7024  * Fork - LGPL
7025  * <script type="text/javascript">
7026  */
7027
7028  
7029 // was in Composite Element!??!?!
7030  
7031 (function(){
7032     var D = Roo.lib.Dom;
7033     var E = Roo.lib.Event;
7034     var A = Roo.lib.Anim;
7035
7036     // local style camelizing for speed
7037     var propCache = {};
7038     var camelRe = /(-[a-z])/gi;
7039     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
7040     var view = document.defaultView;
7041
7042 /**
7043  * @class Roo.Element
7044  * Represents an Element in the DOM.<br><br>
7045  * Usage:<br>
7046 <pre><code>
7047 var el = Roo.get("my-div");
7048
7049 // or with getEl
7050 var el = getEl("my-div");
7051
7052 // or with a DOM element
7053 var el = Roo.get(myDivElement);
7054 </code></pre>
7055  * Using Roo.get() or getEl() instead of calling the constructor directly ensures you get the same object
7056  * each call instead of constructing a new one.<br><br>
7057  * <b>Animations</b><br />
7058  * Many of the functions for manipulating an element have an optional "animate" parameter. The animate parameter
7059  * should either be a boolean (true) or an object literal with animation options. The animation options are:
7060 <pre>
7061 Option    Default   Description
7062 --------- --------  ---------------------------------------------
7063 duration  .35       The duration of the animation in seconds
7064 easing    easeOut   The YUI easing method
7065 callback  none      A function to execute when the anim completes
7066 scope     this      The scope (this) of the callback function
7067 </pre>
7068 * Also, the Anim object being used for the animation will be set on your options object as "anim", which allows you to stop or
7069 * manipulate the animation. Here's an example:
7070 <pre><code>
7071 var el = Roo.get("my-div");
7072
7073 // no animation
7074 el.setWidth(100);
7075
7076 // default animation
7077 el.setWidth(100, true);
7078
7079 // animation with some options set
7080 el.setWidth(100, {
7081     duration: 1,
7082     callback: this.foo,
7083     scope: this
7084 });
7085
7086 // using the "anim" property to get the Anim object
7087 var opt = {
7088     duration: 1,
7089     callback: this.foo,
7090     scope: this
7091 };
7092 el.setWidth(100, opt);
7093 ...
7094 if(opt.anim.isAnimated()){
7095     opt.anim.stop();
7096 }
7097 </code></pre>
7098 * <b> Composite (Collections of) Elements</b><br />
7099  * For working with collections of Elements, see <a href="Roo.CompositeElement.html">Roo.CompositeElement</a>
7100  * @constructor Create a new Element directly.
7101  * @param {String/HTMLElement} element
7102  * @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).
7103  */
7104     Roo.Element = function(element, forceNew){
7105         var dom = typeof element == "string" ?
7106                 document.getElementById(element) : element;
7107         if(!dom){ // invalid id/element
7108             return null;
7109         }
7110         var id = dom.id;
7111         if(forceNew !== true && id && Roo.Element.cache[id]){ // element object already exists
7112             return Roo.Element.cache[id];
7113         }
7114
7115         /**
7116          * The DOM element
7117          * @type HTMLElement
7118          */
7119         this.dom = dom;
7120
7121         /**
7122          * The DOM element ID
7123          * @type String
7124          */
7125         this.id = id || Roo.id(dom);
7126     };
7127
7128     var El = Roo.Element;
7129
7130     El.prototype = {
7131         /**
7132          * The element's default display mode  (defaults to "")
7133          * @type String
7134          */
7135         originalDisplay : "",
7136
7137         visibilityMode : 1,
7138         /**
7139          * The default unit to append to CSS values where a unit isn't provided (defaults to px).
7140          * @type String
7141          */
7142         defaultUnit : "px",
7143         
7144         /**
7145          * Sets the element's visibility mode. When setVisible() is called it
7146          * will use this to determine whether to set the visibility or the display property.
7147          * @param visMode Element.VISIBILITY or Element.DISPLAY
7148          * @return {Roo.Element} this
7149          */
7150         setVisibilityMode : function(visMode){
7151             this.visibilityMode = visMode;
7152             return this;
7153         },
7154         /**
7155          * Convenience method for setVisibilityMode(Element.DISPLAY)
7156          * @param {String} display (optional) What to set display to when visible
7157          * @return {Roo.Element} this
7158          */
7159         enableDisplayMode : function(display){
7160             this.setVisibilityMode(El.DISPLAY);
7161             if(typeof display != "undefined") { this.originalDisplay = display; }
7162             return this;
7163         },
7164
7165         /**
7166          * 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)
7167          * @param {String} selector The simple selector to test
7168          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7169                 search as a number or element (defaults to 10 || document.body)
7170          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7171          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7172          */
7173         findParent : function(simpleSelector, maxDepth, returnEl){
7174             var p = this.dom, b = document.body, depth = 0, dq = Roo.DomQuery, stopEl;
7175             maxDepth = maxDepth || 50;
7176             if(typeof maxDepth != "number"){
7177                 stopEl = Roo.getDom(maxDepth);
7178                 maxDepth = 10;
7179             }
7180             while(p && p.nodeType == 1 && depth < maxDepth && p != b && p != stopEl){
7181                 if(dq.is(p, simpleSelector)){
7182                     return returnEl ? Roo.get(p) : p;
7183                 }
7184                 depth++;
7185                 p = p.parentNode;
7186             }
7187             return null;
7188         },
7189
7190
7191         /**
7192          * Looks at parent nodes for a match of the passed simple selector (e.g. div.some-class or span:first-child)
7193          * @param {String} selector The simple selector to test
7194          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7195                 search as a number or element (defaults to 10 || document.body)
7196          * @param {Boolean} returnEl (optional) True to return a Roo.Element object instead of DOM node
7197          * @return {HTMLElement} The matching DOM node (or null if no match was found)
7198          */
7199         findParentNode : function(simpleSelector, maxDepth, returnEl){
7200             var p = Roo.fly(this.dom.parentNode, '_internal');
7201             return p ? p.findParent(simpleSelector, maxDepth, returnEl) : null;
7202         },
7203         
7204         /**
7205          * Looks at  the scrollable parent element
7206          */
7207         findScrollableParent : function()
7208         {
7209             var overflowRegex = /(auto|scroll)/;
7210             
7211             if(this.getStyle('position') === 'fixed'){
7212                 return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7213             }
7214             
7215             var excludeStaticParent = this.getStyle('position') === "absolute";
7216             
7217             for (var parent = this; (parent = Roo.get(parent.dom.parentNode));){
7218                 
7219                 if (excludeStaticParent && parent.getStyle('position') === "static") {
7220                     continue;
7221                 }
7222                 
7223                 if (overflowRegex.test(parent.getStyle('overflow') + parent.getStyle('overflow-x') + parent.getStyle('overflow-y'))){
7224                     return parent;
7225                 }
7226                 
7227                 if(parent.dom.nodeName.toLowerCase() == 'body'){
7228                     return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7229                 }
7230             }
7231             
7232             return Roo.isAndroid ? Roo.get(document.documentElement) : Roo.get(document.body);
7233         },
7234
7235         /**
7236          * Walks up the dom looking for a parent node that matches the passed simple selector (e.g. div.some-class or span:first-child).
7237          * This is a shortcut for findParentNode() that always returns an Roo.Element.
7238          * @param {String} selector The simple selector to test
7239          * @param {Number/String/HTMLElement/Element} maxDepth (optional) The max depth to
7240                 search as a number or element (defaults to 10 || document.body)
7241          * @return {Roo.Element} The matching DOM node (or null if no match was found)
7242          */
7243         up : function(simpleSelector, maxDepth){
7244             return this.findParentNode(simpleSelector, maxDepth, true);
7245         },
7246
7247
7248
7249         /**
7250          * Returns true if this element matches the passed simple selector (e.g. div.some-class or span:first-child)
7251          * @param {String} selector The simple selector to test
7252          * @return {Boolean} True if this element matches the selector, else false
7253          */
7254         is : function(simpleSelector){
7255             return Roo.DomQuery.is(this.dom, simpleSelector);
7256         },
7257
7258         /**
7259          * Perform animation on this element.
7260          * @param {Object} args The YUI animation control args
7261          * @param {Float} duration (optional) How long the animation lasts in seconds (defaults to .35)
7262          * @param {Function} onComplete (optional) Function to call when animation completes
7263          * @param {String} easing (optional) Easing method to use (defaults to 'easeOut')
7264          * @param {String} animType (optional) 'run' is the default. Can also be 'color', 'motion', or 'scroll'
7265          * @return {Roo.Element} this
7266          */
7267         animate : function(args, duration, onComplete, easing, animType){
7268             this.anim(args, {duration: duration, callback: onComplete, easing: easing}, animType);
7269             return this;
7270         },
7271
7272         /*
7273          * @private Internal animation call
7274          */
7275         anim : function(args, opt, animType, defaultDur, defaultEase, cb){
7276             animType = animType || 'run';
7277             opt = opt || {};
7278             var anim = Roo.lib.Anim[animType](
7279                 this.dom, args,
7280                 (opt.duration || defaultDur) || .35,
7281                 (opt.easing || defaultEase) || 'easeOut',
7282                 function(){
7283                     Roo.callback(cb, this);
7284                     Roo.callback(opt.callback, opt.scope || this, [this, opt]);
7285                 },
7286                 this
7287             );
7288             opt.anim = anim;
7289             return anim;
7290         },
7291
7292         // private legacy anim prep
7293         preanim : function(a, i){
7294             return !a[i] ? false : (typeof a[i] == "object" ? a[i]: {duration: a[i+1], callback: a[i+2], easing: a[i+3]});
7295         },
7296
7297         /**
7298          * Removes worthless text nodes
7299          * @param {Boolean} forceReclean (optional) By default the element
7300          * keeps track if it has been cleaned already so
7301          * you can call this over and over. However, if you update the element and
7302          * need to force a reclean, you can pass true.
7303          */
7304         clean : function(forceReclean){
7305             if(this.isCleaned && forceReclean !== true){
7306                 return this;
7307             }
7308             var ns = /\S/;
7309             var d = this.dom, n = d.firstChild, ni = -1;
7310             while(n){
7311                 var nx = n.nextSibling;
7312                 if(n.nodeType == 3 && !ns.test(n.nodeValue)){
7313                     d.removeChild(n);
7314                 }else{
7315                     n.nodeIndex = ++ni;
7316                 }
7317                 n = nx;
7318             }
7319             this.isCleaned = true;
7320             return this;
7321         },
7322
7323         // private
7324         calcOffsetsTo : function(el){
7325             el = Roo.get(el);
7326             var d = el.dom;
7327             var restorePos = false;
7328             if(el.getStyle('position') == 'static'){
7329                 el.position('relative');
7330                 restorePos = true;
7331             }
7332             var x = 0, y =0;
7333             var op = this.dom;
7334             while(op && op != d && op.tagName != 'HTML'){
7335                 x+= op.offsetLeft;
7336                 y+= op.offsetTop;
7337                 op = op.offsetParent;
7338             }
7339             if(restorePos){
7340                 el.position('static');
7341             }
7342             return [x, y];
7343         },
7344
7345         /**
7346          * Scrolls this element into view within the passed container.
7347          * @param {String/HTMLElement/Element} container (optional) The container element to scroll (defaults to document.body)
7348          * @param {Boolean} hscroll (optional) False to disable horizontal scroll (defaults to true)
7349          * @return {Roo.Element} this
7350          */
7351         scrollIntoView : function(container, hscroll){
7352             var c = Roo.getDom(container) || document.body;
7353             var el = this.dom;
7354
7355             var o = this.calcOffsetsTo(c),
7356                 l = o[0],
7357                 t = o[1],
7358                 b = t+el.offsetHeight,
7359                 r = l+el.offsetWidth;
7360
7361             var ch = c.clientHeight;
7362             var ct = parseInt(c.scrollTop, 10);
7363             var cl = parseInt(c.scrollLeft, 10);
7364             var cb = ct + ch;
7365             var cr = cl + c.clientWidth;
7366
7367             if(t < ct){
7368                 c.scrollTop = t;
7369             }else if(b > cb){
7370                 c.scrollTop = b-ch;
7371             }
7372
7373             if(hscroll !== false){
7374                 if(l < cl){
7375                     c.scrollLeft = l;
7376                 }else if(r > cr){
7377                     c.scrollLeft = r-c.clientWidth;
7378                 }
7379             }
7380             return this;
7381         },
7382
7383         // private
7384         scrollChildIntoView : function(child, hscroll){
7385             Roo.fly(child, '_scrollChildIntoView').scrollIntoView(this, hscroll);
7386         },
7387
7388         /**
7389          * Measures the element's content height and updates height to match. Note: this function uses setTimeout so
7390          * the new height may not be available immediately.
7391          * @param {Boolean} animate (optional) Animate the transition (defaults to false)
7392          * @param {Float} duration (optional) Length of the animation in seconds (defaults to .35)
7393          * @param {Function} onComplete (optional) Function to call when animation completes
7394          * @param {String} easing (optional) Easing method to use (defaults to easeOut)
7395          * @return {Roo.Element} this
7396          */
7397         autoHeight : function(animate, duration, onComplete, easing){
7398             var oldHeight = this.getHeight();
7399             this.clip();
7400             this.setHeight(1); // force clipping
7401             setTimeout(function(){
7402                 var height = parseInt(this.dom.scrollHeight, 10); // parseInt for Safari
7403                 if(!animate){
7404                     this.setHeight(height);
7405                     this.unclip();
7406                     if(typeof onComplete == "function"){
7407                         onComplete();
7408                     }
7409                 }else{
7410                     this.setHeight(oldHeight); // restore original height
7411                     this.setHeight(height, animate, duration, function(){
7412                         this.unclip();
7413                         if(typeof onComplete == "function") { onComplete(); }
7414                     }.createDelegate(this), easing);
7415                 }
7416             }.createDelegate(this), 0);
7417             return this;
7418         },
7419
7420         /**
7421          * Returns true if this element is an ancestor of the passed element
7422          * @param {HTMLElement/String} el The element to check
7423          * @return {Boolean} True if this element is an ancestor of el, else false
7424          */
7425         contains : function(el){
7426             if(!el){return false;}
7427             return D.isAncestor(this.dom, el.dom ? el.dom : el);
7428         },
7429
7430         /**
7431          * Checks whether the element is currently visible using both visibility and display properties.
7432          * @param {Boolean} deep (optional) True to walk the dom and see if parent elements are hidden (defaults to false)
7433          * @return {Boolean} True if the element is currently visible, else false
7434          */
7435         isVisible : function(deep) {
7436             var vis = !(this.getStyle("visibility") == "hidden" || this.getStyle("display") == "none");
7437             if(deep !== true || !vis){
7438                 return vis;
7439             }
7440             var p = this.dom.parentNode;
7441             while(p && p.tagName.toLowerCase() != "body"){
7442                 if(!Roo.fly(p, '_isVisible').isVisible()){
7443                     return false;
7444                 }
7445                 p = p.parentNode;
7446             }
7447             return true;
7448         },
7449
7450         /**
7451          * Creates a {@link Roo.CompositeElement} for child nodes based on the passed CSS selector (the selector should not contain an id).
7452          * @param {String} selector The CSS selector
7453          * @param {Boolean} unique (optional) True to create a unique Roo.Element for each child (defaults to false, which creates a single shared flyweight object)
7454          * @return {CompositeElement/CompositeElementLite} The composite element
7455          */
7456         select : function(selector, unique){
7457             return El.select(selector, unique, this.dom);
7458         },
7459
7460         /**
7461          * Selects child nodes based on the passed CSS selector (the selector should not contain an id).
7462          * @param {String} selector The CSS selector
7463          * @return {Array} An array of the matched nodes
7464          */
7465         query : function(selector, unique){
7466             return Roo.DomQuery.select(selector, this.dom);
7467         },
7468
7469         /**
7470          * Selects a single child at any depth below this element based on the passed CSS selector (the selector should not contain an id).
7471          * @param {String} selector The CSS selector
7472          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7473          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7474          */
7475         child : function(selector, returnDom){
7476             var n = Roo.DomQuery.selectNode(selector, this.dom);
7477             return returnDom ? n : Roo.get(n);
7478         },
7479
7480         /**
7481          * Selects a single *direct* child based on the passed CSS selector (the selector should not contain an id).
7482          * @param {String} selector The CSS selector
7483          * @param {Boolean} returnDom (optional) True to return the DOM node instead of Roo.Element (defaults to false)
7484          * @return {HTMLElement/Roo.Element} The child Roo.Element (or DOM node if returnDom = true)
7485          */
7486         down : function(selector, returnDom){
7487             var n = Roo.DomQuery.selectNode(" > " + selector, this.dom);
7488             return returnDom ? n : Roo.get(n);
7489         },
7490
7491         /**
7492          * Initializes a {@link Roo.dd.DD} drag drop object for this element.
7493          * @param {String} group The group the DD object is member of
7494          * @param {Object} config The DD config object
7495          * @param {Object} overrides An object containing methods to override/implement on the DD object
7496          * @return {Roo.dd.DD} The DD object
7497          */
7498         initDD : function(group, config, overrides){
7499             var dd = new Roo.dd.DD(Roo.id(this.dom), group, config);
7500             return Roo.apply(dd, overrides);
7501         },
7502
7503         /**
7504          * Initializes a {@link Roo.dd.DDProxy} object for this element.
7505          * @param {String} group The group the DDProxy object is member of
7506          * @param {Object} config The DDProxy config object
7507          * @param {Object} overrides An object containing methods to override/implement on the DDProxy object
7508          * @return {Roo.dd.DDProxy} The DDProxy object
7509          */
7510         initDDProxy : function(group, config, overrides){
7511             var dd = new Roo.dd.DDProxy(Roo.id(this.dom), group, config);
7512             return Roo.apply(dd, overrides);
7513         },
7514
7515         /**
7516          * Initializes a {@link Roo.dd.DDTarget} object for this element.
7517          * @param {String} group The group the DDTarget object is member of
7518          * @param {Object} config The DDTarget config object
7519          * @param {Object} overrides An object containing methods to override/implement on the DDTarget object
7520          * @return {Roo.dd.DDTarget} The DDTarget object
7521          */
7522         initDDTarget : function(group, config, overrides){
7523             var dd = new Roo.dd.DDTarget(Roo.id(this.dom), group, config);
7524             return Roo.apply(dd, overrides);
7525         },
7526
7527         /**
7528          * Sets the visibility of the element (see details). If the visibilityMode is set to Element.DISPLAY, it will use
7529          * the display property to hide the element, otherwise it uses visibility. The default is to hide and show using the visibility property.
7530          * @param {Boolean} visible Whether the element is visible
7531          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7532          * @return {Roo.Element} this
7533          */
7534          setVisible : function(visible, animate){
7535             if(!animate || !A){
7536                 if(this.visibilityMode == El.DISPLAY){
7537                     this.setDisplayed(visible);
7538                 }else{
7539                     this.fixDisplay();
7540                     this.dom.style.visibility = visible ? "visible" : "hidden";
7541                 }
7542             }else{
7543                 // closure for composites
7544                 var dom = this.dom;
7545                 var visMode = this.visibilityMode;
7546                 if(visible){
7547                     this.setOpacity(.01);
7548                     this.setVisible(true);
7549                 }
7550                 this.anim({opacity: { to: (visible?1:0) }},
7551                       this.preanim(arguments, 1),
7552                       null, .35, 'easeIn', function(){
7553                          if(!visible){
7554                              if(visMode == El.DISPLAY){
7555                                  dom.style.display = "none";
7556                              }else{
7557                                  dom.style.visibility = "hidden";
7558                              }
7559                              Roo.get(dom).setOpacity(1);
7560                          }
7561                      });
7562             }
7563             return this;
7564         },
7565
7566         /**
7567          * Returns true if display is not "none"
7568          * @return {Boolean}
7569          */
7570         isDisplayed : function() {
7571             return this.getStyle("display") != "none";
7572         },
7573
7574         /**
7575          * Toggles the element's visibility or display, depending on visibility mode.
7576          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7577          * @return {Roo.Element} this
7578          */
7579         toggle : function(animate){
7580             this.setVisible(!this.isVisible(), this.preanim(arguments, 0));
7581             return this;
7582         },
7583
7584         /**
7585          * Sets the CSS display property. Uses originalDisplay if the specified value is a boolean true.
7586          * @param {Boolean} value Boolean value to display the element using its default display, or a string to set the display directly
7587          * @return {Roo.Element} this
7588          */
7589         setDisplayed : function(value) {
7590             if(typeof value == "boolean"){
7591                value = value ? this.originalDisplay : "none";
7592             }
7593             this.setStyle("display", value);
7594             return this;
7595         },
7596
7597         /**
7598          * Tries to focus the element. Any exceptions are caught and ignored.
7599          * @return {Roo.Element} this
7600          */
7601         focus : function() {
7602             try{
7603                 this.dom.focus();
7604             }catch(e){}
7605             return this;
7606         },
7607
7608         /**
7609          * Tries to blur the element. Any exceptions are caught and ignored.
7610          * @return {Roo.Element} this
7611          */
7612         blur : function() {
7613             try{
7614                 this.dom.blur();
7615             }catch(e){}
7616             return this;
7617         },
7618
7619         /**
7620          * Adds one or more CSS classes to the element. Duplicate classes are automatically filtered out.
7621          * @param {String/Array} className The CSS class to add, or an array of classes
7622          * @return {Roo.Element} this
7623          */
7624         addClass : function(className){
7625             if(className instanceof Array){
7626                 for(var i = 0, len = className.length; i < len; i++) {
7627                     this.addClass(className[i]);
7628                 }
7629             }else{
7630                 if(className && !this.hasClass(className)){
7631                     this.dom.className = this.dom.className + " " + className;
7632                 }
7633             }
7634             return this;
7635         },
7636
7637         /**
7638          * Adds one or more CSS classes to this element and removes the same class(es) from all siblings.
7639          * @param {String/Array} className The CSS class to add, or an array of classes
7640          * @return {Roo.Element} this
7641          */
7642         radioClass : function(className){
7643             var siblings = this.dom.parentNode.childNodes;
7644             for(var i = 0; i < siblings.length; i++) {
7645                 var s = siblings[i];
7646                 if(s.nodeType == 1){
7647                     Roo.get(s).removeClass(className);
7648                 }
7649             }
7650             this.addClass(className);
7651             return this;
7652         },
7653
7654         /**
7655          * Removes one or more CSS classes from the element.
7656          * @param {String/Array} className The CSS class to remove, or an array of classes
7657          * @return {Roo.Element} this
7658          */
7659         removeClass : function(className){
7660             if(!className || !this.dom.className){
7661                 return this;
7662             }
7663             if(className instanceof Array){
7664                 for(var i = 0, len = className.length; i < len; i++) {
7665                     this.removeClass(className[i]);
7666                 }
7667             }else{
7668                 if(this.hasClass(className)){
7669                     var re = this.classReCache[className];
7670                     if (!re) {
7671                        re = new RegExp('(?:^|\\s+)' + className + '(?:\\s+|$)', "g");
7672                        this.classReCache[className] = re;
7673                     }
7674                     this.dom.className =
7675                         this.dom.className.replace(re, " ");
7676                 }
7677             }
7678             return this;
7679         },
7680
7681         // private
7682         classReCache: {},
7683
7684         /**
7685          * Toggles the specified CSS class on this element (removes it if it already exists, otherwise adds it).
7686          * @param {String} className The CSS class to toggle
7687          * @return {Roo.Element} this
7688          */
7689         toggleClass : function(className){
7690             if(this.hasClass(className)){
7691                 this.removeClass(className);
7692             }else{
7693                 this.addClass(className);
7694             }
7695             return this;
7696         },
7697
7698         /**
7699          * Checks if the specified CSS class exists on this element's DOM node.
7700          * @param {String} className The CSS class to check for
7701          * @return {Boolean} True if the class exists, else false
7702          */
7703         hasClass : function(className){
7704             return className && (' '+this.dom.className+' ').indexOf(' '+className+' ') != -1;
7705         },
7706
7707         /**
7708          * Replaces a CSS class on the element with another.  If the old name does not exist, the new name will simply be added.
7709          * @param {String} oldClassName The CSS class to replace
7710          * @param {String} newClassName The replacement CSS class
7711          * @return {Roo.Element} this
7712          */
7713         replaceClass : function(oldClassName, newClassName){
7714             this.removeClass(oldClassName);
7715             this.addClass(newClassName);
7716             return this;
7717         },
7718
7719         /**
7720          * Returns an object with properties matching the styles requested.
7721          * For example, el.getStyles('color', 'font-size', 'width') might return
7722          * {'color': '#FFFFFF', 'font-size': '13px', 'width': '100px'}.
7723          * @param {String} style1 A style name
7724          * @param {String} style2 A style name
7725          * @param {String} etc.
7726          * @return {Object} The style object
7727          */
7728         getStyles : function(){
7729             var a = arguments, len = a.length, r = {};
7730             for(var i = 0; i < len; i++){
7731                 r[a[i]] = this.getStyle(a[i]);
7732             }
7733             return r;
7734         },
7735
7736         /**
7737          * Normalizes currentStyle and computedStyle. This is not YUI getStyle, it is an optimised version.
7738          * @param {String} property The style property whose value is returned.
7739          * @return {String} The current value of the style property for this element.
7740          */
7741         getStyle : function(){
7742             return view && view.getComputedStyle ?
7743                 function(prop){
7744                     var el = this.dom, v, cs, camel;
7745                     if(prop == 'float'){
7746                         prop = "cssFloat";
7747                     }
7748                     if(el.style && (v = el.style[prop])){
7749                         return v;
7750                     }
7751                     if(cs = view.getComputedStyle(el, "")){
7752                         if(!(camel = propCache[prop])){
7753                             camel = propCache[prop] = prop.replace(camelRe, camelFn);
7754                         }
7755                         return cs[camel];
7756                     }
7757                     return null;
7758                 } :
7759                 function(prop){
7760                     var el = this.dom, v, cs, camel;
7761                     if(prop == 'opacity'){
7762                         if(typeof el.style.filter == 'string'){
7763                             var m = el.style.filter.match(/alpha\(opacity=(.*)\)/i);
7764                             if(m){
7765                                 var fv = parseFloat(m[1]);
7766                                 if(!isNaN(fv)){
7767                                     return fv ? fv / 100 : 0;
7768                                 }
7769                             }
7770                         }
7771                         return 1;
7772                     }else if(prop == 'float'){
7773                         prop = "styleFloat";
7774                     }
7775                     if(!(camel = propCache[prop])){
7776                         camel = propCache[prop] = prop.replace(camelRe, camelFn);
7777                     }
7778                     if(v = el.style[camel]){
7779                         return v;
7780                     }
7781                     if(cs = el.currentStyle){
7782                         return cs[camel];
7783                     }
7784                     return null;
7785                 };
7786         }(),
7787
7788         /**
7789          * Wrapper for setting style properties, also takes single object parameter of multiple styles.
7790          * @param {String/Object} property The style property to be set, or an object of multiple styles.
7791          * @param {String} value (optional) The value to apply to the given property, or null if an object was passed.
7792          * @return {Roo.Element} this
7793          */
7794         setStyle : function(prop, value){
7795             if(typeof prop == "string"){
7796                 
7797                 if (prop == 'float') {
7798                     this.setStyle(Roo.isIE ? 'styleFloat'  : 'cssFloat', value);
7799                     return this;
7800                 }
7801                 
7802                 var camel;
7803                 if(!(camel = propCache[prop])){
7804                     camel = propCache[prop] = prop.replace(camelRe, camelFn);
7805                 }
7806                 
7807                 if(camel == 'opacity') {
7808                     this.setOpacity(value);
7809                 }else{
7810                     this.dom.style[camel] = value;
7811                 }
7812             }else{
7813                 for(var style in prop){
7814                     if(typeof prop[style] != "function"){
7815                        this.setStyle(style, prop[style]);
7816                     }
7817                 }
7818             }
7819             return this;
7820         },
7821
7822         /**
7823          * More flexible version of {@link #setStyle} for setting style properties.
7824          * @param {String/Object/Function} styles A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
7825          * a function which returns such a specification.
7826          * @return {Roo.Element} this
7827          */
7828         applyStyles : function(style){
7829             Roo.DomHelper.applyStyles(this.dom, style);
7830             return this;
7831         },
7832
7833         /**
7834           * 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).
7835           * @return {Number} The X position of the element
7836           */
7837         getX : function(){
7838             return D.getX(this.dom);
7839         },
7840
7841         /**
7842           * 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).
7843           * @return {Number} The Y position of the element
7844           */
7845         getY : function(){
7846             return D.getY(this.dom);
7847         },
7848
7849         /**
7850           * 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).
7851           * @return {Array} The XY position of the element
7852           */
7853         getXY : function(){
7854             return D.getXY(this.dom);
7855         },
7856
7857         /**
7858          * 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).
7859          * @param {Number} The X position of the element
7860          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7861          * @return {Roo.Element} this
7862          */
7863         setX : function(x, animate){
7864             if(!animate || !A){
7865                 D.setX(this.dom, x);
7866             }else{
7867                 this.setXY([x, this.getY()], this.preanim(arguments, 1));
7868             }
7869             return this;
7870         },
7871
7872         /**
7873          * 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).
7874          * @param {Number} The Y position of the element
7875          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7876          * @return {Roo.Element} this
7877          */
7878         setY : function(y, animate){
7879             if(!animate || !A){
7880                 D.setY(this.dom, y);
7881             }else{
7882                 this.setXY([this.getX(), y], this.preanim(arguments, 1));
7883             }
7884             return this;
7885         },
7886
7887         /**
7888          * Sets the element's left position directly using CSS style (instead of {@link #setX}).
7889          * @param {String} left The left CSS property value
7890          * @return {Roo.Element} this
7891          */
7892         setLeft : function(left){
7893             this.setStyle("left", this.addUnits(left));
7894             return this;
7895         },
7896
7897         /**
7898          * Sets the element's top position directly using CSS style (instead of {@link #setY}).
7899          * @param {String} top The top CSS property value
7900          * @return {Roo.Element} this
7901          */
7902         setTop : function(top){
7903             this.setStyle("top", this.addUnits(top));
7904             return this;
7905         },
7906
7907         /**
7908          * Sets the element's CSS right style.
7909          * @param {String} right The right CSS property value
7910          * @return {Roo.Element} this
7911          */
7912         setRight : function(right){
7913             this.setStyle("right", this.addUnits(right));
7914             return this;
7915         },
7916
7917         /**
7918          * Sets the element's CSS bottom style.
7919          * @param {String} bottom The bottom CSS property value
7920          * @return {Roo.Element} this
7921          */
7922         setBottom : function(bottom){
7923             this.setStyle("bottom", this.addUnits(bottom));
7924             return this;
7925         },
7926
7927         /**
7928          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7929          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7930          * @param {Array} pos Contains X & Y [x, y] values for new position (coordinates are page-based)
7931          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7932          * @return {Roo.Element} this
7933          */
7934         setXY : function(pos, animate){
7935             if(!animate || !A){
7936                 D.setXY(this.dom, pos);
7937             }else{
7938                 this.anim({points: {to: pos}}, this.preanim(arguments, 1), 'motion');
7939             }
7940             return this;
7941         },
7942
7943         /**
7944          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7945          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7946          * @param {Number} x X value for new position (coordinates are page-based)
7947          * @param {Number} y Y value for new position (coordinates are page-based)
7948          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7949          * @return {Roo.Element} this
7950          */
7951         setLocation : function(x, y, animate){
7952             this.setXY([x, y], this.preanim(arguments, 2));
7953             return this;
7954         },
7955
7956         /**
7957          * Sets the position of the element in page coordinates, regardless of how the element is positioned.
7958          * The element must be part of the DOM tree to have page coordinates (display:none or elements not appended return false).
7959          * @param {Number} x X value for new position (coordinates are page-based)
7960          * @param {Number} y Y value for new position (coordinates are page-based)
7961          * @param {Boolean/Object} animate (optional) True for the default animation, or a standard Element animation config object
7962          * @return {Roo.Element} this
7963          */
7964         moveTo : function(x, y, animate){
7965             this.setXY([x, y], this.preanim(arguments, 2));
7966             return this;
7967         },
7968
7969         /**
7970          * Returns the region of the given element.
7971          * The element must be part of the DOM tree to have a region (display:none or elements not appended return false).
7972          * @return {Region} A Roo.lib.Region containing "top, left, bottom, right" member data.
7973          */
7974         getRegion : function(){
7975             return D.getRegion(this.dom);
7976         },
7977
7978         /**
7979          * Returns the offset height of the element
7980          * @param {Boolean} contentHeight (optional) true to get the height minus borders and padding
7981          * @return {Number} The element's height
7982          */
7983         getHeight : function(contentHeight){
7984             var h = this.dom.offsetHeight || 0;
7985             return contentHeight !== true ? h : h-this.getBorderWidth("tb")-this.getPadding("tb");
7986         },
7987
7988         /**
7989          * Returns the offset width of the element
7990          * @param {Boolean} contentWidth (optional) true to get the width minus borders and padding
7991          * @return {Number} The element's width
7992          */
7993         getWidth : function(contentWidth){
7994             var w = this.dom.offsetWidth || 0;
7995             return contentWidth !== true ? w : w-this.getBorderWidth("lr")-this.getPadding("lr");
7996         },
7997
7998         /**
7999          * Returns either the offsetHeight or the height of this element based on CSS height adjusted by padding or borders
8000          * when needed to simulate offsetHeight when offsets aren't available. This may not work on display:none elements
8001          * if a height has not been set using CSS.
8002          * @return {Number}
8003          */
8004         getComputedHeight : function(){
8005             var h = Math.max(this.dom.offsetHeight, this.dom.clientHeight);
8006             if(!h){
8007                 h = parseInt(this.getStyle('height'), 10) || 0;
8008                 if(!this.isBorderBox()){
8009                     h += this.getFrameWidth('tb');
8010                 }
8011             }
8012             return h;
8013         },
8014
8015         /**
8016          * Returns either the offsetWidth or the width of this element based on CSS width adjusted by padding or borders
8017          * when needed to simulate offsetWidth when offsets aren't available. This may not work on display:none elements
8018          * if a width has not been set using CSS.
8019          * @return {Number}
8020          */
8021         getComputedWidth : function(){
8022             var w = Math.max(this.dom.offsetWidth, this.dom.clientWidth);
8023             if(!w){
8024                 w = parseInt(this.getStyle('width'), 10) || 0;
8025                 if(!this.isBorderBox()){
8026                     w += this.getFrameWidth('lr');
8027                 }
8028             }
8029             return w;
8030         },
8031
8032         /**
8033          * Returns the size of the element.
8034          * @param {Boolean} contentSize (optional) true to get the width/size minus borders and padding
8035          * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
8036          */
8037         getSize : function(contentSize){
8038             return {width: this.getWidth(contentSize), height: this.getHeight(contentSize)};
8039         },
8040
8041         /**
8042          * Returns the width and height of the viewport.
8043          * @return {Object} An object containing the viewport's size {width: (viewport width), height: (viewport height)}
8044          */
8045         getViewSize : function(){
8046             var d = this.dom, doc = document, aw = 0, ah = 0;
8047             if(d == doc || d == doc.body){
8048                 return {width : D.getViewWidth(), height: D.getViewHeight()};
8049             }else{
8050                 return {
8051                     width : d.clientWidth,
8052                     height: d.clientHeight
8053                 };
8054             }
8055         },
8056
8057         /**
8058          * Returns the value of the "value" attribute
8059          * @param {Boolean} asNumber true to parse the value as a number
8060          * @return {String/Number}
8061          */
8062         getValue : function(asNumber){
8063             return asNumber ? parseInt(this.dom.value, 10) : this.dom.value;
8064         },
8065
8066         // private
8067         adjustWidth : function(width){
8068             if(typeof width == "number"){
8069                 if(this.autoBoxAdjust && !this.isBorderBox()){
8070                    width -= (this.getBorderWidth("lr") + this.getPadding("lr"));
8071                 }
8072                 if(width < 0){
8073                     width = 0;
8074                 }
8075             }
8076             return width;
8077         },
8078
8079         // private
8080         adjustHeight : function(height){
8081             if(typeof height == "number"){
8082                if(this.autoBoxAdjust && !this.isBorderBox()){
8083                    height -= (this.getBorderWidth("tb") + this.getPadding("tb"));
8084                }
8085                if(height < 0){
8086                    height = 0;
8087                }
8088             }
8089             return height;
8090         },
8091
8092         /**
8093          * Set the width of the element
8094          * @param {Number} width The new width
8095          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8096          * @return {Roo.Element} this
8097          */
8098         setWidth : function(width, animate){
8099             width = this.adjustWidth(width);
8100             if(!animate || !A){
8101                 this.dom.style.width = this.addUnits(width);
8102             }else{
8103                 this.anim({width: {to: width}}, this.preanim(arguments, 1));
8104             }
8105             return this;
8106         },
8107
8108         /**
8109          * Set the height of the element
8110          * @param {Number} height The new height
8111          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8112          * @return {Roo.Element} this
8113          */
8114          setHeight : function(height, animate){
8115             height = this.adjustHeight(height);
8116             if(!animate || !A){
8117                 this.dom.style.height = this.addUnits(height);
8118             }else{
8119                 this.anim({height: {to: height}}, this.preanim(arguments, 1));
8120             }
8121             return this;
8122         },
8123
8124         /**
8125          * Set the size of the element. If animation is true, both width an height will be animated concurrently.
8126          * @param {Number} width The new width
8127          * @param {Number} height The new height
8128          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8129          * @return {Roo.Element} this
8130          */
8131          setSize : function(width, height, animate){
8132             if(typeof width == "object"){ // in case of object from getSize()
8133                 height = width.height; width = width.width;
8134             }
8135             width = this.adjustWidth(width); height = this.adjustHeight(height);
8136             if(!animate || !A){
8137                 this.dom.style.width = this.addUnits(width);
8138                 this.dom.style.height = this.addUnits(height);
8139             }else{
8140                 this.anim({width: {to: width}, height: {to: height}}, this.preanim(arguments, 2));
8141             }
8142             return this;
8143         },
8144
8145         /**
8146          * Sets the element's position and size in one shot. If animation is true then width, height, x and y will be animated concurrently.
8147          * @param {Number} x X value for new position (coordinates are page-based)
8148          * @param {Number} y Y value for new position (coordinates are page-based)
8149          * @param {Number} width The new width
8150          * @param {Number} height The new height
8151          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8152          * @return {Roo.Element} this
8153          */
8154         setBounds : function(x, y, width, height, animate){
8155             if(!animate || !A){
8156                 this.setSize(width, height);
8157                 this.setLocation(x, y);
8158             }else{
8159                 width = this.adjustWidth(width); height = this.adjustHeight(height);
8160                 this.anim({points: {to: [x, y]}, width: {to: width}, height: {to: height}},
8161                               this.preanim(arguments, 4), 'motion');
8162             }
8163             return this;
8164         },
8165
8166         /**
8167          * 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.
8168          * @param {Roo.lib.Region} region The region to fill
8169          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8170          * @return {Roo.Element} this
8171          */
8172         setRegion : function(region, animate){
8173             this.setBounds(region.left, region.top, region.right-region.left, region.bottom-region.top, this.preanim(arguments, 1));
8174             return this;
8175         },
8176
8177         /**
8178          * Appends an event handler
8179          *
8180          * @param {String}   eventName     The type of event to append
8181          * @param {Function} fn        The method the event invokes
8182          * @param {Object} scope       (optional) The scope (this object) of the fn
8183          * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
8184          */
8185         addListener : function(eventName, fn, scope, options){
8186             if (this.dom) {
8187                 Roo.EventManager.on(this.dom,  eventName, fn, scope || this, options);
8188             }
8189         },
8190
8191         /**
8192          * Removes an event handler from this element
8193          * @param {String} eventName the type of event to remove
8194          * @param {Function} fn the method the event invokes
8195          * @return {Roo.Element} this
8196          */
8197         removeListener : function(eventName, fn){
8198             Roo.EventManager.removeListener(this.dom,  eventName, fn);
8199             return this;
8200         },
8201
8202         /**
8203          * Removes all previous added listeners from this element
8204          * @return {Roo.Element} this
8205          */
8206         removeAllListeners : function(){
8207             E.purgeElement(this.dom);
8208             return this;
8209         },
8210
8211         relayEvent : function(eventName, observable){
8212             this.on(eventName, function(e){
8213                 observable.fireEvent(eventName, e);
8214             });
8215         },
8216
8217         /**
8218          * Set the opacity of the element
8219          * @param {Float} opacity The new opacity. 0 = transparent, .5 = 50% visibile, 1 = fully visible, etc
8220          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8221          * @return {Roo.Element} this
8222          */
8223          setOpacity : function(opacity, animate){
8224             if(!animate || !A){
8225                 var s = this.dom.style;
8226                 if(Roo.isIE){
8227                     s.zoom = 1;
8228                     s.filter = (s.filter || '').replace(/alpha\([^\)]*\)/gi,"") +
8229                                (opacity == 1 ? "" : "alpha(opacity=" + opacity * 100 + ")");
8230                 }else{
8231                     s.opacity = opacity;
8232                 }
8233             }else{
8234                 this.anim({opacity: {to: opacity}}, this.preanim(arguments, 1), null, .35, 'easeIn');
8235             }
8236             return this;
8237         },
8238
8239         /**
8240          * Gets the left X coordinate
8241          * @param {Boolean} local True to get the local css position instead of page coordinate
8242          * @return {Number}
8243          */
8244         getLeft : function(local){
8245             if(!local){
8246                 return this.getX();
8247             }else{
8248                 return parseInt(this.getStyle("left"), 10) || 0;
8249             }
8250         },
8251
8252         /**
8253          * Gets the right X coordinate of the element (element X position + element width)
8254          * @param {Boolean} local True to get the local css position instead of page coordinate
8255          * @return {Number}
8256          */
8257         getRight : function(local){
8258             if(!local){
8259                 return this.getX() + this.getWidth();
8260             }else{
8261                 return (this.getLeft(true) + this.getWidth()) || 0;
8262             }
8263         },
8264
8265         /**
8266          * Gets the top Y coordinate
8267          * @param {Boolean} local True to get the local css position instead of page coordinate
8268          * @return {Number}
8269          */
8270         getTop : function(local) {
8271             if(!local){
8272                 return this.getY();
8273             }else{
8274                 return parseInt(this.getStyle("top"), 10) || 0;
8275             }
8276         },
8277
8278         /**
8279          * Gets the bottom Y coordinate of the element (element Y position + element height)
8280          * @param {Boolean} local True to get the local css position instead of page coordinate
8281          * @return {Number}
8282          */
8283         getBottom : function(local){
8284             if(!local){
8285                 return this.getY() + this.getHeight();
8286             }else{
8287                 return (this.getTop(true) + this.getHeight()) || 0;
8288             }
8289         },
8290
8291         /**
8292         * Initializes positioning on this element. If a desired position is not passed, it will make the
8293         * the element positioned relative IF it is not already positioned.
8294         * @param {String} pos (optional) Positioning to use "relative", "absolute" or "fixed"
8295         * @param {Number} zIndex (optional) The zIndex to apply
8296         * @param {Number} x (optional) Set the page X position
8297         * @param {Number} y (optional) Set the page Y position
8298         */
8299         position : function(pos, zIndex, x, y){
8300             if(!pos){
8301                if(this.getStyle('position') == 'static'){
8302                    this.setStyle('position', 'relative');
8303                }
8304             }else{
8305                 this.setStyle("position", pos);
8306             }
8307             if(zIndex){
8308                 this.setStyle("z-index", zIndex);
8309             }
8310             if(x !== undefined && y !== undefined){
8311                 this.setXY([x, y]);
8312             }else if(x !== undefined){
8313                 this.setX(x);
8314             }else if(y !== undefined){
8315                 this.setY(y);
8316             }
8317         },
8318
8319         /**
8320         * Clear positioning back to the default when the document was loaded
8321         * @param {String} value (optional) The value to use for the left,right,top,bottom, defaults to '' (empty string). You could use 'auto'.
8322         * @return {Roo.Element} this
8323          */
8324         clearPositioning : function(value){
8325             value = value ||'';
8326             this.setStyle({
8327                 "left": value,
8328                 "right": value,
8329                 "top": value,
8330                 "bottom": value,
8331                 "z-index": "",
8332                 "position" : "static"
8333             });
8334             return this;
8335         },
8336
8337         /**
8338         * Gets an object with all CSS positioning properties. Useful along with setPostioning to get
8339         * snapshot before performing an update and then restoring the element.
8340         * @return {Object}
8341         */
8342         getPositioning : function(){
8343             var l = this.getStyle("left");
8344             var t = this.getStyle("top");
8345             return {
8346                 "position" : this.getStyle("position"),
8347                 "left" : l,
8348                 "right" : l ? "" : this.getStyle("right"),
8349                 "top" : t,
8350                 "bottom" : t ? "" : this.getStyle("bottom"),
8351                 "z-index" : this.getStyle("z-index")
8352             };
8353         },
8354
8355         /**
8356          * Gets the width of the border(s) for the specified side(s)
8357          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8358          * passing lr would get the border (l)eft width + the border (r)ight width.
8359          * @return {Number} The width of the sides passed added together
8360          */
8361         getBorderWidth : function(side){
8362             return this.addStyles(side, El.borders);
8363         },
8364
8365         /**
8366          * Gets the width of the padding(s) for the specified side(s)
8367          * @param {String} side Can be t, l, r, b or any combination of those to add multiple values. For example,
8368          * passing lr would get the padding (l)eft + the padding (r)ight.
8369          * @return {Number} The padding of the sides passed added together
8370          */
8371         getPadding : function(side){
8372             return this.addStyles(side, El.paddings);
8373         },
8374
8375         /**
8376         * Set positioning with an object returned by getPositioning().
8377         * @param {Object} posCfg
8378         * @return {Roo.Element} this
8379          */
8380         setPositioning : function(pc){
8381             this.applyStyles(pc);
8382             if(pc.right == "auto"){
8383                 this.dom.style.right = "";
8384             }
8385             if(pc.bottom == "auto"){
8386                 this.dom.style.bottom = "";
8387             }
8388             return this;
8389         },
8390
8391         // private
8392         fixDisplay : function(){
8393             if(this.getStyle("display") == "none"){
8394                 this.setStyle("visibility", "hidden");
8395                 this.setStyle("display", this.originalDisplay); // first try reverting to default
8396                 if(this.getStyle("display") == "none"){ // if that fails, default to block
8397                     this.setStyle("display", "block");
8398                 }
8399             }
8400         },
8401
8402         /**
8403          * Quick set left and top adding default units
8404          * @param {String} left The left CSS property value
8405          * @param {String} top The top CSS property value
8406          * @return {Roo.Element} this
8407          */
8408          setLeftTop : function(left, top){
8409             this.dom.style.left = this.addUnits(left);
8410             this.dom.style.top = this.addUnits(top);
8411             return this;
8412         },
8413
8414         /**
8415          * Move this element relative to its current position.
8416          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
8417          * @param {Number} distance How far to move the element in pixels
8418          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8419          * @return {Roo.Element} this
8420          */
8421          move : function(direction, distance, animate){
8422             var xy = this.getXY();
8423             direction = direction.toLowerCase();
8424             switch(direction){
8425                 case "l":
8426                 case "left":
8427                     this.moveTo(xy[0]-distance, xy[1], this.preanim(arguments, 2));
8428                     break;
8429                case "r":
8430                case "right":
8431                     this.moveTo(xy[0]+distance, xy[1], this.preanim(arguments, 2));
8432                     break;
8433                case "t":
8434                case "top":
8435                case "up":
8436                     this.moveTo(xy[0], xy[1]-distance, this.preanim(arguments, 2));
8437                     break;
8438                case "b":
8439                case "bottom":
8440                case "down":
8441                     this.moveTo(xy[0], xy[1]+distance, this.preanim(arguments, 2));
8442                     break;
8443             }
8444             return this;
8445         },
8446
8447         /**
8448          *  Store the current overflow setting and clip overflow on the element - use {@link #unclip} to remove
8449          * @return {Roo.Element} this
8450          */
8451         clip : function(){
8452             if(!this.isClipped){
8453                this.isClipped = true;
8454                this.originalClip = {
8455                    "o": this.getStyle("overflow"),
8456                    "x": this.getStyle("overflow-x"),
8457                    "y": this.getStyle("overflow-y")
8458                };
8459                this.setStyle("overflow", "hidden");
8460                this.setStyle("overflow-x", "hidden");
8461                this.setStyle("overflow-y", "hidden");
8462             }
8463             return this;
8464         },
8465
8466         /**
8467          *  Return clipping (overflow) to original clipping before clip() was called
8468          * @return {Roo.Element} this
8469          */
8470         unclip : function(){
8471             if(this.isClipped){
8472                 this.isClipped = false;
8473                 var o = this.originalClip;
8474                 if(o.o){this.setStyle("overflow", o.o);}
8475                 if(o.x){this.setStyle("overflow-x", o.x);}
8476                 if(o.y){this.setStyle("overflow-y", o.y);}
8477             }
8478             return this;
8479         },
8480
8481
8482         /**
8483          * Gets the x,y coordinates specified by the anchor position on the element.
8484          * @param {String} anchor (optional) The specified anchor position (defaults to "c").  See {@link #alignTo} for details on supported anchor positions.
8485          * @param {Object} size (optional) An object containing the size to use for calculating anchor position
8486          *                       {width: (target width), height: (target height)} (defaults to the element's current size)
8487          * @param {Boolean} local (optional) True to get the local (element top/left-relative) anchor position instead of page coordinates
8488          * @return {Array} [x, y] An array containing the element's x and y coordinates
8489          */
8490         getAnchorXY : function(anchor, local, s){
8491             //Passing a different size is useful for pre-calculating anchors,
8492             //especially for anchored animations that change the el size.
8493
8494             var w, h, vp = false;
8495             if(!s){
8496                 var d = this.dom;
8497                 if(d == document.body || d == document){
8498                     vp = true;
8499                     w = D.getViewWidth(); h = D.getViewHeight();
8500                 }else{
8501                     w = this.getWidth(); h = this.getHeight();
8502                 }
8503             }else{
8504                 w = s.width;  h = s.height;
8505             }
8506             var x = 0, y = 0, r = Math.round;
8507             switch((anchor || "tl").toLowerCase()){
8508                 case "c":
8509                     x = r(w*.5);
8510                     y = r(h*.5);
8511                 break;
8512                 case "t":
8513                     x = r(w*.5);
8514                     y = 0;
8515                 break;
8516                 case "l":
8517                     x = 0;
8518                     y = r(h*.5);
8519                 break;
8520                 case "r":
8521                     x = w;
8522                     y = r(h*.5);
8523                 break;
8524                 case "b":
8525                     x = r(w*.5);
8526                     y = h;
8527                 break;
8528                 case "tl":
8529                     x = 0;
8530                     y = 0;
8531                 break;
8532                 case "bl":
8533                     x = 0;
8534                     y = h;
8535                 break;
8536                 case "br":
8537                     x = w;
8538                     y = h;
8539                 break;
8540                 case "tr":
8541                     x = w;
8542                     y = 0;
8543                 break;
8544             }
8545             if(local === true){
8546                 return [x, y];
8547             }
8548             if(vp){
8549                 var sc = this.getScroll();
8550                 return [x + sc.left, y + sc.top];
8551             }
8552             //Add the element's offset xy
8553             var o = this.getXY();
8554             return [x+o[0], y+o[1]];
8555         },
8556
8557         /**
8558          * Gets the x,y coordinates to align this element with another element. See {@link #alignTo} for more info on the
8559          * supported position values.
8560          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8561          * @param {String} position The position to align to.
8562          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8563          * @return {Array} [x, y]
8564          */
8565         getAlignToXY : function(el, p, o){
8566             el = Roo.get(el);
8567             var d = this.dom;
8568             if(!el.dom){
8569                 throw "Element.alignTo with an element that doesn't exist";
8570             }
8571             var c = false; //constrain to viewport
8572             var p1 = "", p2 = "";
8573             o = o || [0,0];
8574
8575             if(!p){
8576                 p = "tl-bl";
8577             }else if(p == "?"){
8578                 p = "tl-bl?";
8579             }else if(p.indexOf("-") == -1){
8580                 p = "tl-" + p;
8581             }
8582             p = p.toLowerCase();
8583             var m = p.match(/^([a-z]+)-([a-z]+)(\?)?$/);
8584             if(!m){
8585                throw "Element.alignTo with an invalid alignment " + p;
8586             }
8587             p1 = m[1]; p2 = m[2]; c = !!m[3];
8588
8589             //Subtract the aligned el's internal xy from the target's offset xy
8590             //plus custom offset to get the aligned el's new offset xy
8591             var a1 = this.getAnchorXY(p1, true);
8592             var a2 = el.getAnchorXY(p2, false);
8593             var x = a2[0] - a1[0] + o[0];
8594             var y = a2[1] - a1[1] + o[1];
8595             if(c){
8596                 //constrain the aligned el to viewport if necessary
8597                 var w = this.getWidth(), h = this.getHeight(), r = el.getRegion();
8598                 // 5px of margin for ie
8599                 var dw = D.getViewWidth()-5, dh = D.getViewHeight()-5;
8600
8601                 //If we are at a viewport boundary and the aligned el is anchored on a target border that is
8602                 //perpendicular to the vp border, allow the aligned el to slide on that border,
8603                 //otherwise swap the aligned el to the opposite border of the target.
8604                 var p1y = p1.charAt(0), p1x = p1.charAt(p1.length-1);
8605                var p2y = p2.charAt(0), p2x = p2.charAt(p2.length-1);
8606                var swapY = ((p1y=="t" && p2y=="b") || (p1y=="b" && p2y=="t"));
8607                var swapX = ((p1x=="r" && p2x=="l") || (p1x=="l" && p2x=="r"));
8608
8609                var doc = document;
8610                var scrollX = (doc.documentElement.scrollLeft || doc.body.scrollLeft || 0)+5;
8611                var scrollY = (doc.documentElement.scrollTop || doc.body.scrollTop || 0)+5;
8612
8613                if((x+w) > dw + scrollX){
8614                     x = swapX ? r.left-w : dw+scrollX-w;
8615                 }
8616                if(x < scrollX){
8617                    x = swapX ? r.right : scrollX;
8618                }
8619                if((y+h) > dh + scrollY){
8620                     y = swapY ? r.top-h : dh+scrollY-h;
8621                 }
8622                if (y < scrollY){
8623                    y = swapY ? r.bottom : scrollY;
8624                }
8625             }
8626             return [x,y];
8627         },
8628
8629         // private
8630         getConstrainToXY : function(){
8631             var os = {top:0, left:0, bottom:0, right: 0};
8632
8633             return function(el, local, offsets, proposedXY){
8634                 el = Roo.get(el);
8635                 offsets = offsets ? Roo.applyIf(offsets, os) : os;
8636
8637                 var vw, vh, vx = 0, vy = 0;
8638                 if(el.dom == document.body || el.dom == document){
8639                     vw = Roo.lib.Dom.getViewWidth();
8640                     vh = Roo.lib.Dom.getViewHeight();
8641                 }else{
8642                     vw = el.dom.clientWidth;
8643                     vh = el.dom.clientHeight;
8644                     if(!local){
8645                         var vxy = el.getXY();
8646                         vx = vxy[0];
8647                         vy = vxy[1];
8648                     }
8649                 }
8650
8651                 var s = el.getScroll();
8652
8653                 vx += offsets.left + s.left;
8654                 vy += offsets.top + s.top;
8655
8656                 vw -= offsets.right;
8657                 vh -= offsets.bottom;
8658
8659                 var vr = vx+vw;
8660                 var vb = vy+vh;
8661
8662                 var xy = proposedXY || (!local ? this.getXY() : [this.getLeft(true), this.getTop(true)]);
8663                 var x = xy[0], y = xy[1];
8664                 var w = this.dom.offsetWidth, h = this.dom.offsetHeight;
8665
8666                 // only move it if it needs it
8667                 var moved = false;
8668
8669                 // first validate right/bottom
8670                 if((x + w) > vr){
8671                     x = vr - w;
8672                     moved = true;
8673                 }
8674                 if((y + h) > vb){
8675                     y = vb - h;
8676                     moved = true;
8677                 }
8678                 // then make sure top/left isn't negative
8679                 if(x < vx){
8680                     x = vx;
8681                     moved = true;
8682                 }
8683                 if(y < vy){
8684                     y = vy;
8685                     moved = true;
8686                 }
8687                 return moved ? [x, y] : false;
8688             };
8689         }(),
8690
8691         // private
8692         adjustForConstraints : function(xy, parent, offsets){
8693             return this.getConstrainToXY(parent || document, false, offsets, xy) ||  xy;
8694         },
8695
8696         /**
8697          * Aligns this element with another element relative to the specified anchor points. If the other element is the
8698          * document it aligns it to the viewport.
8699          * The position parameter is optional, and can be specified in any one of the following formats:
8700          * <ul>
8701          *   <li><b>Blank</b>: Defaults to aligning the element's top-left corner to the target's bottom-left corner ("tl-bl").</li>
8702          *   <li><b>One anchor (deprecated)</b>: The passed anchor position is used as the target element's anchor point.
8703          *       The element being aligned will position its top-left corner (tl) to that point.  <i>This method has been
8704          *       deprecated in favor of the newer two anchor syntax below</i>.</li>
8705          *   <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
8706          *       element's anchor point, and the second value is used as the target's anchor point.</li>
8707          * </ul>
8708          * In addition to the anchor points, the position parameter also supports the "?" character.  If "?" is passed at the end of
8709          * the position string, the element will attempt to align as specified, but the position will be adjusted to constrain to
8710          * the viewport if necessary.  Note that the element being aligned might be swapped to align to a different position than
8711          * that specified in order to enforce the viewport constraints.
8712          * Following are all of the supported anchor positions:
8713     <pre>
8714     Value  Description
8715     -----  -----------------------------
8716     tl     The top left corner (default)
8717     t      The center of the top edge
8718     tr     The top right corner
8719     l      The center of the left edge
8720     c      In the center of the element
8721     r      The center of the right edge
8722     bl     The bottom left corner
8723     b      The center of the bottom edge
8724     br     The bottom right corner
8725     </pre>
8726     Example Usage:
8727     <pre><code>
8728     // align el to other-el using the default positioning ("tl-bl", non-constrained)
8729     el.alignTo("other-el");
8730
8731     // align the top left corner of el with the top right corner of other-el (constrained to viewport)
8732     el.alignTo("other-el", "tr?");
8733
8734     // align the bottom right corner of el with the center left edge of other-el
8735     el.alignTo("other-el", "br-l?");
8736
8737     // align the center of el with the bottom left corner of other-el and
8738     // adjust the x position by -6 pixels (and the y position by 0)
8739     el.alignTo("other-el", "c-bl", [-6, 0]);
8740     </code></pre>
8741          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8742          * @param {String} position The position to align to.
8743          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8744          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8745          * @return {Roo.Element} this
8746          */
8747         alignTo : function(element, position, offsets, animate){
8748             var xy = this.getAlignToXY(element, position, offsets);
8749             this.setXY(xy, this.preanim(arguments, 3));
8750             return this;
8751         },
8752
8753         /**
8754          * Anchors an element to another element and realigns it when the window is resized.
8755          * @param {String/HTMLElement/Roo.Element} element The element to align to.
8756          * @param {String} position The position to align to.
8757          * @param {Array} offsets (optional) Offset the positioning by [x, y]
8758          * @param {Boolean/Object} animate (optional) True for the default animation or a standard Element animation config object
8759          * @param {Boolean/Number} monitorScroll (optional) True to monitor body scroll and reposition. If this parameter
8760          * is a number, it is used as the buffer delay (defaults to 50ms).
8761          * @param {Function} callback The function to call after the animation finishes
8762          * @return {Roo.Element} this
8763          */
8764         anchorTo : function(el, alignment, offsets, animate, monitorScroll, callback){
8765             var action = function(){
8766                 this.alignTo(el, alignment, offsets, animate);
8767                 Roo.callback(callback, this);
8768             };
8769             Roo.EventManager.onWindowResize(action, this);
8770             var tm = typeof monitorScroll;
8771             if(tm != 'undefined'){
8772                 Roo.EventManager.on(window, 'scroll', action, this,
8773                     {buffer: tm == 'number' ? monitorScroll : 50});
8774             }
8775             action.call(this); // align immediately
8776             return this;
8777         },
8778         /**
8779          * Clears any opacity settings from this element. Required in some cases for IE.
8780          * @return {Roo.Element} this
8781          */
8782         clearOpacity : function(){
8783             if (window.ActiveXObject) {
8784                 if(typeof this.dom.style.filter == 'string' && (/alpha/i).test(this.dom.style.filter)){
8785                     this.dom.style.filter = "";
8786                 }
8787             } else {
8788                 this.dom.style.opacity = "";
8789                 this.dom.style["-moz-opacity"] = "";
8790                 this.dom.style["-khtml-opacity"] = "";
8791             }
8792             return this;
8793         },
8794
8795         /**
8796          * Hide this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8797          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8798          * @return {Roo.Element} this
8799          */
8800         hide : function(animate){
8801             this.setVisible(false, this.preanim(arguments, 0));
8802             return this;
8803         },
8804
8805         /**
8806         * Show this element - Uses display mode to determine whether to use "display" or "visibility". See {@link #setVisible}.
8807         * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
8808          * @return {Roo.Element} this
8809          */
8810         show : function(animate){
8811             this.setVisible(true, this.preanim(arguments, 0));
8812             return this;
8813         },
8814
8815         /**
8816          * @private Test if size has a unit, otherwise appends the default
8817          */
8818         addUnits : function(size){
8819             return Roo.Element.addUnits(size, this.defaultUnit);
8820         },
8821
8822         /**
8823          * Temporarily enables offsets (width,height,x,y) for an element with display:none, use endMeasure() when done.
8824          * @return {Roo.Element} this
8825          */
8826         beginMeasure : function(){
8827             var el = this.dom;
8828             if(el.offsetWidth || el.offsetHeight){
8829                 return this; // offsets work already
8830             }
8831             var changed = [];
8832             var p = this.dom, b = document.body; // start with this element
8833             while((!el.offsetWidth && !el.offsetHeight) && p && p.tagName && p != b){
8834                 var pe = Roo.get(p);
8835                 if(pe.getStyle('display') == 'none'){
8836                     changed.push({el: p, visibility: pe.getStyle("visibility")});
8837                     p.style.visibility = "hidden";
8838                     p.style.display = "block";
8839                 }
8840                 p = p.parentNode;
8841             }
8842             this._measureChanged = changed;
8843             return this;
8844
8845         },
8846
8847         /**
8848          * Restores displays to before beginMeasure was called
8849          * @return {Roo.Element} this
8850          */
8851         endMeasure : function(){
8852             var changed = this._measureChanged;
8853             if(changed){
8854                 for(var i = 0, len = changed.length; i < len; i++) {
8855                     var r = changed[i];
8856                     r.el.style.visibility = r.visibility;
8857                     r.el.style.display = "none";
8858                 }
8859                 this._measureChanged = null;
8860             }
8861             return this;
8862         },
8863
8864         /**
8865         * Update the innerHTML of this element, optionally searching for and processing scripts
8866         * @param {String} html The new HTML
8867         * @param {Boolean} loadScripts (optional) true to look for and process scripts
8868         * @param {Function} callback For async script loading you can be noticed when the update completes
8869         * @return {Roo.Element} this
8870          */
8871         update : function(html, loadScripts, callback){
8872             if(typeof html == "undefined"){
8873                 html = "";
8874             }
8875             if(loadScripts !== true){
8876                 this.dom.innerHTML = html;
8877                 if(typeof callback == "function"){
8878                     callback();
8879                 }
8880                 return this;
8881             }
8882             var id = Roo.id();
8883             var dom = this.dom;
8884
8885             html += '<span id="' + id + '"></span>';
8886
8887             E.onAvailable(id, function(){
8888                 var hd = document.getElementsByTagName("head")[0];
8889                 var re = /(?:<script([^>]*)?>)((\n|\r|.)*?)(?:<\/script>)/ig;
8890                 var srcRe = /\ssrc=([\'\"])(.*?)\1/i;
8891                 var typeRe = /\stype=([\'\"])(.*?)\1/i;
8892
8893                 var match;
8894                 while(match = re.exec(html)){
8895                     var attrs = match[1];
8896                     var srcMatch = attrs ? attrs.match(srcRe) : false;
8897                     if(srcMatch && srcMatch[2]){
8898                        var s = document.createElement("script");
8899                        s.src = srcMatch[2];
8900                        var typeMatch = attrs.match(typeRe);
8901                        if(typeMatch && typeMatch[2]){
8902                            s.type = typeMatch[2];
8903                        }
8904                        hd.appendChild(s);
8905                     }else if(match[2] && match[2].length > 0){
8906                         if(window.execScript) {
8907                            window.execScript(match[2]);
8908                         } else {
8909                             /**
8910                              * eval:var:id
8911                              * eval:var:dom
8912                              * eval:var:html
8913                              * 
8914                              */
8915                            window.eval(match[2]);
8916                         }
8917                     }
8918                 }
8919                 var el = document.getElementById(id);
8920                 if(el){el.parentNode.removeChild(el);}
8921                 if(typeof callback == "function"){
8922                     callback();
8923                 }
8924             });
8925             dom.innerHTML = html.replace(/(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)/ig, "");
8926             return this;
8927         },
8928
8929         /**
8930          * Direct access to the UpdateManager update() method (takes the same parameters).
8931          * @param {String/Function} url The url for this request or a function to call to get the url
8932          * @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}
8933          * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
8934          * @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.
8935          * @return {Roo.Element} this
8936          */
8937         load : function(){
8938             var um = this.getUpdateManager();
8939             um.update.apply(um, arguments);
8940             return this;
8941         },
8942
8943         /**
8944         * Gets this element's UpdateManager
8945         * @return {Roo.UpdateManager} The UpdateManager
8946         */
8947         getUpdateManager : function(){
8948             if(!this.updateManager){
8949                 this.updateManager = new Roo.UpdateManager(this);
8950             }
8951             return this.updateManager;
8952         },
8953
8954         /**
8955          * Disables text selection for this element (normalized across browsers)
8956          * @return {Roo.Element} this
8957          */
8958         unselectable : function(){
8959             this.dom.unselectable = "on";
8960             this.swallowEvent("selectstart", true);
8961             this.applyStyles("-moz-user-select:none;-khtml-user-select:none;");
8962             this.addClass("x-unselectable");
8963             return this;
8964         },
8965
8966         /**
8967         * Calculates the x, y to center this element on the screen
8968         * @return {Array} The x, y values [x, y]
8969         */
8970         getCenterXY : function(){
8971             return this.getAlignToXY(document, 'c-c');
8972         },
8973
8974         /**
8975         * Centers the Element in either the viewport, or another Element.
8976         * @param {String/HTMLElement/Roo.Element} centerIn (optional) The element in which to center the element.
8977         */
8978         center : function(centerIn){
8979             this.alignTo(centerIn || document, 'c-c');
8980             return this;
8981         },
8982
8983         /**
8984          * Tests various css rules/browsers to determine if this element uses a border box
8985          * @return {Boolean}
8986          */
8987         isBorderBox : function(){
8988             return noBoxAdjust[this.dom.tagName.toLowerCase()] || Roo.isBorderBox;
8989         },
8990
8991         /**
8992          * Return a box {x, y, width, height} that can be used to set another elements
8993          * size/location to match this element.
8994          * @param {Boolean} contentBox (optional) If true a box for the content of the element is returned.
8995          * @param {Boolean} local (optional) If true the element's left and top are returned instead of page x/y.
8996          * @return {Object} box An object in the format {x, y, width, height}
8997          */
8998         getBox : function(contentBox, local){
8999             var xy;
9000             if(!local){
9001                 xy = this.getXY();
9002             }else{
9003                 var left = parseInt(this.getStyle("left"), 10) || 0;
9004                 var top = parseInt(this.getStyle("top"), 10) || 0;
9005                 xy = [left, top];
9006             }
9007             var el = this.dom, w = el.offsetWidth, h = el.offsetHeight, bx;
9008             if(!contentBox){
9009                 bx = {x: xy[0], y: xy[1], 0: xy[0], 1: xy[1], width: w, height: h};
9010             }else{
9011                 var l = this.getBorderWidth("l")+this.getPadding("l");
9012                 var r = this.getBorderWidth("r")+this.getPadding("r");
9013                 var t = this.getBorderWidth("t")+this.getPadding("t");
9014                 var b = this.getBorderWidth("b")+this.getPadding("b");
9015                 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)};
9016             }
9017             bx.right = bx.x + bx.width;
9018             bx.bottom = bx.y + bx.height;
9019             return bx;
9020         },
9021
9022         /**
9023          * Returns the sum width of the padding and borders for the passed "sides". See getBorderWidth()
9024          for more information about the sides.
9025          * @param {String} sides
9026          * @return {Number}
9027          */
9028         getFrameWidth : function(sides, onlyContentBox){
9029             return onlyContentBox && Roo.isBorderBox ? 0 : (this.getPadding(sides) + this.getBorderWidth(sides));
9030         },
9031
9032         /**
9033          * 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.
9034          * @param {Object} box The box to fill {x, y, width, height}
9035          * @param {Boolean} adjust (optional) Whether to adjust for box-model issues automatically
9036          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9037          * @return {Roo.Element} this
9038          */
9039         setBox : function(box, adjust, animate){
9040             var w = box.width, h = box.height;
9041             if((adjust && !this.autoBoxAdjust) && !this.isBorderBox()){
9042                w -= (this.getBorderWidth("lr") + this.getPadding("lr"));
9043                h -= (this.getBorderWidth("tb") + this.getPadding("tb"));
9044             }
9045             this.setBounds(box.x, box.y, w, h, this.preanim(arguments, 2));
9046             return this;
9047         },
9048
9049         /**
9050          * Forces the browser to repaint this element
9051          * @return {Roo.Element} this
9052          */
9053          repaint : function(){
9054             var dom = this.dom;
9055             this.addClass("x-repaint");
9056             setTimeout(function(){
9057                 Roo.get(dom).removeClass("x-repaint");
9058             }, 1);
9059             return this;
9060         },
9061
9062         /**
9063          * Returns an object with properties top, left, right and bottom representing the margins of this element unless sides is passed,
9064          * then it returns the calculated width of the sides (see getPadding)
9065          * @param {String} sides (optional) Any combination of l, r, t, b to get the sum of those sides
9066          * @return {Object/Number}
9067          */
9068         getMargins : function(side){
9069             if(!side){
9070                 return {
9071                     top: parseInt(this.getStyle("margin-top"), 10) || 0,
9072                     left: parseInt(this.getStyle("margin-left"), 10) || 0,
9073                     bottom: parseInt(this.getStyle("margin-bottom"), 10) || 0,
9074                     right: parseInt(this.getStyle("margin-right"), 10) || 0
9075                 };
9076             }else{
9077                 return this.addStyles(side, El.margins);
9078              }
9079         },
9080
9081         // private
9082         addStyles : function(sides, styles){
9083             var val = 0, v, w;
9084             for(var i = 0, len = sides.length; i < len; i++){
9085                 v = this.getStyle(styles[sides.charAt(i)]);
9086                 if(v){
9087                      w = parseInt(v, 10);
9088                      if(w){ val += w; }
9089                 }
9090             }
9091             return val;
9092         },
9093
9094         /**
9095          * Creates a proxy element of this element
9096          * @param {String/Object} config The class name of the proxy element or a DomHelper config object
9097          * @param {String/HTMLElement} renderTo (optional) The element or element id to render the proxy to (defaults to document.body)
9098          * @param {Boolean} matchBox (optional) True to align and size the proxy to this element now (defaults to false)
9099          * @return {Roo.Element} The new proxy element
9100          */
9101         createProxy : function(config, renderTo, matchBox){
9102             if(renderTo){
9103                 renderTo = Roo.getDom(renderTo);
9104             }else{
9105                 renderTo = document.body;
9106             }
9107             config = typeof config == "object" ?
9108                 config : {tag : "div", cls: config};
9109             var proxy = Roo.DomHelper.append(renderTo, config, true);
9110             if(matchBox){
9111                proxy.setBox(this.getBox());
9112             }
9113             return proxy;
9114         },
9115
9116         /**
9117          * Puts a mask over this element to disable user interaction. Requires core.css.
9118          * This method can only be applied to elements which accept child nodes.
9119          * @param {String} msg (optional) A message to display in the mask
9120          * @param {String} msgCls (optional) A css class to apply to the msg element
9121          * @return {Element} The mask  element
9122          */
9123         mask : function(msg, msgCls)
9124         {
9125             if(this.getStyle("position") == "static" && this.dom.tagName !== 'BODY'){
9126                 this.setStyle("position", "relative");
9127             }
9128             if(!this._mask){
9129                 this._mask = Roo.DomHelper.append(this.dom, {cls:"roo-el-mask"}, true);
9130             }
9131             
9132             this.addClass("x-masked");
9133             this._mask.setDisplayed(true);
9134             
9135             // we wander
9136             var z = 0;
9137             var dom = this.dom;
9138             while (dom && dom.style) {
9139                 if (!isNaN(parseInt(dom.style.zIndex))) {
9140                     z = Math.max(z, parseInt(dom.style.zIndex));
9141                 }
9142                 dom = dom.parentNode;
9143             }
9144             // if we are masking the body - then it hides everything..
9145             if (this.dom == document.body) {
9146                 z = 1000000;
9147                 this._mask.setWidth(Roo.lib.Dom.getDocumentWidth());
9148                 this._mask.setHeight(Roo.lib.Dom.getDocumentHeight());
9149             }
9150            
9151             if(typeof msg == 'string'){
9152                 if(!this._maskMsg){
9153                     this._maskMsg = Roo.DomHelper.append(this.dom, {
9154                         cls: "roo-el-mask-msg", 
9155                         cn: [
9156                             {
9157                                 tag: 'i',
9158                                 cls: 'fa fa-spinner fa-spin'
9159                             },
9160                             {
9161                                 tag: 'div'
9162                             }   
9163                         ]
9164                     }, true);
9165                 }
9166                 var mm = this._maskMsg;
9167                 mm.dom.className = msgCls ? "roo-el-mask-msg " + msgCls : "roo-el-mask-msg";
9168                 if (mm.dom.lastChild) { // weird IE issue?
9169                     mm.dom.lastChild.innerHTML = msg;
9170                 }
9171                 mm.setDisplayed(true);
9172                 mm.center(this);
9173                 mm.setStyle('z-index', z + 102);
9174             }
9175             if(Roo.isIE && !(Roo.isIE7 && Roo.isStrict) && this.getStyle('height') == 'auto'){ // ie will not expand full height automatically
9176                 this._mask.setHeight(this.getHeight());
9177             }
9178             this._mask.setStyle('z-index', z + 100);
9179             
9180             return this._mask;
9181         },
9182
9183         /**
9184          * Removes a previously applied mask. If removeEl is true the mask overlay is destroyed, otherwise
9185          * it is cached for reuse.
9186          */
9187         unmask : function(removeEl){
9188             if(this._mask){
9189                 if(removeEl === true){
9190                     this._mask.remove();
9191                     delete this._mask;
9192                     if(this._maskMsg){
9193                         this._maskMsg.remove();
9194                         delete this._maskMsg;
9195                     }
9196                 }else{
9197                     this._mask.setDisplayed(false);
9198                     if(this._maskMsg){
9199                         this._maskMsg.setDisplayed(false);
9200                     }
9201                 }
9202             }
9203             this.removeClass("x-masked");
9204         },
9205
9206         /**
9207          * Returns true if this element is masked
9208          * @return {Boolean}
9209          */
9210         isMasked : function(){
9211             return this._mask && this._mask.isVisible();
9212         },
9213
9214         /**
9215          * Creates an iframe shim for this element to keep selects and other windowed objects from
9216          * showing through.
9217          * @return {Roo.Element} The new shim element
9218          */
9219         createShim : function(){
9220             var el = document.createElement('iframe');
9221             el.frameBorder = 'no';
9222             el.className = 'roo-shim';
9223             if(Roo.isIE && Roo.isSecure){
9224                 el.src = Roo.SSL_SECURE_URL;
9225             }
9226             var shim = Roo.get(this.dom.parentNode.insertBefore(el, this.dom));
9227             shim.autoBoxAdjust = false;
9228             return shim;
9229         },
9230
9231         /**
9232          * Removes this element from the DOM and deletes it from the cache
9233          */
9234         remove : function(){
9235             if(this.dom.parentNode){
9236                 this.dom.parentNode.removeChild(this.dom);
9237             }
9238             delete El.cache[this.dom.id];
9239         },
9240
9241         /**
9242          * Sets up event handlers to add and remove a css class when the mouse is over this element
9243          * @param {String} className
9244          * @param {Boolean} preventFlicker (optional) If set to true, it prevents flickering by filtering
9245          * mouseout events for children elements
9246          * @return {Roo.Element} this
9247          */
9248         addClassOnOver : function(className, preventFlicker){
9249             this.on("mouseover", function(){
9250                 Roo.fly(this, '_internal').addClass(className);
9251             }, this.dom);
9252             var removeFn = function(e){
9253                 if(preventFlicker !== true || !e.within(this, true)){
9254                     Roo.fly(this, '_internal').removeClass(className);
9255                 }
9256             };
9257             this.on("mouseout", removeFn, this.dom);
9258             return this;
9259         },
9260
9261         /**
9262          * Sets up event handlers to add and remove a css class when this element has the focus
9263          * @param {String} className
9264          * @return {Roo.Element} this
9265          */
9266         addClassOnFocus : function(className){
9267             this.on("focus", function(){
9268                 Roo.fly(this, '_internal').addClass(className);
9269             }, this.dom);
9270             this.on("blur", function(){
9271                 Roo.fly(this, '_internal').removeClass(className);
9272             }, this.dom);
9273             return this;
9274         },
9275         /**
9276          * 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)
9277          * @param {String} className
9278          * @return {Roo.Element} this
9279          */
9280         addClassOnClick : function(className){
9281             var dom = this.dom;
9282             this.on("mousedown", function(){
9283                 Roo.fly(dom, '_internal').addClass(className);
9284                 var d = Roo.get(document);
9285                 var fn = function(){
9286                     Roo.fly(dom, '_internal').removeClass(className);
9287                     d.removeListener("mouseup", fn);
9288                 };
9289                 d.on("mouseup", fn);
9290             });
9291             return this;
9292         },
9293
9294         /**
9295          * Stops the specified event from bubbling and optionally prevents the default action
9296          * @param {String} eventName
9297          * @param {Boolean} preventDefault (optional) true to prevent the default action too
9298          * @return {Roo.Element} this
9299          */
9300         swallowEvent : function(eventName, preventDefault){
9301             var fn = function(e){
9302                 e.stopPropagation();
9303                 if(preventDefault){
9304                     e.preventDefault();
9305                 }
9306             };
9307             if(eventName instanceof Array){
9308                 for(var i = 0, len = eventName.length; i < len; i++){
9309                      this.on(eventName[i], fn);
9310                 }
9311                 return this;
9312             }
9313             this.on(eventName, fn);
9314             return this;
9315         },
9316
9317         /**
9318          * @private
9319          */
9320       fitToParentDelegate : Roo.emptyFn, // keep a reference to the fitToParent delegate
9321
9322         /**
9323          * Sizes this element to its parent element's dimensions performing
9324          * neccessary box adjustments.
9325          * @param {Boolean} monitorResize (optional) If true maintains the fit when the browser window is resized.
9326          * @param {String/HTMLElment/Element} targetParent (optional) The target parent, default to the parentNode.
9327          * @return {Roo.Element} this
9328          */
9329         fitToParent : function(monitorResize, targetParent) {
9330           Roo.EventManager.removeResizeListener(this.fitToParentDelegate); // always remove previous fitToParent delegate from onWindowResize
9331           this.fitToParentDelegate = Roo.emptyFn; // remove reference to previous delegate
9332           if (monitorResize === true && !this.dom.parentNode) { // check if this Element still exists
9333             return;
9334           }
9335           var p = Roo.get(targetParent || this.dom.parentNode);
9336           this.setSize(p.getComputedWidth() - p.getFrameWidth('lr'), p.getComputedHeight() - p.getFrameWidth('tb'));
9337           if (monitorResize === true) {
9338             this.fitToParentDelegate = this.fitToParent.createDelegate(this, [true, targetParent]);
9339             Roo.EventManager.onWindowResize(this.fitToParentDelegate);
9340           }
9341           return this;
9342         },
9343
9344         /**
9345          * Gets the next sibling, skipping text nodes
9346          * @return {HTMLElement} The next sibling or null
9347          */
9348         getNextSibling : function(){
9349             var n = this.dom.nextSibling;
9350             while(n && n.nodeType != 1){
9351                 n = n.nextSibling;
9352             }
9353             return n;
9354         },
9355
9356         /**
9357          * Gets the previous sibling, skipping text nodes
9358          * @return {HTMLElement} The previous sibling or null
9359          */
9360         getPrevSibling : function(){
9361             var n = this.dom.previousSibling;
9362             while(n && n.nodeType != 1){
9363                 n = n.previousSibling;
9364             }
9365             return n;
9366         },
9367
9368
9369         /**
9370          * Appends the passed element(s) to this element
9371          * @param {String/HTMLElement/Array/Element/CompositeElement} el
9372          * @return {Roo.Element} this
9373          */
9374         appendChild: function(el){
9375             el = Roo.get(el);
9376             el.appendTo(this);
9377             return this;
9378         },
9379
9380         /**
9381          * Creates the passed DomHelper config and appends it to this element or optionally inserts it before the passed child element.
9382          * @param {Object} config DomHelper element config object.  If no tag is specified (e.g., {tag:'input'}) then a div will be
9383          * automatically generated with the specified attributes.
9384          * @param {HTMLElement} insertBefore (optional) a child element of this element
9385          * @param {Boolean} returnDom (optional) true to return the dom node instead of creating an Element
9386          * @return {Roo.Element} The new child element
9387          */
9388         createChild: function(config, insertBefore, returnDom){
9389             config = config || {tag:'div'};
9390             if(insertBefore){
9391                 return Roo.DomHelper.insertBefore(insertBefore, config, returnDom !== true);
9392             }
9393             return Roo.DomHelper[!this.dom.firstChild ? 'overwrite' : 'append'](this.dom, config,  returnDom !== true);
9394         },
9395
9396         /**
9397          * Appends this element to the passed element
9398          * @param {String/HTMLElement/Element} el The new parent element
9399          * @return {Roo.Element} this
9400          */
9401         appendTo: function(el){
9402             el = Roo.getDom(el);
9403             el.appendChild(this.dom);
9404             return this;
9405         },
9406
9407         /**
9408          * Inserts this element before the passed element in the DOM
9409          * @param {String/HTMLElement/Element} el The element to insert before
9410          * @return {Roo.Element} this
9411          */
9412         insertBefore: function(el){
9413             el = Roo.getDom(el);
9414             el.parentNode.insertBefore(this.dom, el);
9415             return this;
9416         },
9417
9418         /**
9419          * Inserts this element after the passed element in the DOM
9420          * @param {String/HTMLElement/Element} el The element to insert after
9421          * @return {Roo.Element} this
9422          */
9423         insertAfter: function(el){
9424             el = Roo.getDom(el);
9425             el.parentNode.insertBefore(this.dom, el.nextSibling);
9426             return this;
9427         },
9428
9429         /**
9430          * Inserts (or creates) an element (or DomHelper config) as the first child of the this element
9431          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9432          * @return {Roo.Element} The new child
9433          */
9434         insertFirst: function(el, returnDom){
9435             el = el || {};
9436             if(typeof el == 'object' && !el.nodeType){ // dh config
9437                 return this.createChild(el, this.dom.firstChild, returnDom);
9438             }else{
9439                 el = Roo.getDom(el);
9440                 this.dom.insertBefore(el, this.dom.firstChild);
9441                 return !returnDom ? Roo.get(el) : el;
9442             }
9443         },
9444
9445         /**
9446          * Inserts (or creates) the passed element (or DomHelper config) as a sibling of this element
9447          * @param {String/HTMLElement/Element/Object} el The id or element to insert or a DomHelper config to create and insert
9448          * @param {String} where (optional) 'before' or 'after' defaults to before
9449          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9450          * @return {Roo.Element} the inserted Element
9451          */
9452         insertSibling: function(el, where, returnDom){
9453             where = where ? where.toLowerCase() : 'before';
9454             el = el || {};
9455             var rt, refNode = where == 'before' ? this.dom : this.dom.nextSibling;
9456
9457             if(typeof el == 'object' && !el.nodeType){ // dh config
9458                 if(where == 'after' && !this.dom.nextSibling){
9459                     rt = Roo.DomHelper.append(this.dom.parentNode, el, !returnDom);
9460                 }else{
9461                     rt = Roo.DomHelper[where == 'after' ? 'insertAfter' : 'insertBefore'](this.dom, el, !returnDom);
9462                 }
9463
9464             }else{
9465                 rt = this.dom.parentNode.insertBefore(Roo.getDom(el),
9466                             where == 'before' ? this.dom : this.dom.nextSibling);
9467                 if(!returnDom){
9468                     rt = Roo.get(rt);
9469                 }
9470             }
9471             return rt;
9472         },
9473
9474         /**
9475          * Creates and wraps this element with another element
9476          * @param {Object} config (optional) DomHelper element config object for the wrapper element or null for an empty div
9477          * @param {Boolean} returnDom (optional) True to return the raw DOM element instead of Roo.Element
9478          * @return {HTMLElement/Element} The newly created wrapper element
9479          */
9480         wrap: function(config, returnDom){
9481             if(!config){
9482                 config = {tag: "div"};
9483             }
9484             var newEl = Roo.DomHelper.insertBefore(this.dom, config, !returnDom);
9485             newEl.dom ? newEl.dom.appendChild(this.dom) : newEl.appendChild(this.dom);
9486             return newEl;
9487         },
9488
9489         /**
9490          * Replaces the passed element with this element
9491          * @param {String/HTMLElement/Element} el The element to replace
9492          * @return {Roo.Element} this
9493          */
9494         replace: function(el){
9495             el = Roo.get(el);
9496             this.insertBefore(el);
9497             el.remove();
9498             return this;
9499         },
9500
9501         /**
9502          * Inserts an html fragment into this element
9503          * @param {String} where Where to insert the html in relation to the this element - beforeBegin, afterBegin, beforeEnd, afterEnd.
9504          * @param {String} html The HTML fragment
9505          * @param {Boolean} returnEl True to return an Roo.Element
9506          * @return {HTMLElement/Roo.Element} The inserted node (or nearest related if more than 1 inserted)
9507          */
9508         insertHtml : function(where, html, returnEl){
9509             var el = Roo.DomHelper.insertHtml(where, this.dom, html);
9510             return returnEl ? Roo.get(el) : el;
9511         },
9512
9513         /**
9514          * Sets the passed attributes as attributes of this element (a style attribute can be a string, object or function)
9515          * @param {Object} o The object with the attributes
9516          * @param {Boolean} useSet (optional) false to override the default setAttribute to use expandos.
9517          * @return {Roo.Element} this
9518          */
9519         set : function(o, useSet){
9520             var el = this.dom;
9521             useSet = typeof useSet == 'undefined' ? (el.setAttribute ? true : false) : useSet;
9522             for(var attr in o){
9523                 if(attr == "style" || typeof o[attr] == "function")  { continue; }
9524                 if(attr=="cls"){
9525                     el.className = o["cls"];
9526                 }else{
9527                     if(useSet) {
9528                         el.setAttribute(attr, o[attr]);
9529                     } else {
9530                         el[attr] = o[attr];
9531                     }
9532                 }
9533             }
9534             if(o.style){
9535                 Roo.DomHelper.applyStyles(el, o.style);
9536             }
9537             return this;
9538         },
9539
9540         /**
9541          * Convenience method for constructing a KeyMap
9542          * @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:
9543          *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
9544          * @param {Function} fn The function to call
9545          * @param {Object} scope (optional) The scope of the function
9546          * @return {Roo.KeyMap} The KeyMap created
9547          */
9548         addKeyListener : function(key, fn, scope){
9549             var config;
9550             if(typeof key != "object" || key instanceof Array){
9551                 config = {
9552                     key: key,
9553                     fn: fn,
9554                     scope: scope
9555                 };
9556             }else{
9557                 config = {
9558                     key : key.key,
9559                     shift : key.shift,
9560                     ctrl : key.ctrl,
9561                     alt : key.alt,
9562                     fn: fn,
9563                     scope: scope
9564                 };
9565             }
9566             return new Roo.KeyMap(this, config);
9567         },
9568
9569         /**
9570          * Creates a KeyMap for this element
9571          * @param {Object} config The KeyMap config. See {@link Roo.KeyMap} for more details
9572          * @return {Roo.KeyMap} The KeyMap created
9573          */
9574         addKeyMap : function(config){
9575             return new Roo.KeyMap(this, config);
9576         },
9577
9578         /**
9579          * Returns true if this element is scrollable.
9580          * @return {Boolean}
9581          */
9582          isScrollable : function(){
9583             var dom = this.dom;
9584             return dom.scrollHeight > dom.clientHeight || dom.scrollWidth > dom.clientWidth;
9585         },
9586
9587         /**
9588          * 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().
9589          * @param {String} side Either "left" for scrollLeft values or "top" for scrollTop values.
9590          * @param {Number} value The new scroll value
9591          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9592          * @return {Element} this
9593          */
9594
9595         scrollTo : function(side, value, animate){
9596             var prop = side.toLowerCase() == "left" ? "scrollLeft" : "scrollTop";
9597             if(!animate || !A){
9598                 this.dom[prop] = value;
9599             }else{
9600                 var to = prop == "scrollLeft" ? [value, this.dom.scrollTop] : [this.dom.scrollLeft, value];
9601                 this.anim({scroll: {"to": to}}, this.preanim(arguments, 2), 'scroll');
9602             }
9603             return this;
9604         },
9605
9606         /**
9607          * Scrolls this element the specified direction. Does bounds checking to make sure the scroll is
9608          * within this element's scrollable range.
9609          * @param {String} direction Possible values are: "l","left" - "r","right" - "t","top","up" - "b","bottom","down".
9610          * @param {Number} distance How far to scroll the element in pixels
9611          * @param {Boolean/Object} animate (optional) true for the default animation or a standard Element animation config object
9612          * @return {Boolean} Returns true if a scroll was triggered or false if the element
9613          * was scrolled as far as it could go.
9614          */
9615          scroll : function(direction, distance, animate){
9616              if(!this.isScrollable()){
9617                  return;
9618              }
9619              var el = this.dom;
9620              var l = el.scrollLeft, t = el.scrollTop;
9621              var w = el.scrollWidth, h = el.scrollHeight;
9622              var cw = el.clientWidth, ch = el.clientHeight;
9623              direction = direction.toLowerCase();
9624              var scrolled = false;
9625              var a = this.preanim(arguments, 2);
9626              switch(direction){
9627                  case "l":
9628                  case "left":
9629                      if(w - l > cw){
9630                          var v = Math.min(l + distance, w-cw);
9631                          this.scrollTo("left", v, a);
9632                          scrolled = true;
9633                      }
9634                      break;
9635                 case "r":
9636                 case "right":
9637                      if(l > 0){
9638                          var v = Math.max(l - distance, 0);
9639                          this.scrollTo("left", v, a);
9640                          scrolled = true;
9641                      }
9642                      break;
9643                 case "t":
9644                 case "top":
9645                 case "up":
9646                      if(t > 0){
9647                          var v = Math.max(t - distance, 0);
9648                          this.scrollTo("top", v, a);
9649                          scrolled = true;
9650                      }
9651                      break;
9652                 case "b":
9653                 case "bottom":
9654                 case "down":
9655                      if(h - t > ch){
9656                          var v = Math.min(t + distance, h-ch);
9657                          this.scrollTo("top", v, a);
9658                          scrolled = true;
9659                      }
9660                      break;
9661              }
9662              return scrolled;
9663         },
9664
9665         /**
9666          * Translates the passed page coordinates into left/top css values for this element
9667          * @param {Number/Array} x The page x or an array containing [x, y]
9668          * @param {Number} y The page y
9669          * @return {Object} An object with left and top properties. e.g. {left: (value), top: (value)}
9670          */
9671         translatePoints : function(x, y){
9672             if(typeof x == 'object' || x instanceof Array){
9673                 y = x[1]; x = x[0];
9674             }
9675             var p = this.getStyle('position');
9676             var o = this.getXY();
9677
9678             var l = parseInt(this.getStyle('left'), 10);
9679             var t = parseInt(this.getStyle('top'), 10);
9680
9681             if(isNaN(l)){
9682                 l = (p == "relative") ? 0 : this.dom.offsetLeft;
9683             }
9684             if(isNaN(t)){
9685                 t = (p == "relative") ? 0 : this.dom.offsetTop;
9686             }
9687
9688             return {left: (x - o[0] + l), top: (y - o[1] + t)};
9689         },
9690
9691         /**
9692          * Returns the current scroll position of the element.
9693          * @return {Object} An object containing the scroll position in the format {left: (scrollLeft), top: (scrollTop)}
9694          */
9695         getScroll : function(){
9696             var d = this.dom, doc = document;
9697             if(d == doc || d == doc.body){
9698                 var l = window.pageXOffset || doc.documentElement.scrollLeft || doc.body.scrollLeft || 0;
9699                 var t = window.pageYOffset || doc.documentElement.scrollTop || doc.body.scrollTop || 0;
9700                 return {left: l, top: t};
9701             }else{
9702                 return {left: d.scrollLeft, top: d.scrollTop};
9703             }
9704         },
9705
9706         /**
9707          * Return the CSS color for the specified CSS attribute. rgb, 3 digit (like #fff) and valid values
9708          * are convert to standard 6 digit hex color.
9709          * @param {String} attr The css attribute
9710          * @param {String} defaultValue The default value to use when a valid color isn't found
9711          * @param {String} prefix (optional) defaults to #. Use an empty string when working with
9712          * YUI color anims.
9713          */
9714         getColor : function(attr, defaultValue, prefix){
9715             var v = this.getStyle(attr);
9716             if(!v || v == "transparent" || v == "inherit") {
9717                 return defaultValue;
9718             }
9719             var color = typeof prefix == "undefined" ? "#" : prefix;
9720             if(v.substr(0, 4) == "rgb("){
9721                 var rvs = v.slice(4, v.length -1).split(",");
9722                 for(var i = 0; i < 3; i++){
9723                     var h = parseInt(rvs[i]).toString(16);
9724                     if(h < 16){
9725                         h = "0" + h;
9726                     }
9727                     color += h;
9728                 }
9729             } else {
9730                 if(v.substr(0, 1) == "#"){
9731                     if(v.length == 4) {
9732                         for(var i = 1; i < 4; i++){
9733                             var c = v.charAt(i);
9734                             color +=  c + c;
9735                         }
9736                     }else if(v.length == 7){
9737                         color += v.substr(1);
9738                     }
9739                 }
9740             }
9741             return(color.length > 5 ? color.toLowerCase() : defaultValue);
9742         },
9743
9744         /**
9745          * Wraps the specified element with a special markup/CSS block that renders by default as a gray container with a
9746          * gradient background, rounded corners and a 4-way shadow.
9747          * @param {String} class (optional) A base CSS class to apply to the containing wrapper element (defaults to 'x-box').
9748          * Note that there are a number of CSS rules that are dependent on this name to make the overall effect work,
9749          * so if you supply an alternate base class, make sure you also supply all of the necessary rules.
9750          * @return {Roo.Element} this
9751          */
9752         boxWrap : function(cls){
9753             cls = cls || 'x-box';
9754             var el = Roo.get(this.insertHtml('beforeBegin', String.format('<div class="{0}">'+El.boxMarkup+'</div>', cls)));
9755             el.child('.'+cls+'-mc').dom.appendChild(this.dom);
9756             return el;
9757         },
9758
9759         /**
9760          * Returns the value of a namespaced attribute from the element's underlying DOM node.
9761          * @param {String} namespace The namespace in which to look for the attribute
9762          * @param {String} name The attribute name
9763          * @return {String} The attribute value
9764          */
9765         getAttributeNS : Roo.isIE ? function(ns, name){
9766             var d = this.dom;
9767             var type = typeof d[ns+":"+name];
9768             if(type != 'undefined' && type != 'unknown'){
9769                 return d[ns+":"+name];
9770             }
9771             return d[name];
9772         } : function(ns, name){
9773             var d = this.dom;
9774             return d.getAttributeNS(ns, name) || d.getAttribute(ns+":"+name) || d.getAttribute(name) || d[name];
9775         },
9776         
9777         
9778         /**
9779          * Sets or Returns the value the dom attribute value
9780          * @param {String|Object} name The attribute name (or object to set multiple attributes)
9781          * @param {String} value (optional) The value to set the attribute to
9782          * @return {String} The attribute value
9783          */
9784         attr : function(name){
9785             if (arguments.length > 1) {
9786                 this.dom.setAttribute(name, arguments[1]);
9787                 return arguments[1];
9788             }
9789             if (typeof(name) == 'object') {
9790                 for(var i in name) {
9791                     this.attr(i, name[i]);
9792                 }
9793                 return name;
9794             }
9795             
9796             
9797             if (!this.dom.hasAttribute(name)) {
9798                 return undefined;
9799             }
9800             return this.dom.getAttribute(name);
9801         }
9802         
9803         
9804         
9805     };
9806
9807     var ep = El.prototype;
9808
9809     /**
9810      * Appends an event handler (Shorthand for addListener)
9811      * @param {String}   eventName     The type of event to append
9812      * @param {Function} fn        The method the event invokes
9813      * @param {Object} scope       (optional) The scope (this object) of the fn
9814      * @param {Object}   options   (optional)An object with standard {@link Roo.EventManager#addListener} options
9815      * @method
9816      */
9817     ep.on = ep.addListener;
9818         // backwards compat
9819     ep.mon = ep.addListener;
9820
9821     /**
9822      * Removes an event handler from this element (shorthand for removeListener)
9823      * @param {String} eventName the type of event to remove
9824      * @param {Function} fn the method the event invokes
9825      * @return {Roo.Element} this
9826      * @method
9827      */
9828     ep.un = ep.removeListener;
9829
9830     /**
9831      * true to automatically adjust width and height settings for box-model issues (default to true)
9832      */
9833     ep.autoBoxAdjust = true;
9834
9835     // private
9836     El.unitPattern = /\d+(px|em|%|en|ex|pt|in|cm|mm|pc)$/i;
9837
9838     // private
9839     El.addUnits = function(v, defaultUnit){
9840         if(v === "" || v == "auto"){
9841             return v;
9842         }
9843         if(v === undefined){
9844             return '';
9845         }
9846         if(typeof v == "number" || !El.unitPattern.test(v)){
9847             return v + (defaultUnit || 'px');
9848         }
9849         return v;
9850     };
9851
9852     // special markup used throughout Roo when box wrapping elements
9853     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>';
9854     /**
9855      * Visibility mode constant - Use visibility to hide element
9856      * @static
9857      * @type Number
9858      */
9859     El.VISIBILITY = 1;
9860     /**
9861      * Visibility mode constant - Use display to hide element
9862      * @static
9863      * @type Number
9864      */
9865     El.DISPLAY = 2;
9866
9867     El.borders = {l: "border-left-width", r: "border-right-width", t: "border-top-width", b: "border-bottom-width"};
9868     El.paddings = {l: "padding-left", r: "padding-right", t: "padding-top", b: "padding-bottom"};
9869     El.margins = {l: "margin-left", r: "margin-right", t: "margin-top", b: "margin-bottom"};
9870
9871
9872
9873     /**
9874      * @private
9875      */
9876     El.cache = {};
9877
9878     var docEl;
9879
9880     /**
9881      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
9882      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
9883      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
9884      * @return {Element} The Element object
9885      * @static
9886      */
9887     El.get = function(el){
9888         var ex, elm, id;
9889         if(!el){ return null; }
9890         if(typeof el == "string"){ // element id
9891             if(!(elm = document.getElementById(el))){
9892                 return null;
9893             }
9894             if(ex = El.cache[el]){
9895                 ex.dom = elm;
9896             }else{
9897                 ex = El.cache[el] = new El(elm);
9898             }
9899             return ex;
9900         }else if(el.tagName){ // dom element
9901             if(!(id = el.id)){
9902                 id = Roo.id(el);
9903             }
9904             if(ex = El.cache[id]){
9905                 ex.dom = el;
9906             }else{
9907                 ex = El.cache[id] = new El(el);
9908             }
9909             return ex;
9910         }else if(el instanceof El){
9911             if(el != docEl){
9912                 el.dom = document.getElementById(el.id) || el.dom; // refresh dom element in case no longer valid,
9913                                                               // catch case where it hasn't been appended
9914                 El.cache[el.id] = el; // in case it was created directly with Element(), let's cache it
9915             }
9916             return el;
9917         }else if(el.isComposite){
9918             return el;
9919         }else if(el instanceof Array){
9920             return El.select(el);
9921         }else if(el == document){
9922             // create a bogus element object representing the document object
9923             if(!docEl){
9924                 var f = function(){};
9925                 f.prototype = El.prototype;
9926                 docEl = new f();
9927                 docEl.dom = document;
9928             }
9929             return docEl;
9930         }
9931         return null;
9932     };
9933
9934     // private
9935     El.uncache = function(el){
9936         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
9937             if(a[i]){
9938                 delete El.cache[a[i].id || a[i]];
9939             }
9940         }
9941     };
9942
9943     // private
9944     // Garbage collection - uncache elements/purge listeners on orphaned elements
9945     // so we don't hold a reference and cause the browser to retain them
9946     El.garbageCollect = function(){
9947         if(!Roo.enableGarbageCollector){
9948             clearInterval(El.collectorThread);
9949             return;
9950         }
9951         for(var eid in El.cache){
9952             var el = El.cache[eid], d = el.dom;
9953             // -------------------------------------------------------
9954             // Determining what is garbage:
9955             // -------------------------------------------------------
9956             // !d
9957             // dom node is null, definitely garbage
9958             // -------------------------------------------------------
9959             // !d.parentNode
9960             // no parentNode == direct orphan, definitely garbage
9961             // -------------------------------------------------------
9962             // !d.offsetParent && !document.getElementById(eid)
9963             // display none elements have no offsetParent so we will
9964             // also try to look it up by it's id. However, check
9965             // offsetParent first so we don't do unneeded lookups.
9966             // This enables collection of elements that are not orphans
9967             // directly, but somewhere up the line they have an orphan
9968             // parent.
9969             // -------------------------------------------------------
9970             if(!d || !d.parentNode || (!d.offsetParent && !document.getElementById(eid))){
9971                 delete El.cache[eid];
9972                 if(d && Roo.enableListenerCollection){
9973                     E.purgeElement(d);
9974                 }
9975             }
9976         }
9977     }
9978     El.collectorThreadId = setInterval(El.garbageCollect, 30000);
9979
9980
9981     // dom is optional
9982     El.Flyweight = function(dom){
9983         this.dom = dom;
9984     };
9985     El.Flyweight.prototype = El.prototype;
9986
9987     El._flyweights = {};
9988     /**
9989      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
9990      * the dom node can be overwritten by other code.
9991      * @param {String/HTMLElement} el The dom node or id
9992      * @param {String} named (optional) Allows for creation of named reusable flyweights to
9993      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
9994      * @static
9995      * @return {Element} The shared Element object
9996      */
9997     El.fly = function(el, named){
9998         named = named || '_global';
9999         el = Roo.getDom(el);
10000         if(!el){
10001             return null;
10002         }
10003         if(!El._flyweights[named]){
10004             El._flyweights[named] = new El.Flyweight();
10005         }
10006         El._flyweights[named].dom = el;
10007         return El._flyweights[named];
10008     };
10009
10010     /**
10011      * Static method to retrieve Element objects. Uses simple caching to consistently return the same object.
10012      * Automatically fixes if an object was recreated with the same id via AJAX or DOM.
10013      * Shorthand of {@link Roo.Element#get}
10014      * @param {String/HTMLElement/Element} el The id of the node, a DOM Node or an existing Element.
10015      * @return {Element} The Element object
10016      * @member Roo
10017      * @method get
10018      */
10019     Roo.get = El.get;
10020     /**
10021      * Gets the globally shared flyweight Element, with the passed node as the active element. Do not store a reference to this element -
10022      * the dom node can be overwritten by other code.
10023      * Shorthand of {@link Roo.Element#fly}
10024      * @param {String/HTMLElement} el The dom node or id
10025      * @param {String} named (optional) Allows for creation of named reusable flyweights to
10026      *                                  prevent conflicts (e.g. internally Roo uses "_internal")
10027      * @static
10028      * @return {Element} The shared Element object
10029      * @member Roo
10030      * @method fly
10031      */
10032     Roo.fly = El.fly;
10033
10034     // speedy lookup for elements never to box adjust
10035     var noBoxAdjust = Roo.isStrict ? {
10036         select:1
10037     } : {
10038         input:1, select:1, textarea:1
10039     };
10040     if(Roo.isIE || Roo.isGecko){
10041         noBoxAdjust['button'] = 1;
10042     }
10043
10044
10045     Roo.EventManager.on(window, 'unload', function(){
10046         delete El.cache;
10047         delete El._flyweights;
10048     });
10049 })();
10050
10051
10052
10053
10054 if(Roo.DomQuery){
10055     Roo.Element.selectorFunction = Roo.DomQuery.select;
10056 }
10057
10058 Roo.Element.select = function(selector, unique, root){
10059     var els;
10060     if(typeof selector == "string"){
10061         els = Roo.Element.selectorFunction(selector, root);
10062     }else if(selector.length !== undefined){
10063         els = selector;
10064     }else{
10065         throw "Invalid selector";
10066     }
10067     if(unique === true){
10068         return new Roo.CompositeElement(els);
10069     }else{
10070         return new Roo.CompositeElementLite(els);
10071     }
10072 };
10073 /**
10074  * Selects elements based on the passed CSS selector to enable working on them as 1.
10075  * @param {String/Array} selector The CSS selector or an array of elements
10076  * @param {Boolean} unique (optional) true to create a unique Roo.Element for each element (defaults to a shared flyweight object)
10077  * @param {HTMLElement/String} root (optional) The root element of the query or id of the root
10078  * @return {CompositeElementLite/CompositeElement}
10079  * @member Roo
10080  * @method select
10081  */
10082 Roo.select = Roo.Element.select;
10083
10084
10085
10086
10087
10088
10089
10090
10091
10092
10093
10094
10095
10096
10097 /*
10098  * Based on:
10099  * Ext JS Library 1.1.1
10100  * Copyright(c) 2006-2007, Ext JS, LLC.
10101  *
10102  * Originally Released Under LGPL - original licence link has changed is not relivant.
10103  *
10104  * Fork - LGPL
10105  * <script type="text/javascript">
10106  */
10107
10108
10109
10110 //Notifies Element that fx methods are available
10111 Roo.enableFx = true;
10112
10113 /**
10114  * @class Roo.Fx
10115  * <p>A class to provide basic animation and visual effects support.  <b>Note:</b> This class is automatically applied
10116  * to the {@link Roo.Element} interface when included, so all effects calls should be performed via Element.
10117  * Conversely, since the effects are not actually defined in Element, Roo.Fx <b>must</b> be included in order for the 
10118  * Element effects to work.</p><br/>
10119  *
10120  * <p>It is important to note that although the Fx methods and many non-Fx Element methods support "method chaining" in that
10121  * they return the Element object itself as the method return value, it is not always possible to mix the two in a single
10122  * method chain.  The Fx methods use an internal effects queue so that each effect can be properly timed and sequenced.
10123  * Non-Fx methods, on the other hand, have no such internal queueing and will always execute immediately.  For this reason,
10124  * while it may be possible to mix certain Fx and non-Fx method calls in a single chain, it may not always provide the
10125  * expected results and should be done with care.</p><br/>
10126  *
10127  * <p>Motion effects support 8-way anchoring, meaning that you can choose one of 8 different anchor points on the Element
10128  * that will serve as either the start or end point of the animation.  Following are all of the supported anchor positions:</p>
10129 <pre>
10130 Value  Description
10131 -----  -----------------------------
10132 tl     The top left corner
10133 t      The center of the top edge
10134 tr     The top right corner
10135 l      The center of the left edge
10136 r      The center of the right edge
10137 bl     The bottom left corner
10138 b      The center of the bottom edge
10139 br     The bottom right corner
10140 </pre>
10141  * <b>Although some Fx methods accept specific custom config parameters, the ones shown in the Config Options section
10142  * below are common options that can be passed to any Fx method.</b>
10143  * @cfg {Function} callback A function called when the effect is finished
10144  * @cfg {Object} scope The scope of the effect function
10145  * @cfg {String} easing A valid Easing value for the effect
10146  * @cfg {String} afterCls A css class to apply after the effect
10147  * @cfg {Number} duration The length of time (in seconds) that the effect should last
10148  * @cfg {Boolean} remove Whether the Element should be removed from the DOM and destroyed after the effect finishes
10149  * @cfg {Boolean} useDisplay Whether to use the <i>display</i> CSS property instead of <i>visibility</i> when hiding Elements (only applies to 
10150  * effects that end with the element being visually hidden, ignored otherwise)
10151  * @cfg {String/Object/Function} afterStyle A style specification string, e.g. "width:100px", or an object in the form {width:"100px"}, or
10152  * a function which returns such a specification that will be applied to the Element after the effect finishes
10153  * @cfg {Boolean} block Whether the effect should block other effects from queueing while it runs
10154  * @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
10155  * @cfg {Boolean} stopFx Whether subsequent effects should be stopped and removed after the current effect finishes
10156  */
10157 Roo.Fx = {
10158         /**
10159          * Slides the element into view.  An anchor point can be optionally passed to set the point of
10160          * origin for the slide effect.  This function automatically handles wrapping the element with
10161          * a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10162          * Usage:
10163          *<pre><code>
10164 // default: slide the element in from the top
10165 el.slideIn();
10166
10167 // custom: slide the element in from the right with a 2-second duration
10168 el.slideIn('r', { duration: 2 });
10169
10170 // common config options shown with default values
10171 el.slideIn('t', {
10172     easing: 'easeOut',
10173     duration: .5
10174 });
10175 </code></pre>
10176          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10177          * @param {Object} options (optional) Object literal with any of the Fx config options
10178          * @return {Roo.Element} The Element
10179          */
10180     slideIn : function(anchor, o){
10181         var el = this.getFxEl();
10182         o = o || {};
10183
10184         el.queueFx(o, function(){
10185
10186             anchor = anchor || "t";
10187
10188             // fix display to visibility
10189             this.fixDisplay();
10190
10191             // restore values after effect
10192             var r = this.getFxRestore();
10193             var b = this.getBox();
10194             // fixed size for slide
10195             this.setSize(b);
10196
10197             // wrap if needed
10198             var wrap = this.fxWrap(r.pos, o, "hidden");
10199
10200             var st = this.dom.style;
10201             st.visibility = "visible";
10202             st.position = "absolute";
10203
10204             // clear out temp styles after slide and unwrap
10205             var after = function(){
10206                 el.fxUnwrap(wrap, r.pos, o);
10207                 st.width = r.width;
10208                 st.height = r.height;
10209                 el.afterFx(o);
10210             };
10211             // time to calc the positions
10212             var a, pt = {to: [b.x, b.y]}, bw = {to: b.width}, bh = {to: b.height};
10213
10214             switch(anchor.toLowerCase()){
10215                 case "t":
10216                     wrap.setSize(b.width, 0);
10217                     st.left = st.bottom = "0";
10218                     a = {height: bh};
10219                 break;
10220                 case "l":
10221                     wrap.setSize(0, b.height);
10222                     st.right = st.top = "0";
10223                     a = {width: bw};
10224                 break;
10225                 case "r":
10226                     wrap.setSize(0, b.height);
10227                     wrap.setX(b.right);
10228                     st.left = st.top = "0";
10229                     a = {width: bw, points: pt};
10230                 break;
10231                 case "b":
10232                     wrap.setSize(b.width, 0);
10233                     wrap.setY(b.bottom);
10234                     st.left = st.top = "0";
10235                     a = {height: bh, points: pt};
10236                 break;
10237                 case "tl":
10238                     wrap.setSize(0, 0);
10239                     st.right = st.bottom = "0";
10240                     a = {width: bw, height: bh};
10241                 break;
10242                 case "bl":
10243                     wrap.setSize(0, 0);
10244                     wrap.setY(b.y+b.height);
10245                     st.right = st.top = "0";
10246                     a = {width: bw, height: bh, points: pt};
10247                 break;
10248                 case "br":
10249                     wrap.setSize(0, 0);
10250                     wrap.setXY([b.right, b.bottom]);
10251                     st.left = st.top = "0";
10252                     a = {width: bw, height: bh, points: pt};
10253                 break;
10254                 case "tr":
10255                     wrap.setSize(0, 0);
10256                     wrap.setX(b.x+b.width);
10257                     st.left = st.bottom = "0";
10258                     a = {width: bw, height: bh, points: pt};
10259                 break;
10260             }
10261             this.dom.style.visibility = "visible";
10262             wrap.show();
10263
10264             arguments.callee.anim = wrap.fxanim(a,
10265                 o,
10266                 'motion',
10267                 .5,
10268                 'easeOut', after);
10269         });
10270         return this;
10271     },
10272     
10273         /**
10274          * Slides the element out of view.  An anchor point can be optionally passed to set the end point
10275          * for the slide effect.  When the effect is completed, the element will be hidden (visibility = 
10276          * 'hidden') but block elements will still take up space in the document.  The element must be removed
10277          * from the DOM using the 'remove' config option if desired.  This function automatically handles 
10278          * wrapping the element with a fixed-size container if needed.  See the Fx class overview for valid anchor point options.
10279          * Usage:
10280          *<pre><code>
10281 // default: slide the element out to the top
10282 el.slideOut();
10283
10284 // custom: slide the element out to the right with a 2-second duration
10285 el.slideOut('r', { duration: 2 });
10286
10287 // common config options shown with default values
10288 el.slideOut('t', {
10289     easing: 'easeOut',
10290     duration: .5,
10291     remove: false,
10292     useDisplay: false
10293 });
10294 </code></pre>
10295          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to top: 't')
10296          * @param {Object} options (optional) Object literal with any of the Fx config options
10297          * @return {Roo.Element} The Element
10298          */
10299     slideOut : function(anchor, o){
10300         var el = this.getFxEl();
10301         o = o || {};
10302
10303         el.queueFx(o, function(){
10304
10305             anchor = anchor || "t";
10306
10307             // restore values after effect
10308             var r = this.getFxRestore();
10309             
10310             var b = this.getBox();
10311             // fixed size for slide
10312             this.setSize(b);
10313
10314             // wrap if needed
10315             var wrap = this.fxWrap(r.pos, o, "visible");
10316
10317             var st = this.dom.style;
10318             st.visibility = "visible";
10319             st.position = "absolute";
10320
10321             wrap.setSize(b);
10322
10323             var after = function(){
10324                 if(o.useDisplay){
10325                     el.setDisplayed(false);
10326                 }else{
10327                     el.hide();
10328                 }
10329
10330                 el.fxUnwrap(wrap, r.pos, o);
10331
10332                 st.width = r.width;
10333                 st.height = r.height;
10334
10335                 el.afterFx(o);
10336             };
10337
10338             var a, zero = {to: 0};
10339             switch(anchor.toLowerCase()){
10340                 case "t":
10341                     st.left = st.bottom = "0";
10342                     a = {height: zero};
10343                 break;
10344                 case "l":
10345                     st.right = st.top = "0";
10346                     a = {width: zero};
10347                 break;
10348                 case "r":
10349                     st.left = st.top = "0";
10350                     a = {width: zero, points: {to:[b.right, b.y]}};
10351                 break;
10352                 case "b":
10353                     st.left = st.top = "0";
10354                     a = {height: zero, points: {to:[b.x, b.bottom]}};
10355                 break;
10356                 case "tl":
10357                     st.right = st.bottom = "0";
10358                     a = {width: zero, height: zero};
10359                 break;
10360                 case "bl":
10361                     st.right = st.top = "0";
10362                     a = {width: zero, height: zero, points: {to:[b.x, b.bottom]}};
10363                 break;
10364                 case "br":
10365                     st.left = st.top = "0";
10366                     a = {width: zero, height: zero, points: {to:[b.x+b.width, b.bottom]}};
10367                 break;
10368                 case "tr":
10369                     st.left = st.bottom = "0";
10370                     a = {width: zero, height: zero, points: {to:[b.right, b.y]}};
10371                 break;
10372             }
10373
10374             arguments.callee.anim = wrap.fxanim(a,
10375                 o,
10376                 'motion',
10377                 .5,
10378                 "easeOut", after);
10379         });
10380         return this;
10381     },
10382
10383         /**
10384          * Fades the element out while slowly expanding it in all directions.  When the effect is completed, the 
10385          * element will be hidden (visibility = 'hidden') but block elements will still take up space in the document. 
10386          * The element must be removed from the DOM using the 'remove' config option if desired.
10387          * Usage:
10388          *<pre><code>
10389 // default
10390 el.puff();
10391
10392 // common config options shown with default values
10393 el.puff({
10394     easing: 'easeOut',
10395     duration: .5,
10396     remove: false,
10397     useDisplay: false
10398 });
10399 </code></pre>
10400          * @param {Object} options (optional) Object literal with any of the Fx config options
10401          * @return {Roo.Element} The Element
10402          */
10403     puff : function(o){
10404         var el = this.getFxEl();
10405         o = o || {};
10406
10407         el.queueFx(o, function(){
10408             this.clearOpacity();
10409             this.show();
10410
10411             // restore values after effect
10412             var r = this.getFxRestore();
10413             var st = this.dom.style;
10414
10415             var after = function(){
10416                 if(o.useDisplay){
10417                     el.setDisplayed(false);
10418                 }else{
10419                     el.hide();
10420                 }
10421
10422                 el.clearOpacity();
10423
10424                 el.setPositioning(r.pos);
10425                 st.width = r.width;
10426                 st.height = r.height;
10427                 st.fontSize = '';
10428                 el.afterFx(o);
10429             };
10430
10431             var width = this.getWidth();
10432             var height = this.getHeight();
10433
10434             arguments.callee.anim = this.fxanim({
10435                     width : {to: this.adjustWidth(width * 2)},
10436                     height : {to: this.adjustHeight(height * 2)},
10437                     points : {by: [-(width * .5), -(height * .5)]},
10438                     opacity : {to: 0},
10439                     fontSize: {to:200, unit: "%"}
10440                 },
10441                 o,
10442                 'motion',
10443                 .5,
10444                 "easeOut", after);
10445         });
10446         return this;
10447     },
10448
10449         /**
10450          * Blinks the element as if it was clicked and then collapses on its center (similar to switching off a television).
10451          * When the effect is completed, the element will be hidden (visibility = 'hidden') but block elements will still 
10452          * take up space in the document. The element must be removed from the DOM using the 'remove' config option if desired.
10453          * Usage:
10454          *<pre><code>
10455 // default
10456 el.switchOff();
10457
10458 // all config options shown with default values
10459 el.switchOff({
10460     easing: 'easeIn',
10461     duration: .3,
10462     remove: false,
10463     useDisplay: false
10464 });
10465 </code></pre>
10466          * @param {Object} options (optional) Object literal with any of the Fx config options
10467          * @return {Roo.Element} The Element
10468          */
10469     switchOff : function(o){
10470         var el = this.getFxEl();
10471         o = o || {};
10472
10473         el.queueFx(o, function(){
10474             this.clearOpacity();
10475             this.clip();
10476
10477             // restore values after effect
10478             var r = this.getFxRestore();
10479             var st = this.dom.style;
10480
10481             var after = function(){
10482                 if(o.useDisplay){
10483                     el.setDisplayed(false);
10484                 }else{
10485                     el.hide();
10486                 }
10487
10488                 el.clearOpacity();
10489                 el.setPositioning(r.pos);
10490                 st.width = r.width;
10491                 st.height = r.height;
10492
10493                 el.afterFx(o);
10494             };
10495
10496             this.fxanim({opacity:{to:0.3}}, null, null, .1, null, function(){
10497                 this.clearOpacity();
10498                 (function(){
10499                     this.fxanim({
10500                         height:{to:1},
10501                         points:{by:[0, this.getHeight() * .5]}
10502                     }, o, 'motion', 0.3, 'easeIn', after);
10503                 }).defer(100, this);
10504             });
10505         });
10506         return this;
10507     },
10508
10509     /**
10510      * Highlights the Element by setting a color (applies to the background-color by default, but can be
10511      * changed using the "attr" config option) and then fading back to the original color. If no original
10512      * color is available, you should provide the "endColor" config option which will be cleared after the animation.
10513      * Usage:
10514 <pre><code>
10515 // default: highlight background to yellow
10516 el.highlight();
10517
10518 // custom: highlight foreground text to blue for 2 seconds
10519 el.highlight("0000ff", { attr: 'color', duration: 2 });
10520
10521 // common config options shown with default values
10522 el.highlight("ffff9c", {
10523     attr: "background-color", //can be any valid CSS property (attribute) that supports a color value
10524     endColor: (current color) or "ffffff",
10525     easing: 'easeIn',
10526     duration: 1
10527 });
10528 </code></pre>
10529      * @param {String} color (optional) The highlight color. Should be a 6 char hex color without the leading # (defaults to yellow: 'ffff9c')
10530      * @param {Object} options (optional) Object literal with any of the Fx config options
10531      * @return {Roo.Element} The Element
10532      */ 
10533     highlight : function(color, o){
10534         var el = this.getFxEl();
10535         o = o || {};
10536
10537         el.queueFx(o, function(){
10538             color = color || "ffff9c";
10539             attr = o.attr || "backgroundColor";
10540
10541             this.clearOpacity();
10542             this.show();
10543
10544             var origColor = this.getColor(attr);
10545             var restoreColor = this.dom.style[attr];
10546             endColor = (o.endColor || origColor) || "ffffff";
10547
10548             var after = function(){
10549                 el.dom.style[attr] = restoreColor;
10550                 el.afterFx(o);
10551             };
10552
10553             var a = {};
10554             a[attr] = {from: color, to: endColor};
10555             arguments.callee.anim = this.fxanim(a,
10556                 o,
10557                 'color',
10558                 1,
10559                 'easeIn', after);
10560         });
10561         return this;
10562     },
10563
10564    /**
10565     * Shows a ripple of exploding, attenuating borders to draw attention to an Element.
10566     * Usage:
10567 <pre><code>
10568 // default: a single light blue ripple
10569 el.frame();
10570
10571 // custom: 3 red ripples lasting 3 seconds total
10572 el.frame("ff0000", 3, { duration: 3 });
10573
10574 // common config options shown with default values
10575 el.frame("C3DAF9", 1, {
10576     duration: 1 //duration of entire animation (not each individual ripple)
10577     // Note: Easing is not configurable and will be ignored if included
10578 });
10579 </code></pre>
10580     * @param {String} color (optional) The color of the border.  Should be a 6 char hex color without the leading # (defaults to light blue: 'C3DAF9').
10581     * @param {Number} count (optional) The number of ripples to display (defaults to 1)
10582     * @param {Object} options (optional) Object literal with any of the Fx config options
10583     * @return {Roo.Element} The Element
10584     */
10585     frame : function(color, count, o){
10586         var el = this.getFxEl();
10587         o = o || {};
10588
10589         el.queueFx(o, function(){
10590             color = color || "#C3DAF9";
10591             if(color.length == 6){
10592                 color = "#" + color;
10593             }
10594             count = count || 1;
10595             duration = o.duration || 1;
10596             this.show();
10597
10598             var b = this.getBox();
10599             var animFn = function(){
10600                 var proxy = this.createProxy({
10601
10602                      style:{
10603                         visbility:"hidden",
10604                         position:"absolute",
10605                         "z-index":"35000", // yee haw
10606                         border:"0px solid " + color
10607                      }
10608                   });
10609                 var scale = Roo.isBorderBox ? 2 : 1;
10610                 proxy.animate({
10611                     top:{from:b.y, to:b.y - 20},
10612                     left:{from:b.x, to:b.x - 20},
10613                     borderWidth:{from:0, to:10},
10614                     opacity:{from:1, to:0},
10615                     height:{from:b.height, to:(b.height + (20*scale))},
10616                     width:{from:b.width, to:(b.width + (20*scale))}
10617                 }, duration, function(){
10618                     proxy.remove();
10619                 });
10620                 if(--count > 0){
10621                      animFn.defer((duration/2)*1000, this);
10622                 }else{
10623                     el.afterFx(o);
10624                 }
10625             };
10626             animFn.call(this);
10627         });
10628         return this;
10629     },
10630
10631    /**
10632     * Creates a pause before any subsequent queued effects begin.  If there are
10633     * no effects queued after the pause it will have no effect.
10634     * Usage:
10635 <pre><code>
10636 el.pause(1);
10637 </code></pre>
10638     * @param {Number} seconds The length of time to pause (in seconds)
10639     * @return {Roo.Element} The Element
10640     */
10641     pause : function(seconds){
10642         var el = this.getFxEl();
10643         var o = {};
10644
10645         el.queueFx(o, function(){
10646             setTimeout(function(){
10647                 el.afterFx(o);
10648             }, seconds * 1000);
10649         });
10650         return this;
10651     },
10652
10653    /**
10654     * Fade an element in (from transparent to opaque).  The ending opacity can be specified
10655     * using the "endOpacity" config option.
10656     * Usage:
10657 <pre><code>
10658 // default: fade in from opacity 0 to 100%
10659 el.fadeIn();
10660
10661 // custom: fade in from opacity 0 to 75% over 2 seconds
10662 el.fadeIn({ endOpacity: .75, duration: 2});
10663
10664 // common config options shown with default values
10665 el.fadeIn({
10666     endOpacity: 1, //can be any value between 0 and 1 (e.g. .5)
10667     easing: 'easeOut',
10668     duration: .5
10669 });
10670 </code></pre>
10671     * @param {Object} options (optional) Object literal with any of the Fx config options
10672     * @return {Roo.Element} The Element
10673     */
10674     fadeIn : function(o){
10675         var el = this.getFxEl();
10676         o = o || {};
10677         el.queueFx(o, function(){
10678             this.setOpacity(0);
10679             this.fixDisplay();
10680             this.dom.style.visibility = 'visible';
10681             var to = o.endOpacity || 1;
10682             arguments.callee.anim = this.fxanim({opacity:{to:to}},
10683                 o, null, .5, "easeOut", function(){
10684                 if(to == 1){
10685                     this.clearOpacity();
10686                 }
10687                 el.afterFx(o);
10688             });
10689         });
10690         return this;
10691     },
10692
10693    /**
10694     * Fade an element out (from opaque to transparent).  The ending opacity can be specified
10695     * using the "endOpacity" config option.
10696     * Usage:
10697 <pre><code>
10698 // default: fade out from the element's current opacity to 0
10699 el.fadeOut();
10700
10701 // custom: fade out from the element's current opacity to 25% over 2 seconds
10702 el.fadeOut({ endOpacity: .25, duration: 2});
10703
10704 // common config options shown with default values
10705 el.fadeOut({
10706     endOpacity: 0, //can be any value between 0 and 1 (e.g. .5)
10707     easing: 'easeOut',
10708     duration: .5
10709     remove: false,
10710     useDisplay: false
10711 });
10712 </code></pre>
10713     * @param {Object} options (optional) Object literal with any of the Fx config options
10714     * @return {Roo.Element} The Element
10715     */
10716     fadeOut : function(o){
10717         var el = this.getFxEl();
10718         o = o || {};
10719         el.queueFx(o, function(){
10720             arguments.callee.anim = this.fxanim({opacity:{to:o.endOpacity || 0}},
10721                 o, null, .5, "easeOut", function(){
10722                 if(this.visibilityMode == Roo.Element.DISPLAY || o.useDisplay){
10723                      this.dom.style.display = "none";
10724                 }else{
10725                      this.dom.style.visibility = "hidden";
10726                 }
10727                 this.clearOpacity();
10728                 el.afterFx(o);
10729             });
10730         });
10731         return this;
10732     },
10733
10734    /**
10735     * Animates the transition of an element's dimensions from a starting height/width
10736     * to an ending height/width.
10737     * Usage:
10738 <pre><code>
10739 // change height and width to 100x100 pixels
10740 el.scale(100, 100);
10741
10742 // common config options shown with default values.  The height and width will default to
10743 // the element's existing values if passed as null.
10744 el.scale(
10745     [element's width],
10746     [element's height], {
10747     easing: 'easeOut',
10748     duration: .35
10749 });
10750 </code></pre>
10751     * @param {Number} width  The new width (pass undefined to keep the original width)
10752     * @param {Number} height  The new height (pass undefined to keep the original height)
10753     * @param {Object} options (optional) Object literal with any of the Fx config options
10754     * @return {Roo.Element} The Element
10755     */
10756     scale : function(w, h, o){
10757         this.shift(Roo.apply({}, o, {
10758             width: w,
10759             height: h
10760         }));
10761         return this;
10762     },
10763
10764    /**
10765     * Animates the transition of any combination of an element's dimensions, xy position and/or opacity.
10766     * Any of these properties not specified in the config object will not be changed.  This effect 
10767     * requires that at least one new dimension, position or opacity setting must be passed in on
10768     * the config object in order for the function to have any effect.
10769     * Usage:
10770 <pre><code>
10771 // slide the element horizontally to x position 200 while changing the height and opacity
10772 el.shift({ x: 200, height: 50, opacity: .8 });
10773
10774 // common config options shown with default values.
10775 el.shift({
10776     width: [element's width],
10777     height: [element's height],
10778     x: [element's x position],
10779     y: [element's y position],
10780     opacity: [element's opacity],
10781     easing: 'easeOut',
10782     duration: .35
10783 });
10784 </code></pre>
10785     * @param {Object} options  Object literal with any of the Fx config options
10786     * @return {Roo.Element} The Element
10787     */
10788     shift : function(o){
10789         var el = this.getFxEl();
10790         o = o || {};
10791         el.queueFx(o, function(){
10792             var a = {}, w = o.width, h = o.height, x = o.x, y = o.y,  op = o.opacity;
10793             if(w !== undefined){
10794                 a.width = {to: this.adjustWidth(w)};
10795             }
10796             if(h !== undefined){
10797                 a.height = {to: this.adjustHeight(h)};
10798             }
10799             if(x !== undefined || y !== undefined){
10800                 a.points = {to: [
10801                     x !== undefined ? x : this.getX(),
10802                     y !== undefined ? y : this.getY()
10803                 ]};
10804             }
10805             if(op !== undefined){
10806                 a.opacity = {to: op};
10807             }
10808             if(o.xy !== undefined){
10809                 a.points = {to: o.xy};
10810             }
10811             arguments.callee.anim = this.fxanim(a,
10812                 o, 'motion', .35, "easeOut", function(){
10813                 el.afterFx(o);
10814             });
10815         });
10816         return this;
10817     },
10818
10819         /**
10820          * Slides the element while fading it out of view.  An anchor point can be optionally passed to set the 
10821          * ending point of the effect.
10822          * Usage:
10823          *<pre><code>
10824 // default: slide the element downward while fading out
10825 el.ghost();
10826
10827 // custom: slide the element out to the right with a 2-second duration
10828 el.ghost('r', { duration: 2 });
10829
10830 // common config options shown with default values
10831 el.ghost('b', {
10832     easing: 'easeOut',
10833     duration: .5
10834     remove: false,
10835     useDisplay: false
10836 });
10837 </code></pre>
10838          * @param {String} anchor (optional) One of the valid Fx anchor positions (defaults to bottom: 'b')
10839          * @param {Object} options (optional) Object literal with any of the Fx config options
10840          * @return {Roo.Element} The Element
10841          */
10842     ghost : function(anchor, o){
10843         var el = this.getFxEl();
10844         o = o || {};
10845
10846         el.queueFx(o, function(){
10847             anchor = anchor || "b";
10848
10849             // restore values after effect
10850             var r = this.getFxRestore();
10851             var w = this.getWidth(),
10852                 h = this.getHeight();
10853
10854             var st = this.dom.style;
10855
10856             var after = function(){
10857                 if(o.useDisplay){
10858                     el.setDisplayed(false);
10859                 }else{
10860                     el.hide();
10861                 }
10862
10863                 el.clearOpacity();
10864                 el.setPositioning(r.pos);
10865                 st.width = r.width;
10866                 st.height = r.height;
10867
10868                 el.afterFx(o);
10869             };
10870
10871             var a = {opacity: {to: 0}, points: {}}, pt = a.points;
10872             switch(anchor.toLowerCase()){
10873                 case "t":
10874                     pt.by = [0, -h];
10875                 break;
10876                 case "l":
10877                     pt.by = [-w, 0];
10878                 break;
10879                 case "r":
10880                     pt.by = [w, 0];
10881                 break;
10882                 case "b":
10883                     pt.by = [0, h];
10884                 break;
10885                 case "tl":
10886                     pt.by = [-w, -h];
10887                 break;
10888                 case "bl":
10889                     pt.by = [-w, h];
10890                 break;
10891                 case "br":
10892                     pt.by = [w, h];
10893                 break;
10894                 case "tr":
10895                     pt.by = [w, -h];
10896                 break;
10897             }
10898
10899             arguments.callee.anim = this.fxanim(a,
10900                 o,
10901                 'motion',
10902                 .5,
10903                 "easeOut", after);
10904         });
10905         return this;
10906     },
10907
10908         /**
10909          * Ensures that all effects queued after syncFx is called on the element are
10910          * run concurrently.  This is the opposite of {@link #sequenceFx}.
10911          * @return {Roo.Element} The Element
10912          */
10913     syncFx : function(){
10914         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10915             block : false,
10916             concurrent : true,
10917             stopFx : false
10918         });
10919         return this;
10920     },
10921
10922         /**
10923          * Ensures that all effects queued after sequenceFx is called on the element are
10924          * run in sequence.  This is the opposite of {@link #syncFx}.
10925          * @return {Roo.Element} The Element
10926          */
10927     sequenceFx : function(){
10928         this.fxDefaults = Roo.apply(this.fxDefaults || {}, {
10929             block : false,
10930             concurrent : false,
10931             stopFx : false
10932         });
10933         return this;
10934     },
10935
10936         /* @private */
10937     nextFx : function(){
10938         var ef = this.fxQueue[0];
10939         if(ef){
10940             ef.call(this);
10941         }
10942     },
10943
10944         /**
10945          * Returns true if the element has any effects actively running or queued, else returns false.
10946          * @return {Boolean} True if element has active effects, else false
10947          */
10948     hasActiveFx : function(){
10949         return this.fxQueue && this.fxQueue[0];
10950     },
10951
10952         /**
10953          * Stops any running effects and clears the element's internal effects queue if it contains
10954          * any additional effects that haven't started yet.
10955          * @return {Roo.Element} The Element
10956          */
10957     stopFx : function(){
10958         if(this.hasActiveFx()){
10959             var cur = this.fxQueue[0];
10960             if(cur && cur.anim && cur.anim.isAnimated()){
10961                 this.fxQueue = [cur]; // clear out others
10962                 cur.anim.stop(true);
10963             }
10964         }
10965         return this;
10966     },
10967
10968         /* @private */
10969     beforeFx : function(o){
10970         if(this.hasActiveFx() && !o.concurrent){
10971            if(o.stopFx){
10972                this.stopFx();
10973                return true;
10974            }
10975            return false;
10976         }
10977         return true;
10978     },
10979
10980         /**
10981          * Returns true if the element is currently blocking so that no other effect can be queued
10982          * until this effect is finished, else returns false if blocking is not set.  This is commonly
10983          * used to ensure that an effect initiated by a user action runs to completion prior to the
10984          * same effect being restarted (e.g., firing only one effect even if the user clicks several times).
10985          * @return {Boolean} True if blocking, else false
10986          */
10987     hasFxBlock : function(){
10988         var q = this.fxQueue;
10989         return q && q[0] && q[0].block;
10990     },
10991
10992         /* @private */
10993     queueFx : function(o, fn){
10994         if(!this.fxQueue){
10995             this.fxQueue = [];
10996         }
10997         if(!this.hasFxBlock()){
10998             Roo.applyIf(o, this.fxDefaults);
10999             if(!o.concurrent){
11000                 var run = this.beforeFx(o);
11001                 fn.block = o.block;
11002                 this.fxQueue.push(fn);
11003                 if(run){
11004                     this.nextFx();
11005                 }
11006             }else{
11007                 fn.call(this);
11008             }
11009         }
11010         return this;
11011     },
11012
11013         /* @private */
11014     fxWrap : function(pos, o, vis){
11015         var wrap;
11016         if(!o.wrap || !(wrap = Roo.get(o.wrap))){
11017             var wrapXY;
11018             if(o.fixPosition){
11019                 wrapXY = this.getXY();
11020             }
11021             var div = document.createElement("div");
11022             div.style.visibility = vis;
11023             wrap = Roo.get(this.dom.parentNode.insertBefore(div, this.dom));
11024             wrap.setPositioning(pos);
11025             if(wrap.getStyle("position") == "static"){
11026                 wrap.position("relative");
11027             }
11028             this.clearPositioning('auto');
11029             wrap.clip();
11030             wrap.dom.appendChild(this.dom);
11031             if(wrapXY){
11032                 wrap.setXY(wrapXY);
11033             }
11034         }
11035         return wrap;
11036     },
11037
11038         /* @private */
11039     fxUnwrap : function(wrap, pos, o){
11040         this.clearPositioning();
11041         this.setPositioning(pos);
11042         if(!o.wrap){
11043             wrap.dom.parentNode.insertBefore(this.dom, wrap.dom);
11044             wrap.remove();
11045         }
11046     },
11047
11048         /* @private */
11049     getFxRestore : function(){
11050         var st = this.dom.style;
11051         return {pos: this.getPositioning(), width: st.width, height : st.height};
11052     },
11053
11054         /* @private */
11055     afterFx : function(o){
11056         if(o.afterStyle){
11057             this.applyStyles(o.afterStyle);
11058         }
11059         if(o.afterCls){
11060             this.addClass(o.afterCls);
11061         }
11062         if(o.remove === true){
11063             this.remove();
11064         }
11065         Roo.callback(o.callback, o.scope, [this]);
11066         if(!o.concurrent){
11067             this.fxQueue.shift();
11068             this.nextFx();
11069         }
11070     },
11071
11072         /* @private */
11073     getFxEl : function(){ // support for composite element fx
11074         return Roo.get(this.dom);
11075     },
11076
11077         /* @private */
11078     fxanim : function(args, opt, animType, defaultDur, defaultEase, cb){
11079         animType = animType || 'run';
11080         opt = opt || {};
11081         var anim = Roo.lib.Anim[animType](
11082             this.dom, args,
11083             (opt.duration || defaultDur) || .35,
11084             (opt.easing || defaultEase) || 'easeOut',
11085             function(){
11086                 Roo.callback(cb, this);
11087             },
11088             this
11089         );
11090         opt.anim = anim;
11091         return anim;
11092     }
11093 };
11094
11095 // backwords compat
11096 Roo.Fx.resize = Roo.Fx.scale;
11097
11098 //When included, Roo.Fx is automatically applied to Element so that all basic
11099 //effects are available directly via the Element API
11100 Roo.apply(Roo.Element.prototype, Roo.Fx);/*
11101  * Based on:
11102  * Ext JS Library 1.1.1
11103  * Copyright(c) 2006-2007, Ext JS, LLC.
11104  *
11105  * Originally Released Under LGPL - original licence link has changed is not relivant.
11106  *
11107  * Fork - LGPL
11108  * <script type="text/javascript">
11109  */
11110
11111
11112 /**
11113  * @class Roo.CompositeElement
11114  * Standard composite class. Creates a Roo.Element for every element in the collection.
11115  * <br><br>
11116  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11117  * actions will be performed on all the elements in this collection.</b>
11118  * <br><br>
11119  * All methods return <i>this</i> and can be chained.
11120  <pre><code>
11121  var els = Roo.select("#some-el div.some-class", true);
11122  // or select directly from an existing element
11123  var el = Roo.get('some-el');
11124  el.select('div.some-class', true);
11125
11126  els.setWidth(100); // all elements become 100 width
11127  els.hide(true); // all elements fade out and hide
11128  // or
11129  els.setWidth(100).hide(true);
11130  </code></pre>
11131  */
11132 Roo.CompositeElement = function(els){
11133     this.elements = [];
11134     this.addElements(els);
11135 };
11136 Roo.CompositeElement.prototype = {
11137     isComposite: true,
11138     addElements : function(els){
11139         if(!els) {
11140             return this;
11141         }
11142         if(typeof els == "string"){
11143             els = Roo.Element.selectorFunction(els);
11144         }
11145         var yels = this.elements;
11146         var index = yels.length-1;
11147         for(var i = 0, len = els.length; i < len; i++) {
11148                 yels[++index] = Roo.get(els[i]);
11149         }
11150         return this;
11151     },
11152
11153     /**
11154     * Clears this composite and adds the elements returned by the passed selector.
11155     * @param {String/Array} els A string CSS selector, an array of elements or an element
11156     * @return {CompositeElement} this
11157     */
11158     fill : function(els){
11159         this.elements = [];
11160         this.add(els);
11161         return this;
11162     },
11163
11164     /**
11165     * Filters this composite to only elements that match the passed selector.
11166     * @param {String} selector A string CSS selector
11167     * @param {Boolean} inverse return inverse filter (not matches)
11168     * @return {CompositeElement} this
11169     */
11170     filter : function(selector, inverse){
11171         var els = [];
11172         inverse = inverse || false;
11173         this.each(function(el){
11174             var match = inverse ? !el.is(selector) : el.is(selector);
11175             if(match){
11176                 els[els.length] = el.dom;
11177             }
11178         });
11179         this.fill(els);
11180         return this;
11181     },
11182
11183     invoke : function(fn, args){
11184         var els = this.elements;
11185         for(var i = 0, len = els.length; i < len; i++) {
11186                 Roo.Element.prototype[fn].apply(els[i], args);
11187         }
11188         return this;
11189     },
11190     /**
11191     * Adds elements to this composite.
11192     * @param {String/Array} els A string CSS selector, an array of elements or an element
11193     * @return {CompositeElement} this
11194     */
11195     add : function(els){
11196         if(typeof els == "string"){
11197             this.addElements(Roo.Element.selectorFunction(els));
11198         }else if(els.length !== undefined){
11199             this.addElements(els);
11200         }else{
11201             this.addElements([els]);
11202         }
11203         return this;
11204     },
11205     /**
11206     * Calls the passed function passing (el, this, index) for each element in this composite.
11207     * @param {Function} fn The function to call
11208     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11209     * @return {CompositeElement} this
11210     */
11211     each : function(fn, scope){
11212         var els = this.elements;
11213         for(var i = 0, len = els.length; i < len; i++){
11214             if(fn.call(scope || els[i], els[i], this, i) === false) {
11215                 break;
11216             }
11217         }
11218         return this;
11219     },
11220
11221     /**
11222      * Returns the Element object at the specified index
11223      * @param {Number} index
11224      * @return {Roo.Element}
11225      */
11226     item : function(index){
11227         return this.elements[index] || null;
11228     },
11229
11230     /**
11231      * Returns the first Element
11232      * @return {Roo.Element}
11233      */
11234     first : function(){
11235         return this.item(0);
11236     },
11237
11238     /**
11239      * Returns the last Element
11240      * @return {Roo.Element}
11241      */
11242     last : function(){
11243         return this.item(this.elements.length-1);
11244     },
11245
11246     /**
11247      * Returns the number of elements in this composite
11248      * @return Number
11249      */
11250     getCount : function(){
11251         return this.elements.length;
11252     },
11253
11254     /**
11255      * Returns true if this composite contains the passed element
11256      * @return Boolean
11257      */
11258     contains : function(el){
11259         return this.indexOf(el) !== -1;
11260     },
11261
11262     /**
11263      * Returns true if this composite contains the passed element
11264      * @return Boolean
11265      */
11266     indexOf : function(el){
11267         return this.elements.indexOf(Roo.get(el));
11268     },
11269
11270
11271     /**
11272     * Removes the specified element(s).
11273     * @param {Mixed} el The id of an element, the Element itself, the index of the element in this composite
11274     * or an array of any of those.
11275     * @param {Boolean} removeDom (optional) True to also remove the element from the document
11276     * @return {CompositeElement} this
11277     */
11278     removeElement : function(el, removeDom){
11279         if(el instanceof Array){
11280             for(var i = 0, len = el.length; i < len; i++){
11281                 this.removeElement(el[i]);
11282             }
11283             return this;
11284         }
11285         var index = typeof el == 'number' ? el : this.indexOf(el);
11286         if(index !== -1){
11287             if(removeDom){
11288                 var d = this.elements[index];
11289                 if(d.dom){
11290                     d.remove();
11291                 }else{
11292                     d.parentNode.removeChild(d);
11293                 }
11294             }
11295             this.elements.splice(index, 1);
11296         }
11297         return this;
11298     },
11299
11300     /**
11301     * Replaces the specified element with the passed element.
11302     * @param {String/HTMLElement/Element/Number} el The id of an element, the Element itself, the index of the element in this composite
11303     * to replace.
11304     * @param {String/HTMLElement/Element} replacement The id of an element or the Element itself.
11305     * @param {Boolean} domReplace (Optional) True to remove and replace the element in the document too.
11306     * @return {CompositeElement} this
11307     */
11308     replaceElement : function(el, replacement, domReplace){
11309         var index = typeof el == 'number' ? el : this.indexOf(el);
11310         if(index !== -1){
11311             if(domReplace){
11312                 this.elements[index].replaceWith(replacement);
11313             }else{
11314                 this.elements.splice(index, 1, Roo.get(replacement))
11315             }
11316         }
11317         return this;
11318     },
11319
11320     /**
11321      * Removes all elements.
11322      */
11323     clear : function(){
11324         this.elements = [];
11325     }
11326 };
11327 (function(){
11328     Roo.CompositeElement.createCall = function(proto, fnName){
11329         if(!proto[fnName]){
11330             proto[fnName] = function(){
11331                 return this.invoke(fnName, arguments);
11332             };
11333         }
11334     };
11335     for(var fnName in Roo.Element.prototype){
11336         if(typeof Roo.Element.prototype[fnName] == "function"){
11337             Roo.CompositeElement.createCall(Roo.CompositeElement.prototype, fnName);
11338         }
11339     };
11340 })();
11341 /*
11342  * Based on:
11343  * Ext JS Library 1.1.1
11344  * Copyright(c) 2006-2007, Ext JS, LLC.
11345  *
11346  * Originally Released Under LGPL - original licence link has changed is not relivant.
11347  *
11348  * Fork - LGPL
11349  * <script type="text/javascript">
11350  */
11351
11352 /**
11353  * @class Roo.CompositeElementLite
11354  * @extends Roo.CompositeElement
11355  * Flyweight composite class. Reuses the same Roo.Element for element operations.
11356  <pre><code>
11357  var els = Roo.select("#some-el div.some-class");
11358  // or select directly from an existing element
11359  var el = Roo.get('some-el');
11360  el.select('div.some-class');
11361
11362  els.setWidth(100); // all elements become 100 width
11363  els.hide(true); // all elements fade out and hide
11364  // or
11365  els.setWidth(100).hide(true);
11366  </code></pre><br><br>
11367  * <b>NOTE: Although they are not listed, this class supports all of the set/update methods of Roo.Element. All Roo.Element
11368  * actions will be performed on all the elements in this collection.</b>
11369  */
11370 Roo.CompositeElementLite = function(els){
11371     Roo.CompositeElementLite.superclass.constructor.call(this, els);
11372     this.el = new Roo.Element.Flyweight();
11373 };
11374 Roo.extend(Roo.CompositeElementLite, Roo.CompositeElement, {
11375     addElements : function(els){
11376         if(els){
11377             if(els instanceof Array){
11378                 this.elements = this.elements.concat(els);
11379             }else{
11380                 var yels = this.elements;
11381                 var index = yels.length-1;
11382                 for(var i = 0, len = els.length; i < len; i++) {
11383                     yels[++index] = els[i];
11384                 }
11385             }
11386         }
11387         return this;
11388     },
11389     invoke : function(fn, args){
11390         var els = this.elements;
11391         var el = this.el;
11392         for(var i = 0, len = els.length; i < len; i++) {
11393             el.dom = els[i];
11394                 Roo.Element.prototype[fn].apply(el, args);
11395         }
11396         return this;
11397     },
11398     /**
11399      * Returns a flyweight Element of the dom element object at the specified index
11400      * @param {Number} index
11401      * @return {Roo.Element}
11402      */
11403     item : function(index){
11404         if(!this.elements[index]){
11405             return null;
11406         }
11407         this.el.dom = this.elements[index];
11408         return this.el;
11409     },
11410
11411     // fixes scope with flyweight
11412     addListener : function(eventName, handler, scope, opt){
11413         var els = this.elements;
11414         for(var i = 0, len = els.length; i < len; i++) {
11415             Roo.EventManager.on(els[i], eventName, handler, scope || els[i], opt);
11416         }
11417         return this;
11418     },
11419
11420     /**
11421     * Calls the passed function passing (el, this, index) for each element in this composite. <b>The element
11422     * passed is the flyweight (shared) Roo.Element instance, so if you require a
11423     * a reference to the dom node, use el.dom.</b>
11424     * @param {Function} fn The function to call
11425     * @param {Object} scope (optional) The <i>this</i> object (defaults to the element)
11426     * @return {CompositeElement} this
11427     */
11428     each : function(fn, scope){
11429         var els = this.elements;
11430         var el = this.el;
11431         for(var i = 0, len = els.length; i < len; i++){
11432             el.dom = els[i];
11433                 if(fn.call(scope || el, el, this, i) === false){
11434                 break;
11435             }
11436         }
11437         return this;
11438     },
11439
11440     indexOf : function(el){
11441         return this.elements.indexOf(Roo.getDom(el));
11442     },
11443
11444     replaceElement : function(el, replacement, domReplace){
11445         var index = typeof el == 'number' ? el : this.indexOf(el);
11446         if(index !== -1){
11447             replacement = Roo.getDom(replacement);
11448             if(domReplace){
11449                 var d = this.elements[index];
11450                 d.parentNode.insertBefore(replacement, d);
11451                 d.parentNode.removeChild(d);
11452             }
11453             this.elements.splice(index, 1, replacement);
11454         }
11455         return this;
11456     }
11457 });
11458 Roo.CompositeElementLite.prototype.on = Roo.CompositeElementLite.prototype.addListener;
11459
11460 /*
11461  * Based on:
11462  * Ext JS Library 1.1.1
11463  * Copyright(c) 2006-2007, Ext JS, LLC.
11464  *
11465  * Originally Released Under LGPL - original licence link has changed is not relivant.
11466  *
11467  * Fork - LGPL
11468  * <script type="text/javascript">
11469  */
11470
11471  
11472
11473 /**
11474  * @class Roo.data.Connection
11475  * @extends Roo.util.Observable
11476  * The class encapsulates a connection to the page's originating domain, allowing requests to be made
11477  * either to a configured URL, or to a URL specified at request time. 
11478  * 
11479  * Requests made by this class are asynchronous, and will return immediately. No data from
11480  * the server will be available to the statement immediately following the {@link #request} call.
11481  * To process returned data, use a callback in the request options object, or an event listener.
11482  * 
11483  * Note: If you are doing a file upload, you will not get a normal response object sent back to
11484  * your callback or event handler.  Since the upload is handled via in IFRAME, there is no XMLHttpRequest.
11485  * The response object is created using the innerHTML of the IFRAME's document as the responseText
11486  * property and, if present, the IFRAME's XML document as the responseXML property.
11487  * 
11488  * This means that a valid XML or HTML document must be returned. If JSON data is required, it is suggested
11489  * that it be placed either inside a &lt;textarea> in an HTML document and retrieved from the responseText
11490  * using a regex, or inside a CDATA section in an XML document and retrieved from the responseXML using
11491  * standard DOM methods.
11492  * @constructor
11493  * @param {Object} config a configuration object.
11494  */
11495 Roo.data.Connection = function(config){
11496     Roo.apply(this, config);
11497     this.addEvents({
11498         /**
11499          * @event beforerequest
11500          * Fires before a network request is made to retrieve a data object.
11501          * @param {Connection} conn This Connection object.
11502          * @param {Object} options The options config object passed to the {@link #request} method.
11503          */
11504         "beforerequest" : true,
11505         /**
11506          * @event requestcomplete
11507          * Fires if the request was successfully completed.
11508          * @param {Connection} conn This Connection object.
11509          * @param {Object} response The XHR object containing the response data.
11510          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11511          * @param {Object} options The options config object passed to the {@link #request} method.
11512          */
11513         "requestcomplete" : true,
11514         /**
11515          * @event requestexception
11516          * Fires if an error HTTP status was returned from the server.
11517          * See {@link http://www.w3.org/Protocols/rfc2616/rfc2616-sec10.html} for details of HTTP status codes.
11518          * @param {Connection} conn This Connection object.
11519          * @param {Object} response The XHR object containing the response data.
11520          * See {@link http://www.w3.org/TR/XMLHttpRequest/} for details.
11521          * @param {Object} options The options config object passed to the {@link #request} method.
11522          */
11523         "requestexception" : true
11524     });
11525     Roo.data.Connection.superclass.constructor.call(this);
11526 };
11527
11528 Roo.extend(Roo.data.Connection, Roo.util.Observable, {
11529     /**
11530      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
11531      */
11532     /**
11533      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
11534      * extra parameters to each request made by this object. (defaults to undefined)
11535      */
11536     /**
11537      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
11538      *  to each request made by this object. (defaults to undefined)
11539      */
11540     /**
11541      * @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)
11542      */
11543     /**
11544      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11545      */
11546     timeout : 30000,
11547     /**
11548      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
11549      * @type Boolean
11550      */
11551     autoAbort:false,
11552
11553     /**
11554      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
11555      * @type Boolean
11556      */
11557     disableCaching: true,
11558
11559     /**
11560      * Sends an HTTP request to a remote server.
11561      * @param {Object} options An object which may contain the following properties:<ul>
11562      * <li><b>url</b> {String} (Optional) The URL to which to send the request. Defaults to configured URL</li>
11563      * <li><b>params</b> {Object/String/Function} (Optional) An object containing properties which are used as parameters to the
11564      * request, a url encoded string or a function to call to get either.</li>
11565      * <li><b>method</b> {String} (Optional) The HTTP method to use for the request. Defaults to the configured method, or
11566      * if no method was configured, "GET" if no parameters are being sent, and "POST" if parameters are being sent.</li>
11567      * <li><b>callback</b> {Function} (Optional) The function to be called upon receipt of the HTTP response.
11568      * The callback is called regardless of success or failure and is passed the following parameters:<ul>
11569      * <li>options {Object} The parameter to the request call.</li>
11570      * <li>success {Boolean} True if the request succeeded.</li>
11571      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11572      * </ul></li>
11573      * <li><b>success</b> {Function} (Optional) The function to be called upon success of the request.
11574      * The callback is passed the following parameters:<ul>
11575      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11576      * <li>options {Object} The parameter to the request call.</li>
11577      * </ul></li>
11578      * <li><b>failure</b> {Function} (Optional) The function to be called upon failure of the request.
11579      * The callback is passed the following parameters:<ul>
11580      * <li>response {Object} The XMLHttpRequest object containing the response data.</li>
11581      * <li>options {Object} The parameter to the request call.</li>
11582      * </ul></li>
11583      * <li><b>scope</b> {Object} (Optional) The scope in which to execute the callbacks: The "this" object
11584      * for the callback function. Defaults to the browser window.</li>
11585      * <li><b>form</b> {Object/String} (Optional) A form object or id to pull parameters from.</li>
11586      * <li><b>isUpload</b> {Boolean} (Optional) True if the form object is a file upload (will usually be automatically detected).</li>
11587      * <li><b>headers</b> {Object} (Optional) Request headers to set for the request.</li>
11588      * <li><b>xmlData</b> {Object} (Optional) XML document to use for the post. Note: This will be used instead of
11589      * params for the post data. Any params will be appended to the URL.</li>
11590      * <li><b>disableCaching</b> {Boolean} (Optional) True to add a unique cache-buster param to GET requests.</li>
11591      * </ul>
11592      * @return {Number} transactionId
11593      */
11594     request : function(o){
11595         if(this.fireEvent("beforerequest", this, o) !== false){
11596             var p = o.params;
11597
11598             if(typeof p == "function"){
11599                 p = p.call(o.scope||window, o);
11600             }
11601             if(typeof p == "object"){
11602                 p = Roo.urlEncode(o.params);
11603             }
11604             if(this.extraParams){
11605                 var extras = Roo.urlEncode(this.extraParams);
11606                 p = p ? (p + '&' + extras) : extras;
11607             }
11608
11609             var url = o.url || this.url;
11610             if(typeof url == 'function'){
11611                 url = url.call(o.scope||window, o);
11612             }
11613
11614             if(o.form){
11615                 var form = Roo.getDom(o.form);
11616                 url = url || form.action;
11617
11618                 var enctype = form.getAttribute("enctype");
11619                 
11620                 if (o.formData) {
11621                     return this.doFormDataUpload(o,p,url);
11622                 }
11623                 
11624                 if(o.isUpload || (enctype && enctype.toLowerCase() == 'multipart/form-data')){
11625                     return this.doFormUpload(o, p, url);
11626                 }
11627                 var f = Roo.lib.Ajax.serializeForm(form);
11628                 p = p ? (p + '&' + f) : f;
11629             }
11630
11631             var hs = o.headers;
11632             if(this.defaultHeaders){
11633                 hs = Roo.apply(hs || {}, this.defaultHeaders);
11634                 if(!o.headers){
11635                     o.headers = hs;
11636                 }
11637             }
11638
11639             var cb = {
11640                 success: this.handleResponse,
11641                 failure: this.handleFailure,
11642                 scope: this,
11643                 argument: {options: o},
11644                 timeout : o.timeout || this.timeout
11645             };
11646
11647             var method = o.method||this.method||(p ? "POST" : "GET");
11648
11649             if(method == 'GET' && (this.disableCaching && o.disableCaching !== false) || o.disableCaching === true){
11650                 url += (url.indexOf('?') != -1 ? '&' : '?') + '_dc=' + (new Date().getTime());
11651             }
11652
11653             if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11654                 if(o.autoAbort){
11655                     this.abort();
11656                 }
11657             }else if(this.autoAbort !== false){
11658                 this.abort();
11659             }
11660
11661             if((method == 'GET' && p) || o.xmlData){
11662                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
11663                 p = '';
11664             }
11665             Roo.lib.Ajax.useDefaultHeader = typeof(o.headers) == 'undefined' || typeof(o.headers['Content-Type']) == 'undefined';
11666             this.transId = Roo.lib.Ajax.request(method, url, cb, p, o);
11667             Roo.lib.Ajax.useDefaultHeader == true;
11668             return this.transId;
11669         }else{
11670             Roo.callback(o.callback, o.scope, [o, null, null]);
11671             return null;
11672         }
11673     },
11674
11675     /**
11676      * Determine whether this object has a request outstanding.
11677      * @param {Number} transactionId (Optional) defaults to the last transaction
11678      * @return {Boolean} True if there is an outstanding request.
11679      */
11680     isLoading : function(transId){
11681         if(transId){
11682             return Roo.lib.Ajax.isCallInProgress(transId);
11683         }else{
11684             return this.transId ? true : false;
11685         }
11686     },
11687
11688     /**
11689      * Aborts any outstanding request.
11690      * @param {Number} transactionId (Optional) defaults to the last transaction
11691      */
11692     abort : function(transId){
11693         if(transId || this.isLoading()){
11694             Roo.lib.Ajax.abort(transId || this.transId);
11695         }
11696     },
11697
11698     // private
11699     handleResponse : function(response){
11700         this.transId = false;
11701         var options = response.argument.options;
11702         response.argument = options ? options.argument : null;
11703         this.fireEvent("requestcomplete", this, response, options);
11704         Roo.callback(options.success, options.scope, [response, options]);
11705         Roo.callback(options.callback, options.scope, [options, true, response]);
11706     },
11707
11708     // private
11709     handleFailure : function(response, e){
11710         this.transId = false;
11711         var options = response.argument.options;
11712         response.argument = options ? options.argument : null;
11713         this.fireEvent("requestexception", this, response, options, e);
11714         Roo.callback(options.failure, options.scope, [response, options]);
11715         Roo.callback(options.callback, options.scope, [options, false, response]);
11716     },
11717
11718     // private
11719     doFormUpload : function(o, ps, url){
11720         var id = Roo.id();
11721         var frame = document.createElement('iframe');
11722         frame.id = id;
11723         frame.name = id;
11724         frame.className = 'x-hidden';
11725         if(Roo.isIE){
11726             frame.src = Roo.SSL_SECURE_URL;
11727         }
11728         document.body.appendChild(frame);
11729
11730         if(Roo.isIE){
11731            document.frames[id].name = id;
11732         }
11733
11734         var form = Roo.getDom(o.form);
11735         form.target = id;
11736         form.method = 'POST';
11737         form.enctype = form.encoding = 'multipart/form-data';
11738         if(url){
11739             form.action = url;
11740         }
11741
11742         var hiddens, hd;
11743         if(ps){ // add dynamic params
11744             hiddens = [];
11745             ps = Roo.urlDecode(ps, false);
11746             for(var k in ps){
11747                 if(ps.hasOwnProperty(k)){
11748                     hd = document.createElement('input');
11749                     hd.type = 'hidden';
11750                     hd.name = k;
11751                     hd.value = ps[k];
11752                     form.appendChild(hd);
11753                     hiddens.push(hd);
11754                 }
11755             }
11756         }
11757
11758         function cb(){
11759             var r = {  // bogus response object
11760                 responseText : '',
11761                 responseXML : null
11762             };
11763
11764             r.argument = o ? o.argument : null;
11765
11766             try { //
11767                 var doc;
11768                 if(Roo.isIE){
11769                     doc = frame.contentWindow.document;
11770                 }else {
11771                     doc = (frame.contentDocument || window.frames[id].document);
11772                 }
11773                 if(doc && doc.body){
11774                     r.responseText = doc.body.innerHTML;
11775                 }
11776                 if(doc && doc.XMLDocument){
11777                     r.responseXML = doc.XMLDocument;
11778                 }else {
11779                     r.responseXML = doc;
11780                 }
11781             }
11782             catch(e) {
11783                 // ignore
11784             }
11785
11786             Roo.EventManager.removeListener(frame, 'load', cb, this);
11787
11788             this.fireEvent("requestcomplete", this, r, o);
11789             Roo.callback(o.success, o.scope, [r, o]);
11790             Roo.callback(o.callback, o.scope, [o, true, r]);
11791
11792             setTimeout(function(){document.body.removeChild(frame);}, 100);
11793         }
11794
11795         Roo.EventManager.on(frame, 'load', cb, this);
11796         form.submit();
11797
11798         if(hiddens){ // remove dynamic params
11799             for(var i = 0, len = hiddens.length; i < len; i++){
11800                 form.removeChild(hiddens[i]);
11801             }
11802         }
11803     },
11804     // this is a 'formdata version???'
11805     
11806     
11807     doFormDataUpload : function(o, ps, url)
11808     {
11809         var form = Roo.getDom(o.form);
11810         form.enctype = form.encoding = 'multipart/form-data';
11811         var formData = o.formData === true ? new FormData(form) : o.formData;
11812       
11813         var cb = {
11814             success: this.handleResponse,
11815             failure: this.handleFailure,
11816             scope: this,
11817             argument: {options: o},
11818             timeout : o.timeout || this.timeout
11819         };
11820  
11821         if(typeof o.autoAbort == 'boolean'){ // options gets top priority
11822             if(o.autoAbort){
11823                 this.abort();
11824             }
11825         }else if(this.autoAbort !== false){
11826             this.abort();
11827         }
11828
11829         //Roo.lib.Ajax.defaultPostHeader = null;
11830         Roo.lib.Ajax.useDefaultHeader = false;
11831         this.transId = Roo.lib.Ajax.request( "POST", url, cb, o.formData, o);
11832         Roo.lib.Ajax.useDefaultHeader = true;
11833  
11834          
11835     }
11836     
11837 });
11838 /*
11839  * Based on:
11840  * Ext JS Library 1.1.1
11841  * Copyright(c) 2006-2007, Ext JS, LLC.
11842  *
11843  * Originally Released Under LGPL - original licence link has changed is not relivant.
11844  *
11845  * Fork - LGPL
11846  * <script type="text/javascript">
11847  */
11848  
11849 /**
11850  * Global Ajax request class.
11851  * 
11852  * @class Roo.Ajax
11853  * @extends Roo.data.Connection
11854  * @static
11855  * 
11856  * @cfg {String} url  The default URL to be used for requests to the server. (defaults to undefined)
11857  * @cfg {Object} extraParams  An object containing properties which are used as extra parameters to each request made by this object. (defaults to undefined)
11858  * @cfg {Object} defaultHeaders  An object containing request headers which are added to each request made by this object. (defaults to undefined)
11859  * @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)
11860  * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
11861  * @cfg {Boolean} autoAbort (Optional) Whether a new request should abort any pending requests. (defaults to false)
11862  * @cfg {Boolean} disableCaching (Optional)   True to add a unique cache-buster param to GET requests. (defaults to true)
11863  */
11864 Roo.Ajax = new Roo.data.Connection({
11865     // fix up the docs
11866     /**
11867      * @scope Roo.Ajax
11868      * @type {Boolear} 
11869      */
11870     autoAbort : false,
11871
11872     /**
11873      * Serialize the passed form into a url encoded string
11874      * @scope Roo.Ajax
11875      * @param {String/HTMLElement} form
11876      * @return {String}
11877      */
11878     serializeForm : function(form){
11879         return Roo.lib.Ajax.serializeForm(form);
11880     }
11881 });/*
11882  * Based on:
11883  * Ext JS Library 1.1.1
11884  * Copyright(c) 2006-2007, Ext JS, LLC.
11885  *
11886  * Originally Released Under LGPL - original licence link has changed is not relivant.
11887  *
11888  * Fork - LGPL
11889  * <script type="text/javascript">
11890  */
11891
11892  
11893 /**
11894  * @class Roo.UpdateManager
11895  * @extends Roo.util.Observable
11896  * Provides AJAX-style update for Element object.<br><br>
11897  * Usage:<br>
11898  * <pre><code>
11899  * // Get it from a Roo.Element object
11900  * var el = Roo.get("foo");
11901  * var mgr = el.getUpdateManager();
11902  * mgr.update("http://myserver.com/index.php", "param1=1&amp;param2=2");
11903  * ...
11904  * mgr.formUpdate("myFormId", "http://myserver.com/index.php");
11905  * <br>
11906  * // or directly (returns the same UpdateManager instance)
11907  * var mgr = new Roo.UpdateManager("myElementId");
11908  * mgr.startAutoRefresh(60, "http://myserver.com/index.php");
11909  * mgr.on("update", myFcnNeedsToKnow);
11910  * <br>
11911    // short handed call directly from the element object
11912    Roo.get("foo").load({
11913         url: "bar.php",
11914         scripts:true,
11915         params: "for=bar",
11916         text: "Loading Foo..."
11917    });
11918  * </code></pre>
11919  * @constructor
11920  * Create new UpdateManager directly.
11921  * @param {String/HTMLElement/Roo.Element} el The element to update
11922  * @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).
11923  */
11924 Roo.UpdateManager = function(el, forceNew){
11925     el = Roo.get(el);
11926     if(!forceNew && el.updateManager){
11927         return el.updateManager;
11928     }
11929     /**
11930      * The Element object
11931      * @type Roo.Element
11932      */
11933     this.el = el;
11934     /**
11935      * Cached url to use for refreshes. Overwritten every time update() is called unless "discardUrl" param is set to true.
11936      * @type String
11937      */
11938     this.defaultUrl = null;
11939
11940     this.addEvents({
11941         /**
11942          * @event beforeupdate
11943          * Fired before an update is made, return false from your handler and the update is cancelled.
11944          * @param {Roo.Element} el
11945          * @param {String/Object/Function} url
11946          * @param {String/Object} params
11947          */
11948         "beforeupdate": true,
11949         /**
11950          * @event update
11951          * Fired after successful update is made.
11952          * @param {Roo.Element} el
11953          * @param {Object} oResponseObject The response Object
11954          */
11955         "update": true,
11956         /**
11957          * @event failure
11958          * Fired on update failure.
11959          * @param {Roo.Element} el
11960          * @param {Object} oResponseObject The response Object
11961          */
11962         "failure": true
11963     });
11964     var d = Roo.UpdateManager.defaults;
11965     /**
11966      * Blank page URL to use with SSL file uploads (Defaults to Roo.UpdateManager.defaults.sslBlankUrl or "about:blank").
11967      * @type String
11968      */
11969     this.sslBlankUrl = d.sslBlankUrl;
11970     /**
11971      * Whether to append unique parameter on get request to disable caching (Defaults to Roo.UpdateManager.defaults.disableCaching or false).
11972      * @type Boolean
11973      */
11974     this.disableCaching = d.disableCaching;
11975     /**
11976      * Text for loading indicator (Defaults to Roo.UpdateManager.defaults.indicatorText or '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
11977      * @type String
11978      */
11979     this.indicatorText = d.indicatorText;
11980     /**
11981      * Whether to show indicatorText when loading (Defaults to Roo.UpdateManager.defaults.showLoadIndicator or true).
11982      * @type String
11983      */
11984     this.showLoadIndicator = d.showLoadIndicator;
11985     /**
11986      * Timeout for requests or form posts in seconds (Defaults to Roo.UpdateManager.defaults.timeout or 30 seconds).
11987      * @type Number
11988      */
11989     this.timeout = d.timeout;
11990
11991     /**
11992      * True to process scripts in the output (Defaults to Roo.UpdateManager.defaults.loadScripts (false)).
11993      * @type Boolean
11994      */
11995     this.loadScripts = d.loadScripts;
11996
11997     /**
11998      * Transaction object of current executing transaction
11999      */
12000     this.transaction = null;
12001
12002     /**
12003      * @private
12004      */
12005     this.autoRefreshProcId = null;
12006     /**
12007      * Delegate for refresh() prebound to "this", use myUpdater.refreshDelegate.createCallback(arg1, arg2) to bind arguments
12008      * @type Function
12009      */
12010     this.refreshDelegate = this.refresh.createDelegate(this);
12011     /**
12012      * Delegate for update() prebound to "this", use myUpdater.updateDelegate.createCallback(arg1, arg2) to bind arguments
12013      * @type Function
12014      */
12015     this.updateDelegate = this.update.createDelegate(this);
12016     /**
12017      * Delegate for formUpdate() prebound to "this", use myUpdater.formUpdateDelegate.createCallback(arg1, arg2) to bind arguments
12018      * @type Function
12019      */
12020     this.formUpdateDelegate = this.formUpdate.createDelegate(this);
12021     /**
12022      * @private
12023      */
12024     this.successDelegate = this.processSuccess.createDelegate(this);
12025     /**
12026      * @private
12027      */
12028     this.failureDelegate = this.processFailure.createDelegate(this);
12029
12030     if(!this.renderer){
12031      /**
12032       * The renderer for this UpdateManager. Defaults to {@link Roo.UpdateManager.BasicRenderer}.
12033       */
12034     this.renderer = new Roo.UpdateManager.BasicRenderer();
12035     }
12036     
12037     Roo.UpdateManager.superclass.constructor.call(this);
12038 };
12039
12040 Roo.extend(Roo.UpdateManager, Roo.util.Observable, {
12041     /**
12042      * Get the Element this UpdateManager is bound to
12043      * @return {Roo.Element} The element
12044      */
12045     getEl : function(){
12046         return this.el;
12047     },
12048     /**
12049      * Performs an async request, updating this element with the response. If params are specified it uses POST, otherwise it uses GET.
12050      * @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:
12051 <pre><code>
12052 um.update({<br/>
12053     url: "your-url.php",<br/>
12054     params: {param1: "foo", param2: "bar"}, // or a URL encoded string<br/>
12055     callback: yourFunction,<br/>
12056     scope: yourObject, //(optional scope)  <br/>
12057     discardUrl: false, <br/>
12058     nocache: false,<br/>
12059     text: "Loading...",<br/>
12060     timeout: 30,<br/>
12061     scripts: false<br/>
12062 });
12063 </code></pre>
12064      * The only required property is url. The optional properties nocache, text and scripts
12065      * are shorthand for disableCaching, indicatorText and loadScripts and are used to set their associated property on this UpdateManager instance.
12066      * @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}
12067      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12068      * @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.
12069      */
12070     update : function(url, params, callback, discardUrl){
12071         if(this.fireEvent("beforeupdate", this.el, url, params) !== false){
12072             var method = this.method,
12073                 cfg;
12074             if(typeof url == "object"){ // must be config object
12075                 cfg = url;
12076                 url = cfg.url;
12077                 params = params || cfg.params;
12078                 callback = callback || cfg.callback;
12079                 discardUrl = discardUrl || cfg.discardUrl;
12080                 if(callback && cfg.scope){
12081                     callback = callback.createDelegate(cfg.scope);
12082                 }
12083                 if(typeof cfg.method != "undefined"){method = cfg.method;};
12084                 if(typeof cfg.nocache != "undefined"){this.disableCaching = cfg.nocache;};
12085                 if(typeof cfg.text != "undefined"){this.indicatorText = '<div class="loading-indicator">'+cfg.text+"</div>";};
12086                 if(typeof cfg.scripts != "undefined"){this.loadScripts = cfg.scripts;};
12087                 if(typeof cfg.timeout != "undefined"){this.timeout = cfg.timeout;};
12088             }
12089             this.showLoading();
12090             if(!discardUrl){
12091                 this.defaultUrl = url;
12092             }
12093             if(typeof url == "function"){
12094                 url = url.call(this);
12095             }
12096
12097             method = method || (params ? "POST" : "GET");
12098             if(method == "GET"){
12099                 url = this.prepareUrl(url);
12100             }
12101
12102             var o = Roo.apply(cfg ||{}, {
12103                 url : url,
12104                 params: params,
12105                 success: this.successDelegate,
12106                 failure: this.failureDelegate,
12107                 callback: undefined,
12108                 timeout: (this.timeout*1000),
12109                 argument: {"url": url, "form": null, "callback": callback, "params": params}
12110             });
12111             Roo.log("updated manager called with timeout of " + o.timeout);
12112             this.transaction = Roo.Ajax.request(o);
12113         }
12114     },
12115
12116     /**
12117      * 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.
12118      * Uses this.sslBlankUrl for SSL file uploads to prevent IE security warning.
12119      * @param {String/HTMLElement} form The form Id or form element
12120      * @param {String} url (optional) The url to pass the form to. If omitted the action attribute on the form will be used.
12121      * @param {Boolean} reset (optional) Whether to try to reset the form after the update
12122      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess, oResponse)
12123      */
12124     formUpdate : function(form, url, reset, callback){
12125         if(this.fireEvent("beforeupdate", this.el, form, url) !== false){
12126             if(typeof url == "function"){
12127                 url = url.call(this);
12128             }
12129             form = Roo.getDom(form);
12130             this.transaction = Roo.Ajax.request({
12131                 form: form,
12132                 url:url,
12133                 success: this.successDelegate,
12134                 failure: this.failureDelegate,
12135                 timeout: (this.timeout*1000),
12136                 argument: {"url": url, "form": form, "callback": callback, "reset": reset}
12137             });
12138             this.showLoading.defer(1, this);
12139         }
12140     },
12141
12142     /**
12143      * Refresh the element with the last used url or defaultUrl. If there is no url, it returns immediately
12144      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12145      */
12146     refresh : function(callback){
12147         if(this.defaultUrl == null){
12148             return;
12149         }
12150         this.update(this.defaultUrl, null, callback, true);
12151     },
12152
12153     /**
12154      * Set this element to auto refresh.
12155      * @param {Number} interval How often to update (in seconds).
12156      * @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)
12157      * @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}
12158      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
12159      * @param {Boolean} refreshNow (optional) Whether to execute the refresh now, or wait the interval
12160      */
12161     startAutoRefresh : function(interval, url, params, callback, refreshNow){
12162         if(refreshNow){
12163             this.update(url || this.defaultUrl, params, callback, true);
12164         }
12165         if(this.autoRefreshProcId){
12166             clearInterval(this.autoRefreshProcId);
12167         }
12168         this.autoRefreshProcId = setInterval(this.update.createDelegate(this, [url || this.defaultUrl, params, callback, true]), interval*1000);
12169     },
12170
12171     /**
12172      * Stop auto refresh on this element.
12173      */
12174      stopAutoRefresh : function(){
12175         if(this.autoRefreshProcId){
12176             clearInterval(this.autoRefreshProcId);
12177             delete this.autoRefreshProcId;
12178         }
12179     },
12180
12181     isAutoRefreshing : function(){
12182        return this.autoRefreshProcId ? true : false;
12183     },
12184     /**
12185      * Called to update the element to "Loading" state. Override to perform custom action.
12186      */
12187     showLoading : function(){
12188         if(this.showLoadIndicator){
12189             this.el.update(this.indicatorText);
12190         }
12191     },
12192
12193     /**
12194      * Adds unique parameter to query string if disableCaching = true
12195      * @private
12196      */
12197     prepareUrl : function(url){
12198         if(this.disableCaching){
12199             var append = "_dc=" + (new Date().getTime());
12200             if(url.indexOf("?") !== -1){
12201                 url += "&" + append;
12202             }else{
12203                 url += "?" + append;
12204             }
12205         }
12206         return url;
12207     },
12208
12209     /**
12210      * @private
12211      */
12212     processSuccess : function(response){
12213         this.transaction = null;
12214         if(response.argument.form && response.argument.reset){
12215             try{ // put in try/catch since some older FF releases had problems with this
12216                 response.argument.form.reset();
12217             }catch(e){}
12218         }
12219         if(this.loadScripts){
12220             this.renderer.render(this.el, response, this,
12221                 this.updateComplete.createDelegate(this, [response]));
12222         }else{
12223             this.renderer.render(this.el, response, this);
12224             this.updateComplete(response);
12225         }
12226     },
12227
12228     updateComplete : function(response){
12229         this.fireEvent("update", this.el, response);
12230         if(typeof response.argument.callback == "function"){
12231             response.argument.callback(this.el, true, response);
12232         }
12233     },
12234
12235     /**
12236      * @private
12237      */
12238     processFailure : function(response){
12239         this.transaction = null;
12240         this.fireEvent("failure", this.el, response);
12241         if(typeof response.argument.callback == "function"){
12242             response.argument.callback(this.el, false, response);
12243         }
12244     },
12245
12246     /**
12247      * Set the content renderer for this UpdateManager. See {@link Roo.UpdateManager.BasicRenderer#render} for more details.
12248      * @param {Object} renderer The object implementing the render() method
12249      */
12250     setRenderer : function(renderer){
12251         this.renderer = renderer;
12252     },
12253
12254     getRenderer : function(){
12255        return this.renderer;
12256     },
12257
12258     /**
12259      * Set the defaultUrl used for updates
12260      * @param {String/Function} defaultUrl The url or a function to call to get the url
12261      */
12262     setDefaultUrl : function(defaultUrl){
12263         this.defaultUrl = defaultUrl;
12264     },
12265
12266     /**
12267      * Aborts the executing transaction
12268      */
12269     abort : function(){
12270         if(this.transaction){
12271             Roo.Ajax.abort(this.transaction);
12272         }
12273     },
12274
12275     /**
12276      * Returns true if an update is in progress
12277      * @return {Boolean}
12278      */
12279     isUpdating : function(){
12280         if(this.transaction){
12281             return Roo.Ajax.isLoading(this.transaction);
12282         }
12283         return false;
12284     }
12285 });
12286
12287 /**
12288  * @class Roo.UpdateManager.defaults
12289  * @static (not really - but it helps the doc tool)
12290  * The defaults collection enables customizing the default properties of UpdateManager
12291  */
12292    Roo.UpdateManager.defaults = {
12293        /**
12294          * Timeout for requests or form posts in seconds (Defaults 30 seconds).
12295          * @type Number
12296          */
12297          timeout : 30,
12298
12299          /**
12300          * True to process scripts by default (Defaults to false).
12301          * @type Boolean
12302          */
12303         loadScripts : false,
12304
12305         /**
12306         * Blank page URL to use with SSL file uploads (Defaults to "javascript:false").
12307         * @type String
12308         */
12309         sslBlankUrl : (Roo.SSL_SECURE_URL || "javascript:false"),
12310         /**
12311          * Whether to append unique parameter on get request to disable caching (Defaults to false).
12312          * @type Boolean
12313          */
12314         disableCaching : false,
12315         /**
12316          * Whether to show indicatorText when loading (Defaults to true).
12317          * @type Boolean
12318          */
12319         showLoadIndicator : true,
12320         /**
12321          * Text for loading indicator (Defaults to '&lt;div class="loading-indicator"&gt;Loading...&lt;/div&gt;').
12322          * @type String
12323          */
12324         indicatorText : '<div class="loading-indicator">Loading...</div>'
12325    };
12326
12327 /**
12328  * Static convenience method. This method is deprecated in favor of el.load({url:'foo.php', ...}).
12329  *Usage:
12330  * <pre><code>Roo.UpdateManager.updateElement("my-div", "stuff.php");</code></pre>
12331  * @param {String/HTMLElement/Roo.Element} el The element to update
12332  * @param {String} url The url
12333  * @param {String/Object} params (optional) Url encoded param string or an object of name/value pairs
12334  * @param {Object} options (optional) A config object with any of the UpdateManager properties you want to set - for example: {disableCaching:true, indicatorText: "Loading data..."}
12335  * @static
12336  * @deprecated
12337  * @member Roo.UpdateManager
12338  */
12339 Roo.UpdateManager.updateElement = function(el, url, params, options){
12340     var um = Roo.get(el, true).getUpdateManager();
12341     Roo.apply(um, options);
12342     um.update(url, params, options ? options.callback : null);
12343 };
12344 // alias for backwards compat
12345 Roo.UpdateManager.update = Roo.UpdateManager.updateElement;
12346 /**
12347  * @class Roo.UpdateManager.BasicRenderer
12348  * Default Content renderer. Updates the elements innerHTML with the responseText.
12349  */
12350 Roo.UpdateManager.BasicRenderer = function(){};
12351
12352 Roo.UpdateManager.BasicRenderer.prototype = {
12353     /**
12354      * This is called when the transaction is completed and it's time to update the element - The BasicRenderer
12355      * updates the elements innerHTML with the responseText - To perform a custom render (i.e. XML or JSON processing),
12356      * create an object with a "render(el, response)" method and pass it to setRenderer on the UpdateManager.
12357      * @param {Roo.Element} el The element being rendered
12358      * @param {Object} response The YUI Connect response object
12359      * @param {UpdateManager} updateManager The calling update manager
12360      * @param {Function} callback A callback that will need to be called if loadScripts is true on the UpdateManager
12361      */
12362      render : function(el, response, updateManager, callback){
12363         el.update(response.responseText, updateManager.loadScripts, callback);
12364     }
12365 };
12366 /*
12367  * Based on:
12368  * Roo JS
12369  * (c)) Alan Knowles
12370  * Licence : LGPL
12371  */
12372
12373
12374 /**
12375  * @class Roo.DomTemplate
12376  * @extends Roo.Template
12377  * An effort at a dom based template engine..
12378  *
12379  * Similar to XTemplate, except it uses dom parsing to create the template..
12380  *
12381  * Supported features:
12382  *
12383  *  Tags:
12384
12385 <pre><code>
12386       {a_variable} - output encoded.
12387       {a_variable.format:("Y-m-d")} - call a method on the variable
12388       {a_variable:raw} - unencoded output
12389       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
12390       {a_variable:this.method_on_template(...)} - call a method on the template object.
12391  
12392 </code></pre>
12393  *  The tpl tag:
12394 <pre><code>
12395         &lt;div roo-for="a_variable or condition.."&gt;&lt;/div&gt;
12396         &lt;div roo-if="a_variable or condition"&gt;&lt;/div&gt;
12397         &lt;div roo-exec="some javascript"&gt;&lt;/div&gt;
12398         &lt;div roo-name="named_template"&gt;&lt;/div&gt; 
12399   
12400 </code></pre>
12401  *      
12402  */
12403 Roo.DomTemplate = function()
12404 {
12405      Roo.DomTemplate.superclass.constructor.apply(this, arguments);
12406      if (this.html) {
12407         this.compile();
12408      }
12409 };
12410
12411
12412 Roo.extend(Roo.DomTemplate, Roo.Template, {
12413     /**
12414      * id counter for sub templates.
12415      */
12416     id : 0,
12417     /**
12418      * flag to indicate if dom parser is inside a pre,
12419      * it will strip whitespace if not.
12420      */
12421     inPre : false,
12422     
12423     /**
12424      * The various sub templates
12425      */
12426     tpls : false,
12427     
12428     
12429     
12430     /**
12431      *
12432      * basic tag replacing syntax
12433      * WORD:WORD()
12434      *
12435      * // you can fake an object call by doing this
12436      *  x.t:(test,tesT) 
12437      * 
12438      */
12439     re : /(\{|\%7B)([\w-\.]+)(?:\:([\w\.]*)(?:\(([^)]*?)?\))?)?(\}|\%7D)/g,
12440     //re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
12441     
12442     iterChild : function (node, method) {
12443         
12444         var oldPre = this.inPre;
12445         if (node.tagName == 'PRE') {
12446             this.inPre = true;
12447         }
12448         for( var i = 0; i < node.childNodes.length; i++) {
12449             method.call(this, node.childNodes[i]);
12450         }
12451         this.inPre = oldPre;
12452     },
12453     
12454     
12455     
12456     /**
12457      * compile the template
12458      *
12459      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
12460      *
12461      */
12462     compile: function()
12463     {
12464         var s = this.html;
12465         
12466         // covert the html into DOM...
12467         var doc = false;
12468         var div =false;
12469         try {
12470             doc = document.implementation.createHTMLDocument("");
12471             doc.documentElement.innerHTML =   this.html  ;
12472             div = doc.documentElement;
12473         } catch (e) {
12474             // old IE... - nasty -- it causes all sorts of issues.. with
12475             // images getting pulled from server..
12476             div = document.createElement('div');
12477             div.innerHTML = this.html;
12478         }
12479         //doc.documentElement.innerHTML = htmlBody
12480          
12481         
12482         
12483         this.tpls = [];
12484         var _t = this;
12485         this.iterChild(div, function(n) {_t.compileNode(n, true); });
12486         
12487         var tpls = this.tpls;
12488         
12489         // create a top level template from the snippet..
12490         
12491         //Roo.log(div.innerHTML);
12492         
12493         var tpl = {
12494             uid : 'master',
12495             id : this.id++,
12496             attr : false,
12497             value : false,
12498             body : div.innerHTML,
12499             
12500             forCall : false,
12501             execCall : false,
12502             dom : div,
12503             isTop : true
12504             
12505         };
12506         tpls.unshift(tpl);
12507         
12508         
12509         // compile them...
12510         this.tpls = [];
12511         Roo.each(tpls, function(tp){
12512             this.compileTpl(tp);
12513             this.tpls[tp.id] = tp;
12514         }, this);
12515         
12516         this.master = tpls[0];
12517         return this;
12518         
12519         
12520     },
12521     
12522     compileNode : function(node, istop) {
12523         // test for
12524         //Roo.log(node);
12525         
12526         
12527         // skip anything not a tag..
12528         if (node.nodeType != 1) {
12529             if (node.nodeType == 3 && !this.inPre) {
12530                 // reduce white space..
12531                 node.nodeValue = node.nodeValue.replace(/\s+/g, ' '); 
12532                 
12533             }
12534             return;
12535         }
12536         
12537         var tpl = {
12538             uid : false,
12539             id : false,
12540             attr : false,
12541             value : false,
12542             body : '',
12543             
12544             forCall : false,
12545             execCall : false,
12546             dom : false,
12547             isTop : istop
12548             
12549             
12550         };
12551         
12552         
12553         switch(true) {
12554             case (node.hasAttribute('roo-for')): tpl.attr = 'for'; break;
12555             case (node.hasAttribute('roo-if')): tpl.attr = 'if'; break;
12556             case (node.hasAttribute('roo-name')): tpl.attr = 'name'; break;
12557             case (node.hasAttribute('roo-exec')): tpl.attr = 'exec'; break;
12558             // no default..
12559         }
12560         
12561         
12562         if (!tpl.attr) {
12563             // just itterate children..
12564             this.iterChild(node,this.compileNode);
12565             return;
12566         }
12567         tpl.uid = this.id++;
12568         tpl.value = node.getAttribute('roo-' +  tpl.attr);
12569         node.removeAttribute('roo-'+ tpl.attr);
12570         if (tpl.attr != 'name') {
12571             var placeholder = document.createTextNode('{domtpl' + tpl.uid + '}');
12572             node.parentNode.replaceChild(placeholder,  node);
12573         } else {
12574             
12575             var placeholder =  document.createElement('span');
12576             placeholder.className = 'roo-tpl-' + tpl.value;
12577             node.parentNode.replaceChild(placeholder,  node);
12578         }
12579         
12580         // parent now sees '{domtplXXXX}
12581         this.iterChild(node,this.compileNode);
12582         
12583         // we should now have node body...
12584         var div = document.createElement('div');
12585         div.appendChild(node);
12586         tpl.dom = node;
12587         // this has the unfortunate side effect of converting tagged attributes
12588         // eg. href="{...}" into %7C...%7D
12589         // this has been fixed by searching for those combo's although it's a bit hacky..
12590         
12591         
12592         tpl.body = div.innerHTML;
12593         
12594         
12595          
12596         tpl.id = tpl.uid;
12597         switch(tpl.attr) {
12598             case 'for' :
12599                 switch (tpl.value) {
12600                     case '.':  tpl.forCall = new Function('values', 'parent', 'with(values){ return values; }'); break;
12601                     case '..': tpl.forCall= new Function('values', 'parent', 'with(values){ return parent; }'); break;
12602                     default:   tpl.forCall= new Function('values', 'parent', 'with(values){ return '+tpl.value+'; }');
12603                 }
12604                 break;
12605             
12606             case 'exec':
12607                 tpl.execCall = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12608                 break;
12609             
12610             case 'if':     
12611                 tpl.ifCall = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(tpl.value))+'; }');
12612                 break;
12613             
12614             case 'name':
12615                 tpl.id  = tpl.value; // replace non characters???
12616                 break;
12617             
12618         }
12619         
12620         
12621         this.tpls.push(tpl);
12622         
12623         
12624         
12625     },
12626     
12627     
12628     
12629     
12630     /**
12631      * Compile a segment of the template into a 'sub-template'
12632      *
12633      * 
12634      * 
12635      *
12636      */
12637     compileTpl : function(tpl)
12638     {
12639         var fm = Roo.util.Format;
12640         var useF = this.disableFormats !== true;
12641         
12642         var sep = Roo.isGecko ? "+\n" : ",\n";
12643         
12644         var undef = function(str) {
12645             Roo.debug && Roo.log("Property not found :"  + str);
12646             return '';
12647         };
12648           
12649         //Roo.log(tpl.body);
12650         
12651         
12652         
12653         var fn = function(m, lbrace, name, format, args)
12654         {
12655             //Roo.log("ARGS");
12656             //Roo.log(arguments);
12657             args = args ? args.replace(/\\'/g,"'") : args;
12658             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
12659             if (typeof(format) == 'undefined') {
12660                 format =  'htmlEncode'; 
12661             }
12662             if (format == 'raw' ) {
12663                 format = false;
12664             }
12665             
12666             if(name.substr(0, 6) == 'domtpl'){
12667                 return "'"+ sep +'this.applySubTemplate('+name.substr(6)+', values, parent)'+sep+"'";
12668             }
12669             
12670             // build an array of options to determine if value is undefined..
12671             
12672             // basically get 'xxxx.yyyy' then do
12673             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
12674             //    (function () { Roo.log("Property not found"); return ''; })() :
12675             //    ......
12676             
12677             var udef_ar = [];
12678             var lookfor = '';
12679             Roo.each(name.split('.'), function(st) {
12680                 lookfor += (lookfor.length ? '.': '') + st;
12681                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
12682             });
12683             
12684             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
12685             
12686             
12687             if(format && useF){
12688                 
12689                 args = args ? ',' + args : "";
12690                  
12691                 if(format.substr(0, 5) != "this."){
12692                     format = "fm." + format + '(';
12693                 }else{
12694                     format = 'this.call("'+ format.substr(5) + '", ';
12695                     args = ", values";
12696                 }
12697                 
12698                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
12699             }
12700              
12701             if (args && args.length) {
12702                 // called with xxyx.yuu:(test,test)
12703                 // change to ()
12704                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
12705             }
12706             // raw.. - :raw modifier..
12707             return "'"+ sep + udef_st  + name + ")"+sep+"'";
12708             
12709         };
12710         var body;
12711         // branched to use + in gecko and [].join() in others
12712         if(Roo.isGecko){
12713             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
12714                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
12715                     "';};};";
12716         }else{
12717             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
12718             body.push(tpl.body.replace(/(\r\n|\n)/g,
12719                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
12720             body.push("'].join('');};};");
12721             body = body.join('');
12722         }
12723         
12724         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
12725        
12726         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
12727         eval(body);
12728         
12729         return this;
12730     },
12731      
12732     /**
12733      * same as applyTemplate, except it's done to one of the subTemplates
12734      * when using named templates, you can do:
12735      *
12736      * var str = pl.applySubTemplate('your-name', values);
12737      *
12738      * 
12739      * @param {Number} id of the template
12740      * @param {Object} values to apply to template
12741      * @param {Object} parent (normaly the instance of this object)
12742      */
12743     applySubTemplate : function(id, values, parent)
12744     {
12745         
12746         
12747         var t = this.tpls[id];
12748         
12749         
12750         try { 
12751             if(t.ifCall && !t.ifCall.call(this, values, parent)){
12752                 Roo.debug && Roo.log('if call on ' + t.value + ' return false');
12753                 return '';
12754             }
12755         } catch(e) {
12756             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-if="' + t.value + '" - ' + e.toString());
12757             Roo.log(values);
12758           
12759             return '';
12760         }
12761         try { 
12762             
12763             if(t.execCall && t.execCall.call(this, values, parent)){
12764                 return '';
12765             }
12766         } catch(e) {
12767             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12768             Roo.log(values);
12769             return '';
12770         }
12771         
12772         try {
12773             var vs = t.forCall ? t.forCall.call(this, values, parent) : values;
12774             parent = t.target ? values : parent;
12775             if(t.forCall && vs instanceof Array){
12776                 var buf = [];
12777                 for(var i = 0, len = vs.length; i < len; i++){
12778                     try {
12779                         buf[buf.length] = t.compiled.call(this, vs[i], parent);
12780                     } catch (e) {
12781                         Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12782                         Roo.log(e.body);
12783                         //Roo.log(t.compiled);
12784                         Roo.log(vs[i]);
12785                     }   
12786                 }
12787                 return buf.join('');
12788             }
12789         } catch (e) {
12790             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on roo-for="' + t.value + '" - ' + e.toString());
12791             Roo.log(values);
12792             return '';
12793         }
12794         try {
12795             return t.compiled.call(this, vs, parent);
12796         } catch (e) {
12797             Roo.log('Xtemplate.applySubTemplate('+ id+ '): Exception thrown on body="' + t.value + '" - ' + e.toString());
12798             Roo.log(e.body);
12799             //Roo.log(t.compiled);
12800             Roo.log(values);
12801             return '';
12802         }
12803     },
12804
12805    
12806
12807     applyTemplate : function(values){
12808         return this.master.compiled.call(this, values, {});
12809         //var s = this.subs;
12810     },
12811
12812     apply : function(){
12813         return this.applyTemplate.apply(this, arguments);
12814     }
12815
12816  });
12817
12818 Roo.DomTemplate.from = function(el){
12819     el = Roo.getDom(el);
12820     return new Roo.Domtemplate(el.value || el.innerHTML);
12821 };/*
12822  * Based on:
12823  * Ext JS Library 1.1.1
12824  * Copyright(c) 2006-2007, Ext JS, LLC.
12825  *
12826  * Originally Released Under LGPL - original licence link has changed is not relivant.
12827  *
12828  * Fork - LGPL
12829  * <script type="text/javascript">
12830  */
12831
12832 /**
12833  * @class Roo.util.DelayedTask
12834  * Provides a convenient method of performing setTimeout where a new
12835  * timeout cancels the old timeout. An example would be performing validation on a keypress.
12836  * You can use this class to buffer
12837  * the keypress events for a certain number of milliseconds, and perform only if they stop
12838  * for that amount of time.
12839  * @constructor The parameters to this constructor serve as defaults and are not required.
12840  * @param {Function} fn (optional) The default function to timeout
12841  * @param {Object} scope (optional) The default scope of that timeout
12842  * @param {Array} args (optional) The default Array of arguments
12843  */
12844 Roo.util.DelayedTask = function(fn, scope, args){
12845     var id = null, d, t;
12846
12847     var call = function(){
12848         var now = new Date().getTime();
12849         if(now - t >= d){
12850             clearInterval(id);
12851             id = null;
12852             fn.apply(scope, args || []);
12853         }
12854     };
12855     /**
12856      * Cancels any pending timeout and queues a new one
12857      * @param {Number} delay The milliseconds to delay
12858      * @param {Function} newFn (optional) Overrides function passed to constructor
12859      * @param {Object} newScope (optional) Overrides scope passed to constructor
12860      * @param {Array} newArgs (optional) Overrides args passed to constructor
12861      */
12862     this.delay = function(delay, newFn, newScope, newArgs){
12863         if(id && delay != d){
12864             this.cancel();
12865         }
12866         d = delay;
12867         t = new Date().getTime();
12868         fn = newFn || fn;
12869         scope = newScope || scope;
12870         args = newArgs || args;
12871         if(!id){
12872             id = setInterval(call, d);
12873         }
12874     };
12875
12876     /**
12877      * Cancel the last queued timeout
12878      */
12879     this.cancel = function(){
12880         if(id){
12881             clearInterval(id);
12882             id = null;
12883         }
12884     };
12885 };/*
12886  * Based on:
12887  * Ext JS Library 1.1.1
12888  * Copyright(c) 2006-2007, Ext JS, LLC.
12889  *
12890  * Originally Released Under LGPL - original licence link has changed is not relivant.
12891  *
12892  * Fork - LGPL
12893  * <script type="text/javascript">
12894  */
12895  
12896  
12897 Roo.util.TaskRunner = function(interval){
12898     interval = interval || 10;
12899     var tasks = [], removeQueue = [];
12900     var id = 0;
12901     var running = false;
12902
12903     var stopThread = function(){
12904         running = false;
12905         clearInterval(id);
12906         id = 0;
12907     };
12908
12909     var startThread = function(){
12910         if(!running){
12911             running = true;
12912             id = setInterval(runTasks, interval);
12913         }
12914     };
12915
12916     var removeTask = function(task){
12917         removeQueue.push(task);
12918         if(task.onStop){
12919             task.onStop();
12920         }
12921     };
12922
12923     var runTasks = function(){
12924         if(removeQueue.length > 0){
12925             for(var i = 0, len = removeQueue.length; i < len; i++){
12926                 tasks.remove(removeQueue[i]);
12927             }
12928             removeQueue = [];
12929             if(tasks.length < 1){
12930                 stopThread();
12931                 return;
12932             }
12933         }
12934         var now = new Date().getTime();
12935         for(var i = 0, len = tasks.length; i < len; ++i){
12936             var t = tasks[i];
12937             var itime = now - t.taskRunTime;
12938             if(t.interval <= itime){
12939                 var rt = t.run.apply(t.scope || t, t.args || [++t.taskRunCount]);
12940                 t.taskRunTime = now;
12941                 if(rt === false || t.taskRunCount === t.repeat){
12942                     removeTask(t);
12943                     return;
12944                 }
12945             }
12946             if(t.duration && t.duration <= (now - t.taskStartTime)){
12947                 removeTask(t);
12948             }
12949         }
12950     };
12951
12952     /**
12953      * Queues a new task.
12954      * @param {Object} task
12955      */
12956     this.start = function(task){
12957         tasks.push(task);
12958         task.taskStartTime = new Date().getTime();
12959         task.taskRunTime = 0;
12960         task.taskRunCount = 0;
12961         startThread();
12962         return task;
12963     };
12964
12965     this.stop = function(task){
12966         removeTask(task);
12967         return task;
12968     };
12969
12970     this.stopAll = function(){
12971         stopThread();
12972         for(var i = 0, len = tasks.length; i < len; i++){
12973             if(tasks[i].onStop){
12974                 tasks[i].onStop();
12975             }
12976         }
12977         tasks = [];
12978         removeQueue = [];
12979     };
12980 };
12981
12982 Roo.TaskMgr = new Roo.util.TaskRunner();/*
12983  * Based on:
12984  * Ext JS Library 1.1.1
12985  * Copyright(c) 2006-2007, Ext JS, LLC.
12986  *
12987  * Originally Released Under LGPL - original licence link has changed is not relivant.
12988  *
12989  * Fork - LGPL
12990  * <script type="text/javascript">
12991  */
12992
12993  
12994 /**
12995  * @class Roo.util.MixedCollection
12996  * @extends Roo.util.Observable
12997  * A Collection class that maintains both numeric indexes and keys and exposes events.
12998  * @constructor
12999  * @param {Boolean} allowFunctions True if the addAll function should add function references to the
13000  * collection (defaults to false)
13001  * @param {Function} keyFn A function that can accept an item of the type(s) stored in this MixedCollection
13002  * and return the key value for that item.  This is used when available to look up the key on items that
13003  * were passed without an explicit key parameter to a MixedCollection method.  Passing this parameter is
13004  * equivalent to providing an implementation for the {@link #getKey} method.
13005  */
13006 Roo.util.MixedCollection = function(allowFunctions, keyFn){
13007     this.items = [];
13008     this.map = {};
13009     this.keys = [];
13010     this.length = 0;
13011     this.addEvents({
13012         /**
13013          * @event clear
13014          * Fires when the collection is cleared.
13015          */
13016         "clear" : true,
13017         /**
13018          * @event add
13019          * Fires when an item is added to the collection.
13020          * @param {Number} index The index at which the item was added.
13021          * @param {Object} o The item added.
13022          * @param {String} key The key associated with the added item.
13023          */
13024         "add" : true,
13025         /**
13026          * @event replace
13027          * Fires when an item is replaced in the collection.
13028          * @param {String} key he key associated with the new added.
13029          * @param {Object} old The item being replaced.
13030          * @param {Object} new The new item.
13031          */
13032         "replace" : true,
13033         /**
13034          * @event remove
13035          * Fires when an item is removed from the collection.
13036          * @param {Object} o The item being removed.
13037          * @param {String} key (optional) The key associated with the removed item.
13038          */
13039         "remove" : true,
13040         "sort" : true
13041     });
13042     this.allowFunctions = allowFunctions === true;
13043     if(keyFn){
13044         this.getKey = keyFn;
13045     }
13046     Roo.util.MixedCollection.superclass.constructor.call(this);
13047 };
13048
13049 Roo.extend(Roo.util.MixedCollection, Roo.util.Observable, {
13050     allowFunctions : false,
13051     
13052 /**
13053  * Adds an item to the collection.
13054  * @param {String} key The key to associate with the item
13055  * @param {Object} o The item to add.
13056  * @return {Object} The item added.
13057  */
13058     add : function(key, o){
13059         if(arguments.length == 1){
13060             o = arguments[0];
13061             key = this.getKey(o);
13062         }
13063         if(typeof key == "undefined" || key === null){
13064             this.length++;
13065             this.items.push(o);
13066             this.keys.push(null);
13067         }else{
13068             var old = this.map[key];
13069             if(old){
13070                 return this.replace(key, o);
13071             }
13072             this.length++;
13073             this.items.push(o);
13074             this.map[key] = o;
13075             this.keys.push(key);
13076         }
13077         this.fireEvent("add", this.length-1, o, key);
13078         return o;
13079     },
13080        
13081 /**
13082   * MixedCollection has a generic way to fetch keys if you implement getKey.
13083 <pre><code>
13084 // normal way
13085 var mc = new Roo.util.MixedCollection();
13086 mc.add(someEl.dom.id, someEl);
13087 mc.add(otherEl.dom.id, otherEl);
13088 //and so on
13089
13090 // using getKey
13091 var mc = new Roo.util.MixedCollection();
13092 mc.getKey = function(el){
13093    return el.dom.id;
13094 };
13095 mc.add(someEl);
13096 mc.add(otherEl);
13097
13098 // or via the constructor
13099 var mc = new Roo.util.MixedCollection(false, function(el){
13100    return el.dom.id;
13101 });
13102 mc.add(someEl);
13103 mc.add(otherEl);
13104 </code></pre>
13105  * @param o {Object} The item for which to find the key.
13106  * @return {Object} The key for the passed item.
13107  */
13108     getKey : function(o){
13109          return o.id; 
13110     },
13111    
13112 /**
13113  * Replaces an item in the collection.
13114  * @param {String} key The key associated with the item to replace, or the item to replace.
13115  * @param o {Object} o (optional) If the first parameter passed was a key, the item to associate with that key.
13116  * @return {Object}  The new item.
13117  */
13118     replace : function(key, o){
13119         if(arguments.length == 1){
13120             o = arguments[0];
13121             key = this.getKey(o);
13122         }
13123         var old = this.item(key);
13124         if(typeof key == "undefined" || key === null || typeof old == "undefined"){
13125              return this.add(key, o);
13126         }
13127         var index = this.indexOfKey(key);
13128         this.items[index] = o;
13129         this.map[key] = o;
13130         this.fireEvent("replace", key, old, o);
13131         return o;
13132     },
13133    
13134 /**
13135  * Adds all elements of an Array or an Object to the collection.
13136  * @param {Object/Array} objs An Object containing properties which will be added to the collection, or
13137  * an Array of values, each of which are added to the collection.
13138  */
13139     addAll : function(objs){
13140         if(arguments.length > 1 || objs instanceof Array){
13141             var args = arguments.length > 1 ? arguments : objs;
13142             for(var i = 0, len = args.length; i < len; i++){
13143                 this.add(args[i]);
13144             }
13145         }else{
13146             for(var key in objs){
13147                 if(this.allowFunctions || typeof objs[key] != "function"){
13148                     this.add(key, objs[key]);
13149                 }
13150             }
13151         }
13152     },
13153    
13154 /**
13155  * Executes the specified function once for every item in the collection, passing each
13156  * item as the first and only parameter. returning false from the function will stop the iteration.
13157  * @param {Function} fn The function to execute for each item.
13158  * @param {Object} scope (optional) The scope in which to execute the function.
13159  */
13160     each : function(fn, scope){
13161         var items = [].concat(this.items); // each safe for removal
13162         for(var i = 0, len = items.length; i < len; i++){
13163             if(fn.call(scope || items[i], items[i], i, len) === false){
13164                 break;
13165             }
13166         }
13167     },
13168    
13169 /**
13170  * Executes the specified function once for every key in the collection, passing each
13171  * key, and its associated item as the first two parameters.
13172  * @param {Function} fn The function to execute for each item.
13173  * @param {Object} scope (optional) The scope in which to execute the function.
13174  */
13175     eachKey : function(fn, scope){
13176         for(var i = 0, len = this.keys.length; i < len; i++){
13177             fn.call(scope || window, this.keys[i], this.items[i], i, len);
13178         }
13179     },
13180    
13181 /**
13182  * Returns the first item in the collection which elicits a true return value from the
13183  * passed selection function.
13184  * @param {Function} fn The selection function to execute for each item.
13185  * @param {Object} scope (optional) The scope in which to execute the function.
13186  * @return {Object} The first item in the collection which returned true from the selection function.
13187  */
13188     find : function(fn, scope){
13189         for(var i = 0, len = this.items.length; i < len; i++){
13190             if(fn.call(scope || window, this.items[i], this.keys[i])){
13191                 return this.items[i];
13192             }
13193         }
13194         return null;
13195     },
13196    
13197 /**
13198  * Inserts an item at the specified index in the collection.
13199  * @param {Number} index The index to insert the item at.
13200  * @param {String} key The key to associate with the new item, or the item itself.
13201  * @param {Object} o  (optional) If the second parameter was a key, the new item.
13202  * @return {Object} The item inserted.
13203  */
13204     insert : function(index, key, o){
13205         if(arguments.length == 2){
13206             o = arguments[1];
13207             key = this.getKey(o);
13208         }
13209         if(index >= this.length){
13210             return this.add(key, o);
13211         }
13212         this.length++;
13213         this.items.splice(index, 0, o);
13214         if(typeof key != "undefined" && key != null){
13215             this.map[key] = o;
13216         }
13217         this.keys.splice(index, 0, key);
13218         this.fireEvent("add", index, o, key);
13219         return o;
13220     },
13221    
13222 /**
13223  * Removed an item from the collection.
13224  * @param {Object} o The item to remove.
13225  * @return {Object} The item removed.
13226  */
13227     remove : function(o){
13228         return this.removeAt(this.indexOf(o));
13229     },
13230    
13231 /**
13232  * Remove an item from a specified index in the collection.
13233  * @param {Number} index The index within the collection of the item to remove.
13234  */
13235     removeAt : function(index){
13236         if(index < this.length && index >= 0){
13237             this.length--;
13238             var o = this.items[index];
13239             this.items.splice(index, 1);
13240             var key = this.keys[index];
13241             if(typeof key != "undefined"){
13242                 delete this.map[key];
13243             }
13244             this.keys.splice(index, 1);
13245             this.fireEvent("remove", o, key);
13246         }
13247     },
13248    
13249 /**
13250  * Removed an item associated with the passed key fom the collection.
13251  * @param {String} key The key of the item to remove.
13252  */
13253     removeKey : function(key){
13254         return this.removeAt(this.indexOfKey(key));
13255     },
13256    
13257 /**
13258  * Returns the number of items in the collection.
13259  * @return {Number} the number of items in the collection.
13260  */
13261     getCount : function(){
13262         return this.length; 
13263     },
13264    
13265 /**
13266  * Returns index within the collection of the passed Object.
13267  * @param {Object} o The item to find the index of.
13268  * @return {Number} index of the item.
13269  */
13270     indexOf : function(o){
13271         if(!this.items.indexOf){
13272             for(var i = 0, len = this.items.length; i < len; i++){
13273                 if(this.items[i] == o) {
13274                     return i;
13275                 }
13276             }
13277             return -1;
13278         }else{
13279             return this.items.indexOf(o);
13280         }
13281     },
13282    
13283 /**
13284  * Returns index within the collection of the passed key.
13285  * @param {String} key The key to find the index of.
13286  * @return {Number} index of the key.
13287  */
13288     indexOfKey : function(key){
13289         if(!this.keys.indexOf){
13290             for(var i = 0, len = this.keys.length; i < len; i++){
13291                 if(this.keys[i] == key) {
13292                     return i;
13293                 }
13294             }
13295             return -1;
13296         }else{
13297             return this.keys.indexOf(key);
13298         }
13299     },
13300    
13301 /**
13302  * Returns the item associated with the passed key OR index. Key has priority over index.
13303  * @param {String/Number} key The key or index of the item.
13304  * @return {Object} The item associated with the passed key.
13305  */
13306     item : function(key){
13307         var item = typeof this.map[key] != "undefined" ? this.map[key] : this.items[key];
13308         return typeof item != 'function' || this.allowFunctions ? item : null; // for prototype!
13309     },
13310     
13311 /**
13312  * Returns the item at the specified index.
13313  * @param {Number} index The index of the item.
13314  * @return {Object}
13315  */
13316     itemAt : function(index){
13317         return this.items[index];
13318     },
13319     
13320 /**
13321  * Returns the item associated with the passed key.
13322  * @param {String/Number} key The key of the item.
13323  * @return {Object} The item associated with the passed key.
13324  */
13325     key : function(key){
13326         return this.map[key];
13327     },
13328    
13329 /**
13330  * Returns true if the collection contains the passed Object as an item.
13331  * @param {Object} o  The Object to look for in the collection.
13332  * @return {Boolean} True if the collection contains the Object as an item.
13333  */
13334     contains : function(o){
13335         return this.indexOf(o) != -1;
13336     },
13337    
13338 /**
13339  * Returns true if the collection contains the passed Object as a key.
13340  * @param {String} key The key to look for in the collection.
13341  * @return {Boolean} True if the collection contains the Object as a key.
13342  */
13343     containsKey : function(key){
13344         return typeof this.map[key] != "undefined";
13345     },
13346    
13347 /**
13348  * Removes all items from the collection.
13349  */
13350     clear : function(){
13351         this.length = 0;
13352         this.items = [];
13353         this.keys = [];
13354         this.map = {};
13355         this.fireEvent("clear");
13356     },
13357    
13358 /**
13359  * Returns the first item in the collection.
13360  * @return {Object} the first item in the collection..
13361  */
13362     first : function(){
13363         return this.items[0]; 
13364     },
13365    
13366 /**
13367  * Returns the last item in the collection.
13368  * @return {Object} the last item in the collection..
13369  */
13370     last : function(){
13371         return this.items[this.length-1];   
13372     },
13373     
13374     _sort : function(property, dir, fn){
13375         var dsc = String(dir).toUpperCase() == "DESC" ? -1 : 1;
13376         fn = fn || function(a, b){
13377             return a-b;
13378         };
13379         var c = [], k = this.keys, items = this.items;
13380         for(var i = 0, len = items.length; i < len; i++){
13381             c[c.length] = {key: k[i], value: items[i], index: i};
13382         }
13383         c.sort(function(a, b){
13384             var v = fn(a[property], b[property]) * dsc;
13385             if(v == 0){
13386                 v = (a.index < b.index ? -1 : 1);
13387             }
13388             return v;
13389         });
13390         for(var i = 0, len = c.length; i < len; i++){
13391             items[i] = c[i].value;
13392             k[i] = c[i].key;
13393         }
13394         this.fireEvent("sort", this);
13395     },
13396     
13397     /**
13398      * Sorts this collection with the passed comparison function
13399      * @param {String} direction (optional) "ASC" or "DESC"
13400      * @param {Function} fn (optional) comparison function
13401      */
13402     sort : function(dir, fn){
13403         this._sort("value", dir, fn);
13404     },
13405     
13406     /**
13407      * Sorts this collection by keys
13408      * @param {String} direction (optional) "ASC" or "DESC"
13409      * @param {Function} fn (optional) a comparison function (defaults to case insensitive string)
13410      */
13411     keySort : function(dir, fn){
13412         this._sort("key", dir, fn || function(a, b){
13413             return String(a).toUpperCase()-String(b).toUpperCase();
13414         });
13415     },
13416     
13417     /**
13418      * Returns a range of items in this collection
13419      * @param {Number} startIndex (optional) defaults to 0
13420      * @param {Number} endIndex (optional) default to the last item
13421      * @return {Array} An array of items
13422      */
13423     getRange : function(start, end){
13424         var items = this.items;
13425         if(items.length < 1){
13426             return [];
13427         }
13428         start = start || 0;
13429         end = Math.min(typeof end == "undefined" ? this.length-1 : end, this.length-1);
13430         var r = [];
13431         if(start <= end){
13432             for(var i = start; i <= end; i++) {
13433                     r[r.length] = items[i];
13434             }
13435         }else{
13436             for(var i = start; i >= end; i--) {
13437                     r[r.length] = items[i];
13438             }
13439         }
13440         return r;
13441     },
13442         
13443     /**
13444      * Filter the <i>objects</i> in this collection by a specific property. 
13445      * Returns a new collection that has been filtered.
13446      * @param {String} property A property on your objects
13447      * @param {String/RegExp} value Either string that the property values 
13448      * should start with or a RegExp to test against the property
13449      * @return {MixedCollection} The new filtered collection
13450      */
13451     filter : function(property, value){
13452         if(!value.exec){ // not a regex
13453             value = String(value);
13454             if(value.length == 0){
13455                 return this.clone();
13456             }
13457             value = new RegExp("^" + Roo.escapeRe(value), "i");
13458         }
13459         return this.filterBy(function(o){
13460             return o && value.test(o[property]);
13461         });
13462         },
13463     
13464     /**
13465      * Filter by a function. * Returns a new collection that has been filtered.
13466      * The passed function will be called with each 
13467      * object in the collection. If the function returns true, the value is included 
13468      * otherwise it is filtered.
13469      * @param {Function} fn The function to be called, it will receive the args o (the object), k (the key)
13470      * @param {Object} scope (optional) The scope of the function (defaults to this) 
13471      * @return {MixedCollection} The new filtered collection
13472      */
13473     filterBy : function(fn, scope){
13474         var r = new Roo.util.MixedCollection();
13475         r.getKey = this.getKey;
13476         var k = this.keys, it = this.items;
13477         for(var i = 0, len = it.length; i < len; i++){
13478             if(fn.call(scope||this, it[i], k[i])){
13479                                 r.add(k[i], it[i]);
13480                         }
13481         }
13482         return r;
13483     },
13484     
13485     /**
13486      * Creates a duplicate of this collection
13487      * @return {MixedCollection}
13488      */
13489     clone : function(){
13490         var r = new Roo.util.MixedCollection();
13491         var k = this.keys, it = this.items;
13492         for(var i = 0, len = it.length; i < len; i++){
13493             r.add(k[i], it[i]);
13494         }
13495         r.getKey = this.getKey;
13496         return r;
13497     }
13498 });
13499 /**
13500  * Returns the item associated with the passed key or index.
13501  * @method
13502  * @param {String/Number} key The key or index of the item.
13503  * @return {Object} The item associated with the passed key.
13504  */
13505 Roo.util.MixedCollection.prototype.get = Roo.util.MixedCollection.prototype.item;/*
13506  * Based on:
13507  * Ext JS Library 1.1.1
13508  * Copyright(c) 2006-2007, Ext JS, LLC.
13509  *
13510  * Originally Released Under LGPL - original licence link has changed is not relivant.
13511  *
13512  * Fork - LGPL
13513  * <script type="text/javascript">
13514  */
13515 /**
13516  * @class Roo.util.JSON
13517  * Modified version of Douglas Crockford"s json.js that doesn"t
13518  * mess with the Object prototype 
13519  * http://www.json.org/js.html
13520  * @singleton
13521  */
13522 Roo.util.JSON = new (function(){
13523     var useHasOwn = {}.hasOwnProperty ? true : false;
13524     
13525     // crashes Safari in some instances
13526     //var validRE = /^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/;
13527     
13528     var pad = function(n) {
13529         return n < 10 ? "0" + n : n;
13530     };
13531     
13532     var m = {
13533         "\b": '\\b',
13534         "\t": '\\t',
13535         "\n": '\\n',
13536         "\f": '\\f',
13537         "\r": '\\r',
13538         '"' : '\\"',
13539         "\\": '\\\\'
13540     };
13541
13542     var encodeString = function(s){
13543         if (/["\\\x00-\x1f]/.test(s)) {
13544             return '"' + s.replace(/([\x00-\x1f\\"])/g, function(a, b) {
13545                 var c = m[b];
13546                 if(c){
13547                     return c;
13548                 }
13549                 c = b.charCodeAt();
13550                 return "\\u00" +
13551                     Math.floor(c / 16).toString(16) +
13552                     (c % 16).toString(16);
13553             }) + '"';
13554         }
13555         return '"' + s + '"';
13556     };
13557     
13558     var encodeArray = function(o){
13559         var a = ["["], b, i, l = o.length, v;
13560             for (i = 0; i < l; i += 1) {
13561                 v = o[i];
13562                 switch (typeof v) {
13563                     case "undefined":
13564                     case "function":
13565                     case "unknown":
13566                         break;
13567                     default:
13568                         if (b) {
13569                             a.push(',');
13570                         }
13571                         a.push(v === null ? "null" : Roo.util.JSON.encode(v));
13572                         b = true;
13573                 }
13574             }
13575             a.push("]");
13576             return a.join("");
13577     };
13578     
13579     var encodeDate = function(o){
13580         return '"' + o.getFullYear() + "-" +
13581                 pad(o.getMonth() + 1) + "-" +
13582                 pad(o.getDate()) + "T" +
13583                 pad(o.getHours()) + ":" +
13584                 pad(o.getMinutes()) + ":" +
13585                 pad(o.getSeconds()) + '"';
13586     };
13587     
13588     /**
13589      * Encodes an Object, Array or other value
13590      * @param {Mixed} o The variable to encode
13591      * @return {String} The JSON string
13592      */
13593     this.encode = function(o)
13594     {
13595         // should this be extended to fully wrap stringify..
13596         
13597         if(typeof o == "undefined" || o === null){
13598             return "null";
13599         }else if(o instanceof Array){
13600             return encodeArray(o);
13601         }else if(o instanceof Date){
13602             return encodeDate(o);
13603         }else if(typeof o == "string"){
13604             return encodeString(o);
13605         }else if(typeof o == "number"){
13606             return isFinite(o) ? String(o) : "null";
13607         }else if(typeof o == "boolean"){
13608             return String(o);
13609         }else {
13610             var a = ["{"], b, i, v;
13611             for (i in o) {
13612                 if(!useHasOwn || o.hasOwnProperty(i)) {
13613                     v = o[i];
13614                     switch (typeof v) {
13615                     case "undefined":
13616                     case "function":
13617                     case "unknown":
13618                         break;
13619                     default:
13620                         if(b){
13621                             a.push(',');
13622                         }
13623                         a.push(this.encode(i), ":",
13624                                 v === null ? "null" : this.encode(v));
13625                         b = true;
13626                     }
13627                 }
13628             }
13629             a.push("}");
13630             return a.join("");
13631         }
13632     };
13633     
13634     /**
13635      * Decodes (parses) a JSON string to an object. If the JSON is invalid, this function throws a SyntaxError.
13636      * @param {String} json The JSON string
13637      * @return {Object} The resulting object
13638      */
13639     this.decode = function(json){
13640         
13641         return  /** eval:var:json */ eval("(" + json + ')');
13642     };
13643 })();
13644 /** 
13645  * Shorthand for {@link Roo.util.JSON#encode}
13646  * @member Roo encode 
13647  * @method */
13648 Roo.encode = typeof(JSON) != 'undefined' && JSON.stringify ? JSON.stringify : Roo.util.JSON.encode;
13649 /** 
13650  * Shorthand for {@link Roo.util.JSON#decode}
13651  * @member Roo decode 
13652  * @method */
13653 Roo.decode = typeof(JSON) != 'undefined' && JSON.parse ? JSON.parse : Roo.util.JSON.decode;
13654 /*
13655  * Based on:
13656  * Ext JS Library 1.1.1
13657  * Copyright(c) 2006-2007, Ext JS, LLC.
13658  *
13659  * Originally Released Under LGPL - original licence link has changed is not relivant.
13660  *
13661  * Fork - LGPL
13662  * <script type="text/javascript">
13663  */
13664  
13665 /**
13666  * @class Roo.util.Format
13667  * Reusable data formatting functions
13668  * @singleton
13669  */
13670 Roo.util.Format = function(){
13671     var trimRe = /^\s+|\s+$/g;
13672     return {
13673         /**
13674          * Truncate a string and add an ellipsis ('...') to the end if it exceeds the specified length
13675          * @param {String} value The string to truncate
13676          * @param {Number} length The maximum length to allow before truncating
13677          * @return {String} The converted text
13678          */
13679         ellipsis : function(value, len){
13680             if(value && value.length > len){
13681                 return value.substr(0, len-3)+"...";
13682             }
13683             return value;
13684         },
13685
13686         /**
13687          * Checks a reference and converts it to empty string if it is undefined
13688          * @param {Mixed} value Reference to check
13689          * @return {Mixed} Empty string if converted, otherwise the original value
13690          */
13691         undef : function(value){
13692             return typeof value != "undefined" ? value : "";
13693         },
13694
13695         /**
13696          * Convert certain characters (&, <, >, and ') to their HTML character equivalents for literal display in web pages.
13697          * @param {String} value The string to encode
13698          * @return {String} The encoded text
13699          */
13700         htmlEncode : function(value){
13701             return !value ? value : String(value).replace(/&/g, "&amp;").replace(/>/g, "&gt;").replace(/</g, "&lt;").replace(/"/g, "&quot;");
13702         },
13703
13704         /**
13705          * Convert certain characters (&, <, >, and ') from their HTML character equivalents.
13706          * @param {String} value The string to decode
13707          * @return {String} The decoded text
13708          */
13709         htmlDecode : function(value){
13710             return !value ? value : String(value).replace(/&amp;/g, "&").replace(/&gt;/g, ">").replace(/&lt;/g, "<").replace(/&quot;/g, '"');
13711         },
13712
13713         /**
13714          * Trims any whitespace from either side of a string
13715          * @param {String} value The text to trim
13716          * @return {String} The trimmed text
13717          */
13718         trim : function(value){
13719             return String(value).replace(trimRe, "");
13720         },
13721
13722         /**
13723          * Returns a substring from within an original string
13724          * @param {String} value The original text
13725          * @param {Number} start The start index of the substring
13726          * @param {Number} length The length of the substring
13727          * @return {String} The substring
13728          */
13729         substr : function(value, start, length){
13730             return String(value).substr(start, length);
13731         },
13732
13733         /**
13734          * Converts a string to all lower case letters
13735          * @param {String} value The text to convert
13736          * @return {String} The converted text
13737          */
13738         lowercase : function(value){
13739             return String(value).toLowerCase();
13740         },
13741
13742         /**
13743          * Converts a string to all upper case letters
13744          * @param {String} value The text to convert
13745          * @return {String} The converted text
13746          */
13747         uppercase : function(value){
13748             return String(value).toUpperCase();
13749         },
13750
13751         /**
13752          * Converts the first character only of a string to upper case
13753          * @param {String} value The text to convert
13754          * @return {String} The converted text
13755          */
13756         capitalize : function(value){
13757             return !value ? value : value.charAt(0).toUpperCase() + value.substr(1).toLowerCase();
13758         },
13759
13760         // private
13761         call : function(value, fn){
13762             if(arguments.length > 2){
13763                 var args = Array.prototype.slice.call(arguments, 2);
13764                 args.unshift(value);
13765                  
13766                 return /** eval:var:value */  eval(fn).apply(window, args);
13767             }else{
13768                 /** eval:var:value */
13769                 return /** eval:var:value */ eval(fn).call(window, value);
13770             }
13771         },
13772
13773        
13774         /**
13775          * safer version of Math.toFixed..??/
13776          * @param {Number/String} value The numeric value to format
13777          * @param {Number/String} value Decimal places 
13778          * @return {String} The formatted currency string
13779          */
13780         toFixed : function(v, n)
13781         {
13782             // why not use to fixed - precision is buggered???
13783             if (!n) {
13784                 return Math.round(v-0);
13785             }
13786             var fact = Math.pow(10,n+1);
13787             v = (Math.round((v-0)*fact))/fact;
13788             var z = (''+fact).substring(2);
13789             if (v == Math.floor(v)) {
13790                 return Math.floor(v) + '.' + z;
13791             }
13792             
13793             // now just padd decimals..
13794             var ps = String(v).split('.');
13795             var fd = (ps[1] + z);
13796             var r = fd.substring(0,n); 
13797             var rm = fd.substring(n); 
13798             if (rm < 5) {
13799                 return ps[0] + '.' + r;
13800             }
13801             r*=1; // turn it into a number;
13802             r++;
13803             if (String(r).length != n) {
13804                 ps[0]*=1;
13805                 ps[0]++;
13806                 r = String(r).substring(1); // chop the end off.
13807             }
13808             
13809             return ps[0] + '.' + r;
13810              
13811         },
13812         
13813         /**
13814          * Format a number as US currency
13815          * @param {Number/String} value The numeric value to format
13816          * @return {String} The formatted currency string
13817          */
13818         usMoney : function(v){
13819             return '$' + Roo.util.Format.number(v);
13820         },
13821         
13822         /**
13823          * Format a number
13824          * eventually this should probably emulate php's number_format
13825          * @param {Number/String} value The numeric value to format
13826          * @param {Number} decimals number of decimal places
13827          * @param {String} delimiter for thousands (default comma)
13828          * @return {String} The formatted currency string
13829          */
13830         number : function(v, decimals, thousandsDelimiter)
13831         {
13832             // multiply and round.
13833             decimals = typeof(decimals) == 'undefined' ? 2 : decimals;
13834             thousandsDelimiter = typeof(thousandsDelimiter) == 'undefined' ? ',' : thousandsDelimiter;
13835             
13836             var mul = Math.pow(10, decimals);
13837             var zero = String(mul).substring(1);
13838             v = (Math.round((v-0)*mul))/mul;
13839             
13840             // if it's '0' number.. then
13841             
13842             //v = (v == Math.floor(v)) ? v + "." + zero : ((v*10 == Math.floor(v*10)) ? v + "0" : v);
13843             v = String(v);
13844             var ps = v.split('.');
13845             var whole = ps[0];
13846             
13847             var r = /(\d+)(\d{3})/;
13848             // add comma's
13849             
13850             if(thousandsDelimiter.length != 0) {
13851                 whole = whole.replace(/\B(?=(\d{3})+(?!\d))/g, thousandsDelimiter );
13852             } 
13853             
13854             var sub = ps[1] ?
13855                     // has decimals..
13856                     (decimals ?  ('.'+ ps[1] + zero.substring(ps[1].length)) : '') :
13857                     // does not have decimals
13858                     (decimals ? ('.' + zero) : '');
13859             
13860             
13861             return whole + sub ;
13862         },
13863         
13864         /**
13865          * Parse a value into a formatted date using the specified format pattern.
13866          * @param {Mixed} value The value to format
13867          * @param {String} format (optional) Any valid date format string (defaults to 'm/d/Y')
13868          * @return {String} The formatted date string
13869          */
13870         date : function(v, format){
13871             if(!v){
13872                 return "";
13873             }
13874             if(!(v instanceof Date)){
13875                 v = new Date(Date.parse(v));
13876             }
13877             return v.dateFormat(format || Roo.util.Format.defaults.date);
13878         },
13879
13880         /**
13881          * Returns a date rendering function that can be reused to apply a date format multiple times efficiently
13882          * @param {String} format Any valid date format string
13883          * @return {Function} The date formatting function
13884          */
13885         dateRenderer : function(format){
13886             return function(v){
13887                 return Roo.util.Format.date(v, format);  
13888             };
13889         },
13890
13891         // private
13892         stripTagsRE : /<\/?[^>]+>/gi,
13893         
13894         /**
13895          * Strips all HTML tags
13896          * @param {Mixed} value The text from which to strip tags
13897          * @return {String} The stripped text
13898          */
13899         stripTags : function(v){
13900             return !v ? v : String(v).replace(this.stripTagsRE, "");
13901         }
13902     };
13903 }();
13904 Roo.util.Format.defaults = {
13905     date : 'd/M/Y'
13906 };/*
13907  * Based on:
13908  * Ext JS Library 1.1.1
13909  * Copyright(c) 2006-2007, Ext JS, LLC.
13910  *
13911  * Originally Released Under LGPL - original licence link has changed is not relivant.
13912  *
13913  * Fork - LGPL
13914  * <script type="text/javascript">
13915  */
13916
13917
13918  
13919
13920 /**
13921  * @class Roo.MasterTemplate
13922  * @extends Roo.Template
13923  * Provides a template that can have child templates. The syntax is:
13924 <pre><code>
13925 var t = new Roo.MasterTemplate(
13926         '&lt;select name="{name}"&gt;',
13927                 '&lt;tpl name="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
13928         '&lt;/select&gt;'
13929 );
13930 t.add('options', {value: 'foo', text: 'bar'});
13931 // or you can add multiple child elements in one shot
13932 t.addAll('options', [
13933     {value: 'foo', text: 'bar'},
13934     {value: 'foo2', text: 'bar2'},
13935     {value: 'foo3', text: 'bar3'}
13936 ]);
13937 // then append, applying the master template values
13938 t.append('my-form', {name: 'my-select'});
13939 </code></pre>
13940 * A name attribute for the child template is not required if you have only one child
13941 * template or you want to refer to them by index.
13942  */
13943 Roo.MasterTemplate = function(){
13944     Roo.MasterTemplate.superclass.constructor.apply(this, arguments);
13945     this.originalHtml = this.html;
13946     var st = {};
13947     var m, re = this.subTemplateRe;
13948     re.lastIndex = 0;
13949     var subIndex = 0;
13950     while(m = re.exec(this.html)){
13951         var name = m[1], content = m[2];
13952         st[subIndex] = {
13953             name: name,
13954             index: subIndex,
13955             buffer: [],
13956             tpl : new Roo.Template(content)
13957         };
13958         if(name){
13959             st[name] = st[subIndex];
13960         }
13961         st[subIndex].tpl.compile();
13962         st[subIndex].tpl.call = this.call.createDelegate(this);
13963         subIndex++;
13964     }
13965     this.subCount = subIndex;
13966     this.subs = st;
13967 };
13968 Roo.extend(Roo.MasterTemplate, Roo.Template, {
13969     /**
13970     * The regular expression used to match sub templates
13971     * @type RegExp
13972     * @property
13973     */
13974     subTemplateRe : /<tpl(?:\sname="([\w-]+)")?>((?:.|\n)*?)<\/tpl>/gi,
13975
13976     /**
13977      * Applies the passed values to a child template.
13978      * @param {String/Number} name (optional) The name or index of the child template
13979      * @param {Array/Object} values The values to be applied to the template
13980      * @return {MasterTemplate} this
13981      */
13982      add : function(name, values){
13983         if(arguments.length == 1){
13984             values = arguments[0];
13985             name = 0;
13986         }
13987         var s = this.subs[name];
13988         s.buffer[s.buffer.length] = s.tpl.apply(values);
13989         return this;
13990     },
13991
13992     /**
13993      * Applies all the passed values to a child template.
13994      * @param {String/Number} name (optional) The name or index of the child template
13995      * @param {Array} values The values to be applied to the template, this should be an array of objects.
13996      * @param {Boolean} reset (optional) True to reset the template first
13997      * @return {MasterTemplate} this
13998      */
13999     fill : function(name, values, reset){
14000         var a = arguments;
14001         if(a.length == 1 || (a.length == 2 && typeof a[1] == "boolean")){
14002             values = a[0];
14003             name = 0;
14004             reset = a[1];
14005         }
14006         if(reset){
14007             this.reset();
14008         }
14009         for(var i = 0, len = values.length; i < len; i++){
14010             this.add(name, values[i]);
14011         }
14012         return this;
14013     },
14014
14015     /**
14016      * Resets the template for reuse
14017      * @return {MasterTemplate} this
14018      */
14019      reset : function(){
14020         var s = this.subs;
14021         for(var i = 0; i < this.subCount; i++){
14022             s[i].buffer = [];
14023         }
14024         return this;
14025     },
14026
14027     applyTemplate : function(values){
14028         var s = this.subs;
14029         var replaceIndex = -1;
14030         this.html = this.originalHtml.replace(this.subTemplateRe, function(m, name){
14031             return s[++replaceIndex].buffer.join("");
14032         });
14033         return Roo.MasterTemplate.superclass.applyTemplate.call(this, values);
14034     },
14035
14036     apply : function(){
14037         return this.applyTemplate.apply(this, arguments);
14038     },
14039
14040     compile : function(){return this;}
14041 });
14042
14043 /**
14044  * Alias for fill().
14045  * @method
14046  */
14047 Roo.MasterTemplate.prototype.addAll = Roo.MasterTemplate.prototype.fill;
14048  /**
14049  * Creates a template from the passed element's value (display:none textarea, preferred) or innerHTML. e.g.
14050  * var tpl = Roo.MasterTemplate.from('element-id');
14051  * @param {String/HTMLElement} el
14052  * @param {Object} config
14053  * @static
14054  */
14055 Roo.MasterTemplate.from = function(el, config){
14056     el = Roo.getDom(el);
14057     return new Roo.MasterTemplate(el.value || el.innerHTML, config || '');
14058 };/*
14059  * Based on:
14060  * Ext JS Library 1.1.1
14061  * Copyright(c) 2006-2007, Ext JS, LLC.
14062  *
14063  * Originally Released Under LGPL - original licence link has changed is not relivant.
14064  *
14065  * Fork - LGPL
14066  * <script type="text/javascript">
14067  */
14068
14069  
14070 /**
14071  * @class Roo.util.CSS
14072  * Utility class for manipulating CSS rules
14073  * @singleton
14074  */
14075 Roo.util.CSS = function(){
14076         var rules = null;
14077         var doc = document;
14078
14079     var camelRe = /(-[a-z])/gi;
14080     var camelFn = function(m, a){ return a.charAt(1).toUpperCase(); };
14081
14082    return {
14083    /**
14084     * Very simple dynamic creation of stylesheets from a text blob of rules.  The text will wrapped in a style
14085     * tag and appended to the HEAD of the document.
14086     * @param {String|Object} cssText The text containing the css rules
14087     * @param {String} id An id to add to the stylesheet for later removal
14088     * @return {StyleSheet}
14089     */
14090     createStyleSheet : function(cssText, id){
14091         var ss;
14092         var head = doc.getElementsByTagName("head")[0];
14093         var nrules = doc.createElement("style");
14094         nrules.setAttribute("type", "text/css");
14095         if(id){
14096             nrules.setAttribute("id", id);
14097         }
14098         if (typeof(cssText) != 'string') {
14099             // support object maps..
14100             // not sure if this a good idea.. 
14101             // perhaps it should be merged with the general css handling
14102             // and handle js style props.
14103             var cssTextNew = [];
14104             for(var n in cssText) {
14105                 var citems = [];
14106                 for(var k in cssText[n]) {
14107                     citems.push( k + ' : ' +cssText[n][k] + ';' );
14108                 }
14109                 cssTextNew.push( n + ' { ' + citems.join(' ') + '} ');
14110                 
14111             }
14112             cssText = cssTextNew.join("\n");
14113             
14114         }
14115        
14116        
14117        if(Roo.isIE){
14118            head.appendChild(nrules);
14119            ss = nrules.styleSheet;
14120            ss.cssText = cssText;
14121        }else{
14122            try{
14123                 nrules.appendChild(doc.createTextNode(cssText));
14124            }catch(e){
14125                nrules.cssText = cssText; 
14126            }
14127            head.appendChild(nrules);
14128            ss = nrules.styleSheet ? nrules.styleSheet : (nrules.sheet || doc.styleSheets[doc.styleSheets.length-1]);
14129        }
14130        this.cacheStyleSheet(ss);
14131        return ss;
14132    },
14133
14134    /**
14135     * Removes a style or link tag by id
14136     * @param {String} id The id of the tag
14137     */
14138    removeStyleSheet : function(id){
14139        var existing = doc.getElementById(id);
14140        if(existing){
14141            existing.parentNode.removeChild(existing);
14142        }
14143    },
14144
14145    /**
14146     * Dynamically swaps an existing stylesheet reference for a new one
14147     * @param {String} id The id of an existing link tag to remove
14148     * @param {String} url The href of the new stylesheet to include
14149     */
14150    swapStyleSheet : function(id, url){
14151        this.removeStyleSheet(id);
14152        var ss = doc.createElement("link");
14153        ss.setAttribute("rel", "stylesheet");
14154        ss.setAttribute("type", "text/css");
14155        ss.setAttribute("id", id);
14156        ss.setAttribute("href", url);
14157        doc.getElementsByTagName("head")[0].appendChild(ss);
14158    },
14159    
14160    /**
14161     * Refresh the rule cache if you have dynamically added stylesheets
14162     * @return {Object} An object (hash) of rules indexed by selector
14163     */
14164    refreshCache : function(){
14165        return this.getRules(true);
14166    },
14167
14168    // private
14169    cacheStyleSheet : function(stylesheet){
14170        if(!rules){
14171            rules = {};
14172        }
14173        try{// try catch for cross domain access issue
14174            var ssRules = stylesheet.cssRules || stylesheet.rules;
14175            for(var j = ssRules.length-1; j >= 0; --j){
14176                rules[ssRules[j].selectorText] = ssRules[j];
14177            }
14178        }catch(e){}
14179    },
14180    
14181    /**
14182     * Gets all css rules for the document
14183     * @param {Boolean} refreshCache true to refresh the internal cache
14184     * @return {Object} An object (hash) of rules indexed by selector
14185     */
14186    getRules : function(refreshCache){
14187                 if(rules == null || refreshCache){
14188                         rules = {};
14189                         var ds = doc.styleSheets;
14190                         for(var i =0, len = ds.length; i < len; i++){
14191                             try{
14192                         this.cacheStyleSheet(ds[i]);
14193                     }catch(e){} 
14194                 }
14195                 }
14196                 return rules;
14197         },
14198         
14199         /**
14200     * Gets an an individual CSS rule by selector(s)
14201     * @param {String/Array} selector The CSS selector or an array of selectors to try. The first selector that is found is returned.
14202     * @param {Boolean} refreshCache true to refresh the internal cache if you have recently updated any rules or added styles dynamically
14203     * @return {CSSRule} The CSS rule or null if one is not found
14204     */
14205    getRule : function(selector, refreshCache){
14206                 var rs = this.getRules(refreshCache);
14207                 if(!(selector instanceof Array)){
14208                     return rs[selector];
14209                 }
14210                 for(var i = 0; i < selector.length; i++){
14211                         if(rs[selector[i]]){
14212                                 return rs[selector[i]];
14213                         }
14214                 }
14215                 return null;
14216         },
14217         
14218         
14219         /**
14220     * Updates a rule property
14221     * @param {String/Array} selector If it's an array it tries each selector until it finds one. Stops immediately once one is found.
14222     * @param {String} property The css property
14223     * @param {String} value The new value for the property
14224     * @return {Boolean} true If a rule was found and updated
14225     */
14226    updateRule : function(selector, property, value){
14227                 if(!(selector instanceof Array)){
14228                         var rule = this.getRule(selector);
14229                         if(rule){
14230                                 rule.style[property.replace(camelRe, camelFn)] = value;
14231                                 return true;
14232                         }
14233                 }else{
14234                         for(var i = 0; i < selector.length; i++){
14235                                 if(this.updateRule(selector[i], property, value)){
14236                                         return true;
14237                                 }
14238                         }
14239                 }
14240                 return false;
14241         }
14242    };   
14243 }();/*
14244  * Based on:
14245  * Ext JS Library 1.1.1
14246  * Copyright(c) 2006-2007, Ext JS, LLC.
14247  *
14248  * Originally Released Under LGPL - original licence link has changed is not relivant.
14249  *
14250  * Fork - LGPL
14251  * <script type="text/javascript">
14252  */
14253
14254  
14255
14256 /**
14257  * @class Roo.util.ClickRepeater
14258  * @extends Roo.util.Observable
14259  * 
14260  * A wrapper class which can be applied to any element. Fires a "click" event while the
14261  * mouse is pressed. The interval between firings may be specified in the config but
14262  * defaults to 10 milliseconds.
14263  * 
14264  * Optionally, a CSS class may be applied to the element during the time it is pressed.
14265  * 
14266  * @cfg {String/HTMLElement/Element} el The element to act as a button.
14267  * @cfg {Number} delay The initial delay before the repeating event begins firing.
14268  * Similar to an autorepeat key delay.
14269  * @cfg {Number} interval The interval between firings of the "click" event. Default 10 ms.
14270  * @cfg {String} pressClass A CSS class name to be applied to the element while pressed.
14271  * @cfg {Boolean} accelerate True if autorepeating should start slowly and accelerate.
14272  *           "interval" and "delay" are ignored. "immediate" is honored.
14273  * @cfg {Boolean} preventDefault True to prevent the default click event
14274  * @cfg {Boolean} stopDefault True to stop the default click event
14275  * 
14276  * @history
14277  *     2007-02-02 jvs Original code contributed by Nige "Animal" White
14278  *     2007-02-02 jvs Renamed to ClickRepeater
14279  *   2007-02-03 jvs Modifications for FF Mac and Safari 
14280  *
14281  *  @constructor
14282  * @param {String/HTMLElement/Element} el The element to listen on
14283  * @param {Object} config
14284  **/
14285 Roo.util.ClickRepeater = function(el, config)
14286 {
14287     this.el = Roo.get(el);
14288     this.el.unselectable();
14289
14290     Roo.apply(this, config);
14291
14292     this.addEvents({
14293     /**
14294      * @event mousedown
14295      * Fires when the mouse button is depressed.
14296      * @param {Roo.util.ClickRepeater} this
14297      */
14298         "mousedown" : true,
14299     /**
14300      * @event click
14301      * Fires on a specified interval during the time the element is pressed.
14302      * @param {Roo.util.ClickRepeater} this
14303      */
14304         "click" : true,
14305     /**
14306      * @event mouseup
14307      * Fires when the mouse key is released.
14308      * @param {Roo.util.ClickRepeater} this
14309      */
14310         "mouseup" : true
14311     });
14312
14313     this.el.on("mousedown", this.handleMouseDown, this);
14314     if(this.preventDefault || this.stopDefault){
14315         this.el.on("click", function(e){
14316             if(this.preventDefault){
14317                 e.preventDefault();
14318             }
14319             if(this.stopDefault){
14320                 e.stopEvent();
14321             }
14322         }, this);
14323     }
14324
14325     // allow inline handler
14326     if(this.handler){
14327         this.on("click", this.handler,  this.scope || this);
14328     }
14329
14330     Roo.util.ClickRepeater.superclass.constructor.call(this);
14331 };
14332
14333 Roo.extend(Roo.util.ClickRepeater, Roo.util.Observable, {
14334     interval : 20,
14335     delay: 250,
14336     preventDefault : true,
14337     stopDefault : false,
14338     timer : 0,
14339
14340     // private
14341     handleMouseDown : function(){
14342         clearTimeout(this.timer);
14343         this.el.blur();
14344         if(this.pressClass){
14345             this.el.addClass(this.pressClass);
14346         }
14347         this.mousedownTime = new Date();
14348
14349         Roo.get(document).on("mouseup", this.handleMouseUp, this);
14350         this.el.on("mouseout", this.handleMouseOut, this);
14351
14352         this.fireEvent("mousedown", this);
14353         this.fireEvent("click", this);
14354         
14355         this.timer = this.click.defer(this.delay || this.interval, this);
14356     },
14357
14358     // private
14359     click : function(){
14360         this.fireEvent("click", this);
14361         this.timer = this.click.defer(this.getInterval(), this);
14362     },
14363
14364     // private
14365     getInterval: function(){
14366         if(!this.accelerate){
14367             return this.interval;
14368         }
14369         var pressTime = this.mousedownTime.getElapsed();
14370         if(pressTime < 500){
14371             return 400;
14372         }else if(pressTime < 1700){
14373             return 320;
14374         }else if(pressTime < 2600){
14375             return 250;
14376         }else if(pressTime < 3500){
14377             return 180;
14378         }else if(pressTime < 4400){
14379             return 140;
14380         }else if(pressTime < 5300){
14381             return 80;
14382         }else if(pressTime < 6200){
14383             return 50;
14384         }else{
14385             return 10;
14386         }
14387     },
14388
14389     // private
14390     handleMouseOut : function(){
14391         clearTimeout(this.timer);
14392         if(this.pressClass){
14393             this.el.removeClass(this.pressClass);
14394         }
14395         this.el.on("mouseover", this.handleMouseReturn, this);
14396     },
14397
14398     // private
14399     handleMouseReturn : function(){
14400         this.el.un("mouseover", this.handleMouseReturn);
14401         if(this.pressClass){
14402             this.el.addClass(this.pressClass);
14403         }
14404         this.click();
14405     },
14406
14407     // private
14408     handleMouseUp : function(){
14409         clearTimeout(this.timer);
14410         this.el.un("mouseover", this.handleMouseReturn);
14411         this.el.un("mouseout", this.handleMouseOut);
14412         Roo.get(document).un("mouseup", this.handleMouseUp);
14413         this.el.removeClass(this.pressClass);
14414         this.fireEvent("mouseup", this);
14415     }
14416 });/*
14417  * Based on:
14418  * Ext JS Library 1.1.1
14419  * Copyright(c) 2006-2007, Ext JS, LLC.
14420  *
14421  * Originally Released Under LGPL - original licence link has changed is not relivant.
14422  *
14423  * Fork - LGPL
14424  * <script type="text/javascript">
14425  */
14426
14427  
14428 /**
14429  * @class Roo.KeyNav
14430  * <p>Provides a convenient wrapper for normalized keyboard navigation.  KeyNav allows you to bind
14431  * navigation keys to function calls that will get called when the keys are pressed, providing an easy
14432  * way to implement custom navigation schemes for any UI component.</p>
14433  * <p>The following are all of the possible keys that can be implemented: enter, left, right, up, down, tab, esc,
14434  * pageUp, pageDown, del, home, end.  Usage:</p>
14435  <pre><code>
14436 var nav = new Roo.KeyNav("my-element", {
14437     "left" : function(e){
14438         this.moveLeft(e.ctrlKey);
14439     },
14440     "right" : function(e){
14441         this.moveRight(e.ctrlKey);
14442     },
14443     "enter" : function(e){
14444         this.save();
14445     },
14446     scope : this
14447 });
14448 </code></pre>
14449  * @constructor
14450  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14451  * @param {Object} config The config
14452  */
14453 Roo.KeyNav = function(el, config){
14454     this.el = Roo.get(el);
14455     Roo.apply(this, config);
14456     if(!this.disabled){
14457         this.disabled = true;
14458         this.enable();
14459     }
14460 };
14461
14462 Roo.KeyNav.prototype = {
14463     /**
14464      * @cfg {Boolean} disabled
14465      * True to disable this KeyNav instance (defaults to false)
14466      */
14467     disabled : false,
14468     /**
14469      * @cfg {String} defaultEventAction
14470      * The method to call on the {@link Roo.EventObject} after this KeyNav intercepts a key.  Valid values are
14471      * {@link Roo.EventObject#stopEvent}, {@link Roo.EventObject#preventDefault} and
14472      * {@link Roo.EventObject#stopPropagation} (defaults to 'stopEvent')
14473      */
14474     defaultEventAction: "stopEvent",
14475     /**
14476      * @cfg {Boolean} forceKeyDown
14477      * Handle the keydown event instead of keypress (defaults to false).  KeyNav automatically does this for IE since
14478      * IE does not propagate special keys on keypress, but setting this to true will force other browsers to also
14479      * handle keydown instead of keypress.
14480      */
14481     forceKeyDown : false,
14482
14483     // private
14484     prepareEvent : function(e){
14485         var k = e.getKey();
14486         var h = this.keyToHandler[k];
14487         //if(h && this[h]){
14488         //    e.stopPropagation();
14489         //}
14490         if(Roo.isSafari && h && k >= 37 && k <= 40){
14491             e.stopEvent();
14492         }
14493     },
14494
14495     // private
14496     relay : function(e){
14497         var k = e.getKey();
14498         var h = this.keyToHandler[k];
14499         if(h && this[h]){
14500             if(this.doRelay(e, this[h], h) !== true){
14501                 e[this.defaultEventAction]();
14502             }
14503         }
14504     },
14505
14506     // private
14507     doRelay : function(e, h, hname){
14508         return h.call(this.scope || this, e);
14509     },
14510
14511     // possible handlers
14512     enter : false,
14513     left : false,
14514     right : false,
14515     up : false,
14516     down : false,
14517     tab : false,
14518     esc : false,
14519     pageUp : false,
14520     pageDown : false,
14521     del : false,
14522     home : false,
14523     end : false,
14524
14525     // quick lookup hash
14526     keyToHandler : {
14527         37 : "left",
14528         39 : "right",
14529         38 : "up",
14530         40 : "down",
14531         33 : "pageUp",
14532         34 : "pageDown",
14533         46 : "del",
14534         36 : "home",
14535         35 : "end",
14536         13 : "enter",
14537         27 : "esc",
14538         9  : "tab"
14539     },
14540
14541         /**
14542          * Enable this KeyNav
14543          */
14544         enable: function(){
14545                 if(this.disabled){
14546             // ie won't do special keys on keypress, no one else will repeat keys with keydown
14547             // the EventObject will normalize Safari automatically
14548             if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14549                 this.el.on("keydown", this.relay,  this);
14550             }else{
14551                 this.el.on("keydown", this.prepareEvent,  this);
14552                 this.el.on("keypress", this.relay,  this);
14553             }
14554                     this.disabled = false;
14555                 }
14556         },
14557
14558         /**
14559          * Disable this KeyNav
14560          */
14561         disable: function(){
14562                 if(!this.disabled){
14563                     if(this.forceKeyDown || Roo.isIE || Roo.isAir){
14564                 this.el.un("keydown", this.relay);
14565             }else{
14566                 this.el.un("keydown", this.prepareEvent);
14567                 this.el.un("keypress", this.relay);
14568             }
14569                     this.disabled = true;
14570                 }
14571         }
14572 };/*
14573  * Based on:
14574  * Ext JS Library 1.1.1
14575  * Copyright(c) 2006-2007, Ext JS, LLC.
14576  *
14577  * Originally Released Under LGPL - original licence link has changed is not relivant.
14578  *
14579  * Fork - LGPL
14580  * <script type="text/javascript">
14581  */
14582
14583  
14584 /**
14585  * @class Roo.KeyMap
14586  * Handles mapping keys to actions for an element. One key map can be used for multiple actions.
14587  * The constructor accepts the same config object as defined by {@link #addBinding}.
14588  * If you bind a callback function to a KeyMap, anytime the KeyMap handles an expected key
14589  * combination it will call the function with this signature (if the match is a multi-key
14590  * combination the callback will still be called only once): (String key, Roo.EventObject e)
14591  * A KeyMap can also handle a string representation of keys.<br />
14592  * Usage:
14593  <pre><code>
14594 // map one key by key code
14595 var map = new Roo.KeyMap("my-element", {
14596     key: 13, // or Roo.EventObject.ENTER
14597     fn: myHandler,
14598     scope: myObject
14599 });
14600
14601 // map multiple keys to one action by string
14602 var map = new Roo.KeyMap("my-element", {
14603     key: "a\r\n\t",
14604     fn: myHandler,
14605     scope: myObject
14606 });
14607
14608 // map multiple keys to multiple actions by strings and array of codes
14609 var map = new Roo.KeyMap("my-element", [
14610     {
14611         key: [10,13],
14612         fn: function(){ alert("Return was pressed"); }
14613     }, {
14614         key: "abc",
14615         fn: function(){ alert('a, b or c was pressed'); }
14616     }, {
14617         key: "\t",
14618         ctrl:true,
14619         shift:true,
14620         fn: function(){ alert('Control + shift + tab was pressed.'); }
14621     }
14622 ]);
14623 </code></pre>
14624  * <b>Note: A KeyMap starts enabled</b>
14625  * @constructor
14626  * @param {String/HTMLElement/Roo.Element} el The element to bind to
14627  * @param {Object} config The config (see {@link #addBinding})
14628  * @param {String} eventName (optional) The event to bind to (defaults to "keydown")
14629  */
14630 Roo.KeyMap = function(el, config, eventName){
14631     this.el  = Roo.get(el);
14632     this.eventName = eventName || "keydown";
14633     this.bindings = [];
14634     if(config){
14635         this.addBinding(config);
14636     }
14637     this.enable();
14638 };
14639
14640 Roo.KeyMap.prototype = {
14641     /**
14642      * True to stop the event from bubbling and prevent the default browser action if the
14643      * key was handled by the KeyMap (defaults to false)
14644      * @type Boolean
14645      */
14646     stopEvent : false,
14647
14648     /**
14649      * Add a new binding to this KeyMap. The following config object properties are supported:
14650      * <pre>
14651 Property    Type             Description
14652 ----------  ---------------  ----------------------------------------------------------------------
14653 key         String/Array     A single keycode or an array of keycodes to handle
14654 shift       Boolean          True to handle key only when shift is pressed (defaults to false)
14655 ctrl        Boolean          True to handle key only when ctrl is pressed (defaults to false)
14656 alt         Boolean          True to handle key only when alt is pressed (defaults to false)
14657 fn          Function         The function to call when KeyMap finds the expected key combination
14658 scope       Object           The scope of the callback function
14659 </pre>
14660      *
14661      * Usage:
14662      * <pre><code>
14663 // Create a KeyMap
14664 var map = new Roo.KeyMap(document, {
14665     key: Roo.EventObject.ENTER,
14666     fn: handleKey,
14667     scope: this
14668 });
14669
14670 //Add a new binding to the existing KeyMap later
14671 map.addBinding({
14672     key: 'abc',
14673     shift: true,
14674     fn: handleKey,
14675     scope: this
14676 });
14677 </code></pre>
14678      * @param {Object/Array} config A single KeyMap config or an array of configs
14679      */
14680         addBinding : function(config){
14681         if(config instanceof Array){
14682             for(var i = 0, len = config.length; i < len; i++){
14683                 this.addBinding(config[i]);
14684             }
14685             return;
14686         }
14687         var keyCode = config.key,
14688             shift = config.shift, 
14689             ctrl = config.ctrl, 
14690             alt = config.alt,
14691             fn = config.fn,
14692             scope = config.scope;
14693         if(typeof keyCode == "string"){
14694             var ks = [];
14695             var keyString = keyCode.toUpperCase();
14696             for(var j = 0, len = keyString.length; j < len; j++){
14697                 ks.push(keyString.charCodeAt(j));
14698             }
14699             keyCode = ks;
14700         }
14701         var keyArray = keyCode instanceof Array;
14702         var handler = function(e){
14703             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
14704                 var k = e.getKey();
14705                 if(keyArray){
14706                     for(var i = 0, len = keyCode.length; i < len; i++){
14707                         if(keyCode[i] == k){
14708                           if(this.stopEvent){
14709                               e.stopEvent();
14710                           }
14711                           fn.call(scope || window, k, e);
14712                           return;
14713                         }
14714                     }
14715                 }else{
14716                     if(k == keyCode){
14717                         if(this.stopEvent){
14718                            e.stopEvent();
14719                         }
14720                         fn.call(scope || window, k, e);
14721                     }
14722                 }
14723             }
14724         };
14725         this.bindings.push(handler);  
14726         },
14727
14728     /**
14729      * Shorthand for adding a single key listener
14730      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the
14731      * following options:
14732      * {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
14733      * @param {Function} fn The function to call
14734      * @param {Object} scope (optional) The scope of the function
14735      */
14736     on : function(key, fn, scope){
14737         var keyCode, shift, ctrl, alt;
14738         if(typeof key == "object" && !(key instanceof Array)){
14739             keyCode = key.key;
14740             shift = key.shift;
14741             ctrl = key.ctrl;
14742             alt = key.alt;
14743         }else{
14744             keyCode = key;
14745         }
14746         this.addBinding({
14747             key: keyCode,
14748             shift: shift,
14749             ctrl: ctrl,
14750             alt: alt,
14751             fn: fn,
14752             scope: scope
14753         })
14754     },
14755
14756     // private
14757     handleKeyDown : function(e){
14758             if(this.enabled){ //just in case
14759             var b = this.bindings;
14760             for(var i = 0, len = b.length; i < len; i++){
14761                 b[i].call(this, e);
14762             }
14763             }
14764         },
14765         
14766         /**
14767          * Returns true if this KeyMap is enabled
14768          * @return {Boolean} 
14769          */
14770         isEnabled : function(){
14771             return this.enabled;  
14772         },
14773         
14774         /**
14775          * Enables this KeyMap
14776          */
14777         enable: function(){
14778                 if(!this.enabled){
14779                     this.el.on(this.eventName, this.handleKeyDown, this);
14780                     this.enabled = true;
14781                 }
14782         },
14783
14784         /**
14785          * Disable this KeyMap
14786          */
14787         disable: function(){
14788                 if(this.enabled){
14789                     this.el.removeListener(this.eventName, this.handleKeyDown, this);
14790                     this.enabled = false;
14791                 }
14792         }
14793 };/*
14794  * Based on:
14795  * Ext JS Library 1.1.1
14796  * Copyright(c) 2006-2007, Ext JS, LLC.
14797  *
14798  * Originally Released Under LGPL - original licence link has changed is not relivant.
14799  *
14800  * Fork - LGPL
14801  * <script type="text/javascript">
14802  */
14803
14804  
14805 /**
14806  * @class Roo.util.TextMetrics
14807  * Provides precise pixel measurements for blocks of text so that you can determine exactly how high and
14808  * wide, in pixels, a given block of text will be.
14809  * @singleton
14810  */
14811 Roo.util.TextMetrics = function(){
14812     var shared;
14813     return {
14814         /**
14815          * Measures the size of the specified text
14816          * @param {String/HTMLElement} el The element, dom node or id from which to copy existing CSS styles
14817          * that can affect the size of the rendered text
14818          * @param {String} text The text to measure
14819          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14820          * in order to accurately measure the text height
14821          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14822          */
14823         measure : function(el, text, fixedWidth){
14824             if(!shared){
14825                 shared = Roo.util.TextMetrics.Instance(el, fixedWidth);
14826             }
14827             shared.bind(el);
14828             shared.setFixedWidth(fixedWidth || 'auto');
14829             return shared.getSize(text);
14830         },
14831
14832         /**
14833          * Return a unique TextMetrics instance that can be bound directly to an element and reused.  This reduces
14834          * the overhead of multiple calls to initialize the style properties on each measurement.
14835          * @param {String/HTMLElement} el The element, dom node or id that the instance will be bound to
14836          * @param {Number} fixedWidth (optional) If the text will be multiline, you have to set a fixed width
14837          * in order to accurately measure the text height
14838          * @return {Roo.util.TextMetrics.Instance} instance The new instance
14839          */
14840         createInstance : function(el, fixedWidth){
14841             return Roo.util.TextMetrics.Instance(el, fixedWidth);
14842         }
14843     };
14844 }();
14845
14846  
14847
14848 Roo.util.TextMetrics.Instance = function(bindTo, fixedWidth){
14849     var ml = new Roo.Element(document.createElement('div'));
14850     document.body.appendChild(ml.dom);
14851     ml.position('absolute');
14852     ml.setLeftTop(-1000, -1000);
14853     ml.hide();
14854
14855     if(fixedWidth){
14856         ml.setWidth(fixedWidth);
14857     }
14858      
14859     var instance = {
14860         /**
14861          * Returns the size of the specified text based on the internal element's style and width properties
14862          * @memberOf Roo.util.TextMetrics.Instance#
14863          * @param {String} text The text to measure
14864          * @return {Object} An object containing the text's size {width: (width), height: (height)}
14865          */
14866         getSize : function(text){
14867             ml.update(text);
14868             var s = ml.getSize();
14869             ml.update('');
14870             return s;
14871         },
14872
14873         /**
14874          * Binds this TextMetrics instance to an element from which to copy existing CSS styles
14875          * that can affect the size of the rendered text
14876          * @memberOf Roo.util.TextMetrics.Instance#
14877          * @param {String/HTMLElement} el The element, dom node or id
14878          */
14879         bind : function(el){
14880             ml.setStyle(
14881                 Roo.fly(el).getStyles('font-size','font-style', 'font-weight', 'font-family','line-height')
14882             );
14883         },
14884
14885         /**
14886          * Sets a fixed width on the internal measurement element.  If the text will be multiline, you have
14887          * to set a fixed width in order to accurately measure the text height.
14888          * @memberOf Roo.util.TextMetrics.Instance#
14889          * @param {Number} width The width to set on the element
14890          */
14891         setFixedWidth : function(width){
14892             ml.setWidth(width);
14893         },
14894
14895         /**
14896          * Returns the measured width of the specified text
14897          * @memberOf Roo.util.TextMetrics.Instance#
14898          * @param {String} text The text to measure
14899          * @return {Number} width The width in pixels
14900          */
14901         getWidth : function(text){
14902             ml.dom.style.width = 'auto';
14903             return this.getSize(text).width;
14904         },
14905
14906         /**
14907          * Returns the measured height of the specified text.  For multiline text, be sure to call
14908          * {@link #setFixedWidth} if necessary.
14909          * @memberOf Roo.util.TextMetrics.Instance#
14910          * @param {String} text The text to measure
14911          * @return {Number} height The height in pixels
14912          */
14913         getHeight : function(text){
14914             return this.getSize(text).height;
14915         }
14916     };
14917
14918     instance.bind(bindTo);
14919
14920     return instance;
14921 };
14922
14923 // backwards compat
14924 Roo.Element.measureText = Roo.util.TextMetrics.measure;/*
14925  * Based on:
14926  * Ext JS Library 1.1.1
14927  * Copyright(c) 2006-2007, Ext JS, LLC.
14928  *
14929  * Originally Released Under LGPL - original licence link has changed is not relivant.
14930  *
14931  * Fork - LGPL
14932  * <script type="text/javascript">
14933  */
14934
14935 /**
14936  * @class Roo.state.Provider
14937  * Abstract base class for state provider implementations. This class provides methods
14938  * for encoding and decoding <b>typed</b> variables including dates and defines the 
14939  * Provider interface.
14940  */
14941 Roo.state.Provider = function(){
14942     /**
14943      * @event statechange
14944      * Fires when a state change occurs.
14945      * @param {Provider} this This state provider
14946      * @param {String} key The state key which was changed
14947      * @param {String} value The encoded value for the state
14948      */
14949     this.addEvents({
14950         "statechange": true
14951     });
14952     this.state = {};
14953     Roo.state.Provider.superclass.constructor.call(this);
14954 };
14955 Roo.extend(Roo.state.Provider, Roo.util.Observable, {
14956     /**
14957      * Returns the current value for a key
14958      * @param {String} name The key name
14959      * @param {Mixed} defaultValue A default value to return if the key's value is not found
14960      * @return {Mixed} The state data
14961      */
14962     get : function(name, defaultValue){
14963         return typeof this.state[name] == "undefined" ?
14964             defaultValue : this.state[name];
14965     },
14966     
14967     /**
14968      * Clears a value from the state
14969      * @param {String} name The key name
14970      */
14971     clear : function(name){
14972         delete this.state[name];
14973         this.fireEvent("statechange", this, name, null);
14974     },
14975     
14976     /**
14977      * Sets the value for a key
14978      * @param {String} name The key name
14979      * @param {Mixed} value The value to set
14980      */
14981     set : function(name, value){
14982         this.state[name] = value;
14983         this.fireEvent("statechange", this, name, value);
14984     },
14985     
14986     /**
14987      * Decodes a string previously encoded with {@link #encodeValue}.
14988      * @param {String} value The value to decode
14989      * @return {Mixed} The decoded value
14990      */
14991     decodeValue : function(cookie){
14992         var re = /^(a|n|d|b|s|o)\:(.*)$/;
14993         var matches = re.exec(unescape(cookie));
14994         if(!matches || !matches[1]) {
14995             return; // non state cookie
14996         }
14997         var type = matches[1];
14998         var v = matches[2];
14999         switch(type){
15000             case "n":
15001                 return parseFloat(v);
15002             case "d":
15003                 return new Date(Date.parse(v));
15004             case "b":
15005                 return (v == "1");
15006             case "a":
15007                 var all = [];
15008                 var values = v.split("^");
15009                 for(var i = 0, len = values.length; i < len; i++){
15010                     all.push(this.decodeValue(values[i]));
15011                 }
15012                 return all;
15013            case "o":
15014                 var all = {};
15015                 var values = v.split("^");
15016                 for(var i = 0, len = values.length; i < len; i++){
15017                     var kv = values[i].split("=");
15018                     all[kv[0]] = this.decodeValue(kv[1]);
15019                 }
15020                 return all;
15021            default:
15022                 return v;
15023         }
15024     },
15025     
15026     /**
15027      * Encodes a value including type information.  Decode with {@link #decodeValue}.
15028      * @param {Mixed} value The value to encode
15029      * @return {String} The encoded value
15030      */
15031     encodeValue : function(v){
15032         var enc;
15033         if(typeof v == "number"){
15034             enc = "n:" + v;
15035         }else if(typeof v == "boolean"){
15036             enc = "b:" + (v ? "1" : "0");
15037         }else if(v instanceof Date){
15038             enc = "d:" + v.toGMTString();
15039         }else if(v instanceof Array){
15040             var flat = "";
15041             for(var i = 0, len = v.length; i < len; i++){
15042                 flat += this.encodeValue(v[i]);
15043                 if(i != len-1) {
15044                     flat += "^";
15045                 }
15046             }
15047             enc = "a:" + flat;
15048         }else if(typeof v == "object"){
15049             var flat = "";
15050             for(var key in v){
15051                 if(typeof v[key] != "function"){
15052                     flat += key + "=" + this.encodeValue(v[key]) + "^";
15053                 }
15054             }
15055             enc = "o:" + flat.substring(0, flat.length-1);
15056         }else{
15057             enc = "s:" + v;
15058         }
15059         return escape(enc);        
15060     }
15061 });
15062
15063 /*
15064  * Based on:
15065  * Ext JS Library 1.1.1
15066  * Copyright(c) 2006-2007, Ext JS, LLC.
15067  *
15068  * Originally Released Under LGPL - original licence link has changed is not relivant.
15069  *
15070  * Fork - LGPL
15071  * <script type="text/javascript">
15072  */
15073 /**
15074  * @class Roo.state.Manager
15075  * This is the global state manager. By default all components that are "state aware" check this class
15076  * for state information if you don't pass them a custom state provider. In order for this class
15077  * to be useful, it must be initialized with a provider when your application initializes.
15078  <pre><code>
15079 // in your initialization function
15080 init : function(){
15081    Roo.state.Manager.setProvider(new Roo.state.CookieProvider());
15082    ...
15083    // supposed you have a {@link Roo.BorderLayout}
15084    var layout = new Roo.BorderLayout(...);
15085    layout.restoreState();
15086    // or a {Roo.BasicDialog}
15087    var dialog = new Roo.BasicDialog(...);
15088    dialog.restoreState();
15089  </code></pre>
15090  * @singleton
15091  */
15092 Roo.state.Manager = function(){
15093     var provider = new Roo.state.Provider();
15094     
15095     return {
15096         /**
15097          * Configures the default state provider for your application
15098          * @param {Provider} stateProvider The state provider to set
15099          */
15100         setProvider : function(stateProvider){
15101             provider = stateProvider;
15102         },
15103         
15104         /**
15105          * Returns the current value for a key
15106          * @param {String} name The key name
15107          * @param {Mixed} defaultValue The default value to return if the key lookup does not match
15108          * @return {Mixed} The state data
15109          */
15110         get : function(key, defaultValue){
15111             return provider.get(key, defaultValue);
15112         },
15113         
15114         /**
15115          * Sets the value for a key
15116          * @param {String} name The key name
15117          * @param {Mixed} value The state data
15118          */
15119          set : function(key, value){
15120             provider.set(key, value);
15121         },
15122         
15123         /**
15124          * Clears a value from the state
15125          * @param {String} name The key name
15126          */
15127         clear : function(key){
15128             provider.clear(key);
15129         },
15130         
15131         /**
15132          * Gets the currently configured state provider
15133          * @return {Provider} The state provider
15134          */
15135         getProvider : function(){
15136             return provider;
15137         }
15138     };
15139 }();
15140 /*
15141  * Based on:
15142  * Ext JS Library 1.1.1
15143  * Copyright(c) 2006-2007, Ext JS, LLC.
15144  *
15145  * Originally Released Under LGPL - original licence link has changed is not relivant.
15146  *
15147  * Fork - LGPL
15148  * <script type="text/javascript">
15149  */
15150 /**
15151  * @class Roo.state.CookieProvider
15152  * @extends Roo.state.Provider
15153  * The default Provider implementation which saves state via cookies.
15154  * <br />Usage:
15155  <pre><code>
15156    var cp = new Roo.state.CookieProvider({
15157        path: "/cgi-bin/",
15158        expires: new Date(new Date().getTime()+(1000*60*60*24*30)); //30 days
15159        domain: "roojs.com"
15160    })
15161    Roo.state.Manager.setProvider(cp);
15162  </code></pre>
15163  * @cfg {String} path The path for which the cookie is active (defaults to root '/' which makes it active for all pages in the site)
15164  * @cfg {Date} expires The cookie expiration date (defaults to 7 days from now)
15165  * @cfg {String} domain The domain to save the cookie for.  Note that you cannot specify a different domain than
15166  * your page is on, but you can specify a sub-domain, or simply the domain itself like 'roojs.com' to include
15167  * all sub-domains if you need to access cookies across different sub-domains (defaults to null which uses the same
15168  * domain the page is running on including the 'www' like 'www.roojs.com')
15169  * @cfg {Boolean} secure True if the site is using SSL (defaults to false)
15170  * @constructor
15171  * Create a new CookieProvider
15172  * @param {Object} config The configuration object
15173  */
15174 Roo.state.CookieProvider = function(config){
15175     Roo.state.CookieProvider.superclass.constructor.call(this);
15176     this.path = "/";
15177     this.expires = new Date(new Date().getTime()+(1000*60*60*24*7)); //7 days
15178     this.domain = null;
15179     this.secure = false;
15180     Roo.apply(this, config);
15181     this.state = this.readCookies();
15182 };
15183
15184 Roo.extend(Roo.state.CookieProvider, Roo.state.Provider, {
15185     // private
15186     set : function(name, value){
15187         if(typeof value == "undefined" || value === null){
15188             this.clear(name);
15189             return;
15190         }
15191         this.setCookie(name, value);
15192         Roo.state.CookieProvider.superclass.set.call(this, name, value);
15193     },
15194
15195     // private
15196     clear : function(name){
15197         this.clearCookie(name);
15198         Roo.state.CookieProvider.superclass.clear.call(this, name);
15199     },
15200
15201     // private
15202     readCookies : function(){
15203         var cookies = {};
15204         var c = document.cookie + ";";
15205         var re = /\s?(.*?)=(.*?);/g;
15206         var matches;
15207         while((matches = re.exec(c)) != null){
15208             var name = matches[1];
15209             var value = matches[2];
15210             if(name && name.substring(0,3) == "ys-"){
15211                 cookies[name.substr(3)] = this.decodeValue(value);
15212             }
15213         }
15214         return cookies;
15215     },
15216
15217     // private
15218     setCookie : function(name, value){
15219         document.cookie = "ys-"+ name + "=" + this.encodeValue(value) +
15220            ((this.expires == null) ? "" : ("; expires=" + this.expires.toGMTString())) +
15221            ((this.path == null) ? "" : ("; path=" + this.path)) +
15222            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15223            ((this.secure == true) ? "; secure" : "");
15224     },
15225
15226     // private
15227     clearCookie : function(name){
15228         document.cookie = "ys-" + name + "=null; expires=Thu, 01-Jan-70 00:00:01 GMT" +
15229            ((this.path == null) ? "" : ("; path=" + this.path)) +
15230            ((this.domain == null) ? "" : ("; domain=" + this.domain)) +
15231            ((this.secure == true) ? "; secure" : "");
15232     }
15233 });/*
15234  * Based on:
15235  * Ext JS Library 1.1.1
15236  * Copyright(c) 2006-2007, Ext JS, LLC.
15237  *
15238  * Originally Released Under LGPL - original licence link has changed is not relivant.
15239  *
15240  * Fork - LGPL
15241  * <script type="text/javascript">
15242  */
15243  
15244
15245 /**
15246  * @class Roo.ComponentMgr
15247  * Provides a common registry of all components on a page so that they can be easily accessed by component id (see {@link Roo.getCmp}).
15248  * @singleton
15249  */
15250 Roo.ComponentMgr = function(){
15251     var all = new Roo.util.MixedCollection();
15252
15253     return {
15254         /**
15255          * Registers a component.
15256          * @param {Roo.Component} c The component
15257          */
15258         register : function(c){
15259             all.add(c);
15260         },
15261
15262         /**
15263          * Unregisters a component.
15264          * @param {Roo.Component} c The component
15265          */
15266         unregister : function(c){
15267             all.remove(c);
15268         },
15269
15270         /**
15271          * Returns a component by id
15272          * @param {String} id The component id
15273          */
15274         get : function(id){
15275             return all.get(id);
15276         },
15277
15278         /**
15279          * Registers a function that will be called when a specified component is added to ComponentMgr
15280          * @param {String} id The component id
15281          * @param {Funtction} fn The callback function
15282          * @param {Object} scope The scope of the callback
15283          */
15284         onAvailable : function(id, fn, scope){
15285             all.on("add", function(index, o){
15286                 if(o.id == id){
15287                     fn.call(scope || o, o);
15288                     all.un("add", fn, scope);
15289                 }
15290             });
15291         }
15292     };
15293 }();/*
15294  * Based on:
15295  * Ext JS Library 1.1.1
15296  * Copyright(c) 2006-2007, Ext JS, LLC.
15297  *
15298  * Originally Released Under LGPL - original licence link has changed is not relivant.
15299  *
15300  * Fork - LGPL
15301  * <script type="text/javascript">
15302  */
15303  
15304 /**
15305  * @class Roo.Component
15306  * @extends Roo.util.Observable
15307  * Base class for all major Roo components.  All subclasses of Component can automatically participate in the standard
15308  * Roo component lifecycle of creation, rendering and destruction.  They also have automatic support for basic hide/show
15309  * and enable/disable behavior.  Component allows any subclass to be lazy-rendered into any {@link Roo.Container} and
15310  * to be automatically registered with the {@link Roo.ComponentMgr} so that it can be referenced at any time via {@link Roo.getCmp}.
15311  * All visual components (widgets) that require rendering into a layout should subclass Component.
15312  * @constructor
15313  * @param {Roo.Element/String/Object} config The configuration options.  If an element is passed, it is set as the internal
15314  * 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
15315  * and is used as the component id.  Otherwise, it is assumed to be a standard config object and is applied to the component.
15316  */
15317 Roo.Component = function(config){
15318     config = config || {};
15319     if(config.tagName || config.dom || typeof config == "string"){ // element object
15320         config = {el: config, id: config.id || config};
15321     }
15322     this.initialConfig = config;
15323
15324     Roo.apply(this, config);
15325     this.addEvents({
15326         /**
15327          * @event disable
15328          * Fires after the component is disabled.
15329              * @param {Roo.Component} this
15330              */
15331         disable : true,
15332         /**
15333          * @event enable
15334          * Fires after the component is enabled.
15335              * @param {Roo.Component} this
15336              */
15337         enable : true,
15338         /**
15339          * @event beforeshow
15340          * Fires before the component is shown.  Return false to stop the show.
15341              * @param {Roo.Component} this
15342              */
15343         beforeshow : true,
15344         /**
15345          * @event show
15346          * Fires after the component is shown.
15347              * @param {Roo.Component} this
15348              */
15349         show : true,
15350         /**
15351          * @event beforehide
15352          * Fires before the component is hidden. Return false to stop the hide.
15353              * @param {Roo.Component} this
15354              */
15355         beforehide : true,
15356         /**
15357          * @event hide
15358          * Fires after the component is hidden.
15359              * @param {Roo.Component} this
15360              */
15361         hide : true,
15362         /**
15363          * @event beforerender
15364          * Fires before the component is rendered. Return false to stop the render.
15365              * @param {Roo.Component} this
15366              */
15367         beforerender : true,
15368         /**
15369          * @event render
15370          * Fires after the component is rendered.
15371              * @param {Roo.Component} this
15372              */
15373         render : true,
15374         /**
15375          * @event beforedestroy
15376          * Fires before the component is destroyed. Return false to stop the destroy.
15377              * @param {Roo.Component} this
15378              */
15379         beforedestroy : true,
15380         /**
15381          * @event destroy
15382          * Fires after the component is destroyed.
15383              * @param {Roo.Component} this
15384              */
15385         destroy : true
15386     });
15387     if(!this.id){
15388         this.id = "roo-comp-" + (++Roo.Component.AUTO_ID);
15389     }
15390     Roo.ComponentMgr.register(this);
15391     Roo.Component.superclass.constructor.call(this);
15392     this.initComponent();
15393     if(this.renderTo){ // not supported by all components yet. use at your own risk!
15394         this.render(this.renderTo);
15395         delete this.renderTo;
15396     }
15397 };
15398
15399 /** @private */
15400 Roo.Component.AUTO_ID = 1000;
15401
15402 Roo.extend(Roo.Component, Roo.util.Observable, {
15403     /**
15404      * @scope Roo.Component.prototype
15405      * @type {Boolean}
15406      * true if this component is hidden. Read-only.
15407      */
15408     hidden : false,
15409     /**
15410      * @type {Boolean}
15411      * true if this component is disabled. Read-only.
15412      */
15413     disabled : false,
15414     /**
15415      * @type {Boolean}
15416      * true if this component has been rendered. Read-only.
15417      */
15418     rendered : false,
15419     
15420     /** @cfg {String} disableClass
15421      * CSS class added to the component when it is disabled (defaults to "x-item-disabled").
15422      */
15423     disabledClass : "x-item-disabled",
15424         /** @cfg {Boolean} allowDomMove
15425          * Whether the component can move the Dom node when rendering (defaults to true).
15426          */
15427     allowDomMove : true,
15428     /** @cfg {String} hideMode (display|visibility)
15429      * How this component should hidden. Supported values are
15430      * "visibility" (css visibility), "offsets" (negative offset position) and
15431      * "display" (css display) - defaults to "display".
15432      */
15433     hideMode: 'display',
15434
15435     /** @private */
15436     ctype : "Roo.Component",
15437
15438     /**
15439      * @cfg {String} actionMode 
15440      * which property holds the element that used for  hide() / show() / disable() / enable()
15441      * default is 'el' 
15442      */
15443     actionMode : "el",
15444
15445     /** @private */
15446     getActionEl : function(){
15447         return this[this.actionMode];
15448     },
15449
15450     initComponent : Roo.emptyFn,
15451     /**
15452      * If this is a lazy rendering component, render it to its container element.
15453      * @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.
15454      */
15455     render : function(container, position){
15456         
15457         if(this.rendered){
15458             return this;
15459         }
15460         
15461         if(this.fireEvent("beforerender", this) === false){
15462             return false;
15463         }
15464         
15465         if(!container && this.el){
15466             this.el = Roo.get(this.el);
15467             container = this.el.dom.parentNode;
15468             this.allowDomMove = false;
15469         }
15470         this.container = Roo.get(container);
15471         this.rendered = true;
15472         if(position !== undefined){
15473             if(typeof position == 'number'){
15474                 position = this.container.dom.childNodes[position];
15475             }else{
15476                 position = Roo.getDom(position);
15477             }
15478         }
15479         this.onRender(this.container, position || null);
15480         if(this.cls){
15481             this.el.addClass(this.cls);
15482             delete this.cls;
15483         }
15484         if(this.style){
15485             this.el.applyStyles(this.style);
15486             delete this.style;
15487         }
15488         this.fireEvent("render", this);
15489         this.afterRender(this.container);
15490         if(this.hidden){
15491             this.hide();
15492         }
15493         if(this.disabled){
15494             this.disable();
15495         }
15496
15497         return this;
15498         
15499     },
15500
15501     /** @private */
15502     // default function is not really useful
15503     onRender : function(ct, position){
15504         if(this.el){
15505             this.el = Roo.get(this.el);
15506             if(this.allowDomMove !== false){
15507                 ct.dom.insertBefore(this.el.dom, position);
15508             }
15509         }
15510     },
15511
15512     /** @private */
15513     getAutoCreate : function(){
15514         var cfg = typeof this.autoCreate == "object" ?
15515                       this.autoCreate : Roo.apply({}, this.defaultAutoCreate);
15516         if(this.id && !cfg.id){
15517             cfg.id = this.id;
15518         }
15519         return cfg;
15520     },
15521
15522     /** @private */
15523     afterRender : Roo.emptyFn,
15524
15525     /**
15526      * Destroys this component by purging any event listeners, removing the component's element from the DOM,
15527      * removing the component from its {@link Roo.Container} (if applicable) and unregistering it from {@link Roo.ComponentMgr}.
15528      */
15529     destroy : function(){
15530         if(this.fireEvent("beforedestroy", this) !== false){
15531             this.purgeListeners();
15532             this.beforeDestroy();
15533             if(this.rendered){
15534                 this.el.removeAllListeners();
15535                 this.el.remove();
15536                 if(this.actionMode == "container"){
15537                     this.container.remove();
15538                 }
15539             }
15540             this.onDestroy();
15541             Roo.ComponentMgr.unregister(this);
15542             this.fireEvent("destroy", this);
15543         }
15544     },
15545
15546         /** @private */
15547     beforeDestroy : function(){
15548
15549     },
15550
15551         /** @private */
15552         onDestroy : function(){
15553
15554     },
15555
15556     /**
15557      * Returns the underlying {@link Roo.Element}.
15558      * @return {Roo.Element} The element
15559      */
15560     getEl : function(){
15561         return this.el;
15562     },
15563
15564     /**
15565      * Returns the id of this component.
15566      * @return {String}
15567      */
15568     getId : function(){
15569         return this.id;
15570     },
15571
15572     /**
15573      * Try to focus this component.
15574      * @param {Boolean} selectText True to also select the text in this component (if applicable)
15575      * @return {Roo.Component} this
15576      */
15577     focus : function(selectText){
15578         if(this.rendered){
15579             this.el.focus();
15580             if(selectText === true){
15581                 this.el.dom.select();
15582             }
15583         }
15584         return this;
15585     },
15586
15587     /** @private */
15588     blur : function(){
15589         if(this.rendered){
15590             this.el.blur();
15591         }
15592         return this;
15593     },
15594
15595     /**
15596      * Disable this component.
15597      * @return {Roo.Component} this
15598      */
15599     disable : function(){
15600         if(this.rendered){
15601             this.onDisable();
15602         }
15603         this.disabled = true;
15604         this.fireEvent("disable", this);
15605         return this;
15606     },
15607
15608         // private
15609     onDisable : function(){
15610         this.getActionEl().addClass(this.disabledClass);
15611         this.el.dom.disabled = true;
15612     },
15613
15614     /**
15615      * Enable this component.
15616      * @return {Roo.Component} this
15617      */
15618     enable : function(){
15619         if(this.rendered){
15620             this.onEnable();
15621         }
15622         this.disabled = false;
15623         this.fireEvent("enable", this);
15624         return this;
15625     },
15626
15627         // private
15628     onEnable : function(){
15629         this.getActionEl().removeClass(this.disabledClass);
15630         this.el.dom.disabled = false;
15631     },
15632
15633     /**
15634      * Convenience function for setting disabled/enabled by boolean.
15635      * @param {Boolean} disabled
15636      */
15637     setDisabled : function(disabled){
15638         this[disabled ? "disable" : "enable"]();
15639     },
15640
15641     /**
15642      * Show this component.
15643      * @return {Roo.Component} this
15644      */
15645     show: function(){
15646         if(this.fireEvent("beforeshow", this) !== false){
15647             this.hidden = false;
15648             if(this.rendered){
15649                 this.onShow();
15650             }
15651             this.fireEvent("show", this);
15652         }
15653         return this;
15654     },
15655
15656     // private
15657     onShow : function(){
15658         var ae = this.getActionEl();
15659         if(this.hideMode == 'visibility'){
15660             ae.dom.style.visibility = "visible";
15661         }else if(this.hideMode == 'offsets'){
15662             ae.removeClass('x-hidden');
15663         }else{
15664             ae.dom.style.display = "";
15665         }
15666     },
15667
15668     /**
15669      * Hide this component.
15670      * @return {Roo.Component} this
15671      */
15672     hide: function(){
15673         if(this.fireEvent("beforehide", this) !== false){
15674             this.hidden = true;
15675             if(this.rendered){
15676                 this.onHide();
15677             }
15678             this.fireEvent("hide", this);
15679         }
15680         return this;
15681     },
15682
15683     // private
15684     onHide : function(){
15685         var ae = this.getActionEl();
15686         if(this.hideMode == 'visibility'){
15687             ae.dom.style.visibility = "hidden";
15688         }else if(this.hideMode == 'offsets'){
15689             ae.addClass('x-hidden');
15690         }else{
15691             ae.dom.style.display = "none";
15692         }
15693     },
15694
15695     /**
15696      * Convenience function to hide or show this component by boolean.
15697      * @param {Boolean} visible True to show, false to hide
15698      * @return {Roo.Component} this
15699      */
15700     setVisible: function(visible){
15701         if(visible) {
15702             this.show();
15703         }else{
15704             this.hide();
15705         }
15706         return this;
15707     },
15708
15709     /**
15710      * Returns true if this component is visible.
15711      */
15712     isVisible : function(){
15713         return this.getActionEl().isVisible();
15714     },
15715
15716     cloneConfig : function(overrides){
15717         overrides = overrides || {};
15718         var id = overrides.id || Roo.id();
15719         var cfg = Roo.applyIf(overrides, this.initialConfig);
15720         cfg.id = id; // prevent dup id
15721         return new this.constructor(cfg);
15722     }
15723 });/*
15724  * Based on:
15725  * Ext JS Library 1.1.1
15726  * Copyright(c) 2006-2007, Ext JS, LLC.
15727  *
15728  * Originally Released Under LGPL - original licence link has changed is not relivant.
15729  *
15730  * Fork - LGPL
15731  * <script type="text/javascript">
15732  */
15733
15734 /**
15735  * @class Roo.BoxComponent
15736  * @extends Roo.Component
15737  * Base class for any visual {@link Roo.Component} that uses a box container.  BoxComponent provides automatic box
15738  * model adjustments for sizing and positioning and will work correctly withnin the Component rendering model.  All
15739  * container classes should subclass BoxComponent so that they will work consistently when nested within other Ext
15740  * layout containers.
15741  * @constructor
15742  * @param {Roo.Element/String/Object} config The configuration options.
15743  */
15744 Roo.BoxComponent = function(config){
15745     Roo.Component.call(this, config);
15746     this.addEvents({
15747         /**
15748          * @event resize
15749          * Fires after the component is resized.
15750              * @param {Roo.Component} this
15751              * @param {Number} adjWidth The box-adjusted width that was set
15752              * @param {Number} adjHeight The box-adjusted height that was set
15753              * @param {Number} rawWidth The width that was originally specified
15754              * @param {Number} rawHeight The height that was originally specified
15755              */
15756         resize : true,
15757         /**
15758          * @event move
15759          * Fires after the component is moved.
15760              * @param {Roo.Component} this
15761              * @param {Number} x The new x position
15762              * @param {Number} y The new y position
15763              */
15764         move : true
15765     });
15766 };
15767
15768 Roo.extend(Roo.BoxComponent, Roo.Component, {
15769     // private, set in afterRender to signify that the component has been rendered
15770     boxReady : false,
15771     // private, used to defer height settings to subclasses
15772     deferHeight: false,
15773     /** @cfg {Number} width
15774      * width (optional) size of component
15775      */
15776      /** @cfg {Number} height
15777      * height (optional) size of component
15778      */
15779      
15780     /**
15781      * Sets the width and height of the component.  This method fires the resize event.  This method can accept
15782      * either width and height as separate numeric arguments, or you can pass a size object like {width:10, height:20}.
15783      * @param {Number/Object} width The new width to set, or a size object in the format {width, height}
15784      * @param {Number} height The new height to set (not required if a size object is passed as the first arg)
15785      * @return {Roo.BoxComponent} this
15786      */
15787     setSize : function(w, h){
15788         // support for standard size objects
15789         if(typeof w == 'object'){
15790             h = w.height;
15791             w = w.width;
15792         }
15793         // not rendered
15794         if(!this.boxReady){
15795             this.width = w;
15796             this.height = h;
15797             return this;
15798         }
15799
15800         // prevent recalcs when not needed
15801         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
15802             return this;
15803         }
15804         this.lastSize = {width: w, height: h};
15805
15806         var adj = this.adjustSize(w, h);
15807         var aw = adj.width, ah = adj.height;
15808         if(aw !== undefined || ah !== undefined){ // this code is nasty but performs better with floaters
15809             var rz = this.getResizeEl();
15810             if(!this.deferHeight && aw !== undefined && ah !== undefined){
15811                 rz.setSize(aw, ah);
15812             }else if(!this.deferHeight && ah !== undefined){
15813                 rz.setHeight(ah);
15814             }else if(aw !== undefined){
15815                 rz.setWidth(aw);
15816             }
15817             this.onResize(aw, ah, w, h);
15818             this.fireEvent('resize', this, aw, ah, w, h);
15819         }
15820         return this;
15821     },
15822
15823     /**
15824      * Gets the current size of the component's underlying element.
15825      * @return {Object} An object containing the element's size {width: (element width), height: (element height)}
15826      */
15827     getSize : function(){
15828         return this.el.getSize();
15829     },
15830
15831     /**
15832      * Gets the current XY position of the component's underlying element.
15833      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15834      * @return {Array} The XY position of the element (e.g., [100, 200])
15835      */
15836     getPosition : function(local){
15837         if(local === true){
15838             return [this.el.getLeft(true), this.el.getTop(true)];
15839         }
15840         return this.xy || this.el.getXY();
15841     },
15842
15843     /**
15844      * Gets the current box measurements of the component's underlying element.
15845      * @param {Boolean} local (optional) If true the element's left and top are returned instead of page XY (defaults to false)
15846      * @returns {Object} box An object in the format {x, y, width, height}
15847      */
15848     getBox : function(local){
15849         var s = this.el.getSize();
15850         if(local){
15851             s.x = this.el.getLeft(true);
15852             s.y = this.el.getTop(true);
15853         }else{
15854             var xy = this.xy || this.el.getXY();
15855             s.x = xy[0];
15856             s.y = xy[1];
15857         }
15858         return s;
15859     },
15860
15861     /**
15862      * Sets the current box measurements of the component's underlying element.
15863      * @param {Object} box An object in the format {x, y, width, height}
15864      * @returns {Roo.BoxComponent} this
15865      */
15866     updateBox : function(box){
15867         this.setSize(box.width, box.height);
15868         this.setPagePosition(box.x, box.y);
15869         return this;
15870     },
15871
15872     // protected
15873     getResizeEl : function(){
15874         return this.resizeEl || this.el;
15875     },
15876
15877     // protected
15878     getPositionEl : function(){
15879         return this.positionEl || this.el;
15880     },
15881
15882     /**
15883      * Sets the left and top of the component.  To set the page XY position instead, use {@link #setPagePosition}.
15884      * This method fires the move event.
15885      * @param {Number} left The new left
15886      * @param {Number} top The new top
15887      * @returns {Roo.BoxComponent} this
15888      */
15889     setPosition : function(x, y){
15890         this.x = x;
15891         this.y = y;
15892         if(!this.boxReady){
15893             return this;
15894         }
15895         var adj = this.adjustPosition(x, y);
15896         var ax = adj.x, ay = adj.y;
15897
15898         var el = this.getPositionEl();
15899         if(ax !== undefined || ay !== undefined){
15900             if(ax !== undefined && ay !== undefined){
15901                 el.setLeftTop(ax, ay);
15902             }else if(ax !== undefined){
15903                 el.setLeft(ax);
15904             }else if(ay !== undefined){
15905                 el.setTop(ay);
15906             }
15907             this.onPosition(ax, ay);
15908             this.fireEvent('move', this, ax, ay);
15909         }
15910         return this;
15911     },
15912
15913     /**
15914      * Sets the page XY position of the component.  To set the left and top instead, use {@link #setPosition}.
15915      * This method fires the move event.
15916      * @param {Number} x The new x position
15917      * @param {Number} y The new y position
15918      * @returns {Roo.BoxComponent} this
15919      */
15920     setPagePosition : function(x, y){
15921         this.pageX = x;
15922         this.pageY = y;
15923         if(!this.boxReady){
15924             return;
15925         }
15926         if(x === undefined || y === undefined){ // cannot translate undefined points
15927             return;
15928         }
15929         var p = this.el.translatePoints(x, y);
15930         this.setPosition(p.left, p.top);
15931         return this;
15932     },
15933
15934     // private
15935     onRender : function(ct, position){
15936         Roo.BoxComponent.superclass.onRender.call(this, ct, position);
15937         if(this.resizeEl){
15938             this.resizeEl = Roo.get(this.resizeEl);
15939         }
15940         if(this.positionEl){
15941             this.positionEl = Roo.get(this.positionEl);
15942         }
15943     },
15944
15945     // private
15946     afterRender : function(){
15947         Roo.BoxComponent.superclass.afterRender.call(this);
15948         this.boxReady = true;
15949         this.setSize(this.width, this.height);
15950         if(this.x || this.y){
15951             this.setPosition(this.x, this.y);
15952         }
15953         if(this.pageX || this.pageY){
15954             this.setPagePosition(this.pageX, this.pageY);
15955         }
15956     },
15957
15958     /**
15959      * Force the component's size to recalculate based on the underlying element's current height and width.
15960      * @returns {Roo.BoxComponent} this
15961      */
15962     syncSize : function(){
15963         delete this.lastSize;
15964         this.setSize(this.el.getWidth(), this.el.getHeight());
15965         return this;
15966     },
15967
15968     /**
15969      * Called after the component is resized, this method is empty by default but can be implemented by any
15970      * subclass that needs to perform custom logic after a resize occurs.
15971      * @param {Number} adjWidth The box-adjusted width that was set
15972      * @param {Number} adjHeight The box-adjusted height that was set
15973      * @param {Number} rawWidth The width that was originally specified
15974      * @param {Number} rawHeight The height that was originally specified
15975      */
15976     onResize : function(adjWidth, adjHeight, rawWidth, rawHeight){
15977
15978     },
15979
15980     /**
15981      * Called after the component is moved, this method is empty by default but can be implemented by any
15982      * subclass that needs to perform custom logic after a move occurs.
15983      * @param {Number} x The new x position
15984      * @param {Number} y The new y position
15985      */
15986     onPosition : function(x, y){
15987
15988     },
15989
15990     // private
15991     adjustSize : function(w, h){
15992         if(this.autoWidth){
15993             w = 'auto';
15994         }
15995         if(this.autoHeight){
15996             h = 'auto';
15997         }
15998         return {width : w, height: h};
15999     },
16000
16001     // private
16002     adjustPosition : function(x, y){
16003         return {x : x, y: y};
16004     }
16005 });/*
16006  * Original code for Roojs - LGPL
16007  * <script type="text/javascript">
16008  */
16009  
16010 /**
16011  * @class Roo.XComponent
16012  * A delayed Element creator...
16013  * Or a way to group chunks of interface together.
16014  * technically this is a wrapper around a tree of Roo elements (which defines a 'module'),
16015  *  used in conjunction with XComponent.build() it will create an instance of each element,
16016  *  then call addxtype() to build the User interface.
16017  * 
16018  * Mypart.xyx = new Roo.XComponent({
16019
16020     parent : 'Mypart.xyz', // empty == document.element.!!
16021     order : '001',
16022     name : 'xxxx'
16023     region : 'xxxx'
16024     disabled : function() {} 
16025      
16026     tree : function() { // return an tree of xtype declared components
16027         var MODULE = this;
16028         return 
16029         {
16030             xtype : 'NestedLayoutPanel',
16031             // technicall
16032         }
16033      ]
16034  *})
16035  *
16036  *
16037  * It can be used to build a big heiracy, with parent etc.
16038  * or you can just use this to render a single compoent to a dom element
16039  * MYPART.render(Roo.Element | String(id) | dom_element )
16040  *
16041  *
16042  * Usage patterns.
16043  *
16044  * Classic Roo
16045  *
16046  * Roo is designed primarily as a single page application, so the UI build for a standard interface will
16047  * expect a single 'TOP' level module normally indicated by the 'parent' of the XComponent definition being defined as false.
16048  *
16049  * Each sub module is expected to have a parent pointing to the class name of it's parent module.
16050  *
16051  * When the top level is false, a 'Roo.BorderLayout' is created and the element is flagged as 'topModule'
16052  * - if mulitple topModules exist, the last one is defined as the top module.
16053  *
16054  * Embeded Roo
16055  * 
16056  * When the top level or multiple modules are to embedded into a existing HTML page,
16057  * the parent element can container '#id' of the element where the module will be drawn.
16058  *
16059  * Bootstrap Roo
16060  *
16061  * Unlike classic Roo, the bootstrap tends not to be used as a single page.
16062  * it relies more on a include mechanism, where sub modules are included into an outer page.
16063  * This is normally managed by the builder tools using Roo.apply( options, Included.Sub.Module )
16064  * 
16065  * Bootstrap Roo Included elements
16066  *
16067  * Our builder application needs the ability to preview these sub compoennts. They will normally have parent=false set,
16068  * hence confusing the component builder as it thinks there are multiple top level elements. 
16069  *
16070  * String Over-ride & Translations
16071  *
16072  * Our builder application writes all the strings as _strings and _named_strings. This is to enable the translation of elements,
16073  * and also the 'overlaying of string values - needed when different versions of the same application with different text content
16074  * are needed. @see Roo.XComponent.overlayString  
16075  * 
16076  * 
16077  * 
16078  * @extends Roo.util.Observable
16079  * @constructor
16080  * @param cfg {Object} configuration of component
16081  * 
16082  */
16083 Roo.XComponent = function(cfg) {
16084     Roo.apply(this, cfg);
16085     this.addEvents({ 
16086         /**
16087              * @event built
16088              * Fires when this the componnt is built
16089              * @param {Roo.XComponent} c the component
16090              */
16091         'built' : true
16092         
16093     });
16094     this.region = this.region || 'center'; // default..
16095     Roo.XComponent.register(this);
16096     this.modules = false;
16097     this.el = false; // where the layout goes..
16098     
16099     
16100 }
16101 Roo.extend(Roo.XComponent, Roo.util.Observable, {
16102     /**
16103      * @property el
16104      * The created element (with Roo.factory())
16105      * @type {Roo.Layout}
16106      */
16107     el  : false,
16108     
16109     /**
16110      * @property el
16111      * for BC  - use el in new code
16112      * @type {Roo.Layout}
16113      */
16114     panel : false,
16115     
16116     /**
16117      * @property layout
16118      * for BC  - use el in new code
16119      * @type {Roo.Layout}
16120      */
16121     layout : false,
16122     
16123      /**
16124      * @cfg {Function|boolean} disabled
16125      * If this module is disabled by some rule, return true from the funtion
16126      */
16127     disabled : false,
16128     
16129     /**
16130      * @cfg {String} parent 
16131      * Name of parent element which it get xtype added to..
16132      */
16133     parent: false,
16134     
16135     /**
16136      * @cfg {String} order
16137      * Used to set the order in which elements are created (usefull for multiple tabs)
16138      */
16139     
16140     order : false,
16141     /**
16142      * @cfg {String} name
16143      * String to display while loading.
16144      */
16145     name : false,
16146     /**
16147      * @cfg {String} region
16148      * Region to render component to (defaults to center)
16149      */
16150     region : 'center',
16151     
16152     /**
16153      * @cfg {Array} items
16154      * A single item array - the first element is the root of the tree..
16155      * It's done this way to stay compatible with the Xtype system...
16156      */
16157     items : false,
16158     
16159     /**
16160      * @property _tree
16161      * The method that retuns the tree of parts that make up this compoennt 
16162      * @type {function}
16163      */
16164     _tree  : false,
16165     
16166      /**
16167      * render
16168      * render element to dom or tree
16169      * @param {Roo.Element|String|DomElement} optional render to if parent is not set.
16170      */
16171     
16172     render : function(el)
16173     {
16174         
16175         el = el || false;
16176         var hp = this.parent ? 1 : 0;
16177         Roo.debug &&  Roo.log(this);
16178         
16179         var tree = this._tree ? this._tree() : this.tree();
16180
16181         
16182         if (!el && typeof(this.parent) == 'string' && this.parent.substring(0,1) == '#') {
16183             // if parent is a '#.....' string, then let's use that..
16184             var ename = this.parent.substr(1);
16185             this.parent = false;
16186             Roo.debug && Roo.log(ename);
16187             switch (ename) {
16188                 case 'bootstrap-body':
16189                     if (typeof(tree.el) != 'undefined' && tree.el == document.body)  {
16190                         // this is the BorderLayout standard?
16191                        this.parent = { el : true };
16192                        break;
16193                     }
16194                     if (["Nest", "Content", "Grid", "Tree"].indexOf(tree.xtype)  > -1)  {
16195                         // need to insert stuff...
16196                         this.parent =  {
16197                              el : new Roo.bootstrap.layout.Border({
16198                                  el : document.body, 
16199                      
16200                                  center: {
16201                                     titlebar: false,
16202                                     autoScroll:false,
16203                                     closeOnTab: true,
16204                                     tabPosition: 'top',
16205                                       //resizeTabs: true,
16206                                     alwaysShowTabs: true,
16207                                     hideTabs: false
16208                                      //minTabWidth: 140
16209                                  }
16210                              })
16211                         
16212                          };
16213                          break;
16214                     }
16215                          
16216                     if (typeof(Roo.bootstrap.Body) != 'undefined' ) {
16217                         this.parent = { el :  new  Roo.bootstrap.Body() };
16218                         Roo.debug && Roo.log("setting el to doc body");
16219                          
16220                     } else {
16221                         throw "Container is bootstrap body, but Roo.bootstrap.Body is not defined";
16222                     }
16223                     break;
16224                 case 'bootstrap':
16225                     this.parent = { el : true};
16226                     // fall through
16227                 default:
16228                     el = Roo.get(ename);
16229                     if (typeof(Roo.bootstrap) != 'undefined' && tree['|xns'] == 'Roo.bootstrap') {
16230                         this.parent = { el : true};
16231                     }
16232                     
16233                     break;
16234             }
16235                 
16236             
16237             if (!el && !this.parent) {
16238                 Roo.debug && Roo.log("Warning - element can not be found :#" + ename );
16239                 return;
16240             }
16241         }
16242         
16243         Roo.debug && Roo.log("EL:");
16244         Roo.debug && Roo.log(el);
16245         Roo.debug && Roo.log("this.parent.el:");
16246         Roo.debug && Roo.log(this.parent.el);
16247         
16248
16249         // altertive root elements ??? - we need a better way to indicate these.
16250         var is_alt = Roo.XComponent.is_alt ||
16251                     (typeof(tree.el) != 'undefined' && tree.el == document.body) ||
16252                     (typeof(Roo.bootstrap) != 'undefined' && tree.xns == Roo.bootstrap) ||
16253                     (typeof(Roo.mailer) != 'undefined' && tree.xns == Roo.mailer) ;
16254         
16255         
16256         
16257         if (!this.parent && is_alt) {
16258             //el = Roo.get(document.body);
16259             this.parent = { el : true };
16260         }
16261             
16262             
16263         
16264         if (!this.parent) {
16265             
16266             Roo.debug && Roo.log("no parent - creating one");
16267             
16268             el = el ? Roo.get(el) : false;      
16269             
16270             if (typeof(Roo.BorderLayout) == 'undefined' ) {
16271                 
16272                 this.parent =  {
16273                     el : new Roo.bootstrap.layout.Border({
16274                         el: el || document.body,
16275                     
16276                         center: {
16277                             titlebar: false,
16278                             autoScroll:false,
16279                             closeOnTab: true,
16280                             tabPosition: 'top',
16281                              //resizeTabs: true,
16282                             alwaysShowTabs: false,
16283                             hideTabs: true,
16284                             minTabWidth: 140,
16285                             overflow: 'visible'
16286                          }
16287                      })
16288                 };
16289             } else {
16290             
16291                 // it's a top level one..
16292                 this.parent =  {
16293                     el : new Roo.BorderLayout(el || document.body, {
16294                         center: {
16295                             titlebar: false,
16296                             autoScroll:false,
16297                             closeOnTab: true,
16298                             tabPosition: 'top',
16299                              //resizeTabs: true,
16300                             alwaysShowTabs: el && hp? false :  true,
16301                             hideTabs: el || !hp ? true :  false,
16302                             minTabWidth: 140
16303                          }
16304                     })
16305                 };
16306             }
16307         }
16308         
16309         if (!this.parent.el) {
16310                 // probably an old style ctor, which has been disabled.
16311                 return;
16312
16313         }
16314                 // The 'tree' method is  '_tree now' 
16315             
16316         tree.region = tree.region || this.region;
16317         var is_body = false;
16318         if (this.parent.el === true) {
16319             // bootstrap... - body..
16320             if (el) {
16321                 tree.el = el;
16322             }
16323             this.parent.el = Roo.factory(tree);
16324             is_body = true;
16325         }
16326         
16327         this.el = this.parent.el.addxtype(tree, undefined, is_body);
16328         this.fireEvent('built', this);
16329         
16330         this.panel = this.el;
16331         this.layout = this.panel.layout;
16332         this.parentLayout = this.parent.layout  || false;  
16333          
16334     }
16335     
16336 });
16337
16338 Roo.apply(Roo.XComponent, {
16339     /**
16340      * @property  hideProgress
16341      * true to disable the building progress bar.. usefull on single page renders.
16342      * @type Boolean
16343      */
16344     hideProgress : false,
16345     /**
16346      * @property  buildCompleted
16347      * True when the builder has completed building the interface.
16348      * @type Boolean
16349      */
16350     buildCompleted : false,
16351      
16352     /**
16353      * @property  topModule
16354      * the upper most module - uses document.element as it's constructor.
16355      * @type Object
16356      */
16357      
16358     topModule  : false,
16359       
16360     /**
16361      * @property  modules
16362      * array of modules to be created by registration system.
16363      * @type {Array} of Roo.XComponent
16364      */
16365     
16366     modules : [],
16367     /**
16368      * @property  elmodules
16369      * array of modules to be created by which use #ID 
16370      * @type {Array} of Roo.XComponent
16371      */
16372      
16373     elmodules : [],
16374
16375      /**
16376      * @property  is_alt
16377      * Is an alternative Root - normally used by bootstrap or other systems,
16378      *    where the top element in the tree can wrap 'body' 
16379      * @type {boolean}  (default false)
16380      */
16381      
16382     is_alt : false,
16383     /**
16384      * @property  build_from_html
16385      * Build elements from html - used by bootstrap HTML stuff 
16386      *    - this is cleared after build is completed
16387      * @type {boolean}    (default false)
16388      */
16389      
16390     build_from_html : false,
16391     /**
16392      * Register components to be built later.
16393      *
16394      * This solves the following issues
16395      * - Building is not done on page load, but after an authentication process has occured.
16396      * - Interface elements are registered on page load
16397      * - Parent Interface elements may not be loaded before child, so this handles that..
16398      * 
16399      *
16400      * example:
16401      * 
16402      * MyApp.register({
16403           order : '000001',
16404           module : 'Pman.Tab.projectMgr',
16405           region : 'center',
16406           parent : 'Pman.layout',
16407           disabled : false,  // or use a function..
16408         })
16409      
16410      * * @param {Object} details about module
16411      */
16412     register : function(obj) {
16413                 
16414         Roo.XComponent.event.fireEvent('register', obj);
16415         switch(typeof(obj.disabled) ) {
16416                 
16417             case 'undefined':
16418                 break;
16419             
16420             case 'function':
16421                 if ( obj.disabled() ) {
16422                         return;
16423                 }
16424                 break;
16425             
16426             default:
16427                 if (obj.disabled || obj.region == '#disabled') {
16428                         return;
16429                 }
16430                 break;
16431         }
16432                 
16433         this.modules.push(obj);
16434          
16435     },
16436     /**
16437      * convert a string to an object..
16438      * eg. 'AAA.BBB' -> finds AAA.BBB
16439
16440      */
16441     
16442     toObject : function(str)
16443     {
16444         if (!str || typeof(str) == 'object') {
16445             return str;
16446         }
16447         if (str.substring(0,1) == '#') {
16448             return str;
16449         }
16450
16451         var ar = str.split('.');
16452         var rt, o;
16453         rt = ar.shift();
16454             /** eval:var:o */
16455         try {
16456             eval('if (typeof ' + rt + ' == "undefined"){ o = false;} o = ' + rt + ';');
16457         } catch (e) {
16458             throw "Module not found : " + str;
16459         }
16460         
16461         if (o === false) {
16462             throw "Module not found : " + str;
16463         }
16464         Roo.each(ar, function(e) {
16465             if (typeof(o[e]) == 'undefined') {
16466                 throw "Module not found : " + str;
16467             }
16468             o = o[e];
16469         });
16470         
16471         return o;
16472         
16473     },
16474     
16475     
16476     /**
16477      * move modules into their correct place in the tree..
16478      * 
16479      */
16480     preBuild : function ()
16481     {
16482         var _t = this;
16483         Roo.each(this.modules , function (obj)
16484         {
16485             Roo.XComponent.event.fireEvent('beforebuild', obj);
16486             
16487             var opar = obj.parent;
16488             try { 
16489                 obj.parent = this.toObject(opar);
16490             } catch(e) {
16491                 Roo.debug && Roo.log("parent:toObject failed: " + e.toString());
16492                 return;
16493             }
16494             
16495             if (!obj.parent) {
16496                 Roo.debug && Roo.log("GOT top level module");
16497                 Roo.debug && Roo.log(obj);
16498                 obj.modules = new Roo.util.MixedCollection(false, 
16499                     function(o) { return o.order + '' }
16500                 );
16501                 this.topModule = obj;
16502                 return;
16503             }
16504                         // parent is a string (usually a dom element name..)
16505             if (typeof(obj.parent) == 'string') {
16506                 this.elmodules.push(obj);
16507                 return;
16508             }
16509             if (obj.parent.constructor != Roo.XComponent) {
16510                 Roo.debug && Roo.log("Warning : Object Parent is not instance of XComponent:" + obj.name)
16511             }
16512             if (!obj.parent.modules) {
16513                 obj.parent.modules = new Roo.util.MixedCollection(false, 
16514                     function(o) { return o.order + '' }
16515                 );
16516             }
16517             if (obj.parent.disabled) {
16518                 obj.disabled = true;
16519             }
16520             obj.parent.modules.add(obj);
16521         }, this);
16522     },
16523     
16524      /**
16525      * make a list of modules to build.
16526      * @return {Array} list of modules. 
16527      */ 
16528     
16529     buildOrder : function()
16530     {
16531         var _this = this;
16532         var cmp = function(a,b) {   
16533             return String(a).toUpperCase() > String(b).toUpperCase() ? 1 : -1;
16534         };
16535         if ((!this.topModule || !this.topModule.modules) && !this.elmodules.length) {
16536             throw "No top level modules to build";
16537         }
16538         
16539         // make a flat list in order of modules to build.
16540         var mods = this.topModule ? [ this.topModule ] : [];
16541                 
16542         
16543         // elmodules (is a list of DOM based modules )
16544         Roo.each(this.elmodules, function(e) {
16545             mods.push(e);
16546             if (!this.topModule &&
16547                 typeof(e.parent) == 'string' &&
16548                 e.parent.substring(0,1) == '#' &&
16549                 Roo.get(e.parent.substr(1))
16550                ) {
16551                 
16552                 _this.topModule = e;
16553             }
16554             
16555         });
16556
16557         
16558         // add modules to their parents..
16559         var addMod = function(m) {
16560             Roo.debug && Roo.log("build Order: add: " + m.name);
16561                 
16562             mods.push(m);
16563             if (m.modules && !m.disabled) {
16564                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules");
16565                 m.modules.keySort('ASC',  cmp );
16566                 Roo.debug && Roo.log("build Order: " + m.modules.length + " child modules (after sort)");
16567     
16568                 m.modules.each(addMod);
16569             } else {
16570                 Roo.debug && Roo.log("build Order: no child modules");
16571             }
16572             // not sure if this is used any more..
16573             if (m.finalize) {
16574                 m.finalize.name = m.name + " (clean up) ";
16575                 mods.push(m.finalize);
16576             }
16577             
16578         }
16579         if (this.topModule && this.topModule.modules) { 
16580             this.topModule.modules.keySort('ASC',  cmp );
16581             this.topModule.modules.each(addMod);
16582         } 
16583         return mods;
16584     },
16585     
16586      /**
16587      * Build the registered modules.
16588      * @param {Object} parent element.
16589      * @param {Function} optional method to call after module has been added.
16590      * 
16591      */ 
16592    
16593     build : function(opts) 
16594     {
16595         
16596         if (typeof(opts) != 'undefined') {
16597             Roo.apply(this,opts);
16598         }
16599         
16600         this.preBuild();
16601         var mods = this.buildOrder();
16602       
16603         //this.allmods = mods;
16604         //Roo.debug && Roo.log(mods);
16605         //return;
16606         if (!mods.length) { // should not happen
16607             throw "NO modules!!!";
16608         }
16609         
16610         
16611         var msg = "Building Interface...";
16612         // flash it up as modal - so we store the mask!?
16613         if (!this.hideProgress && Roo.MessageBox) {
16614             Roo.MessageBox.show({ title: 'loading' });
16615             Roo.MessageBox.show({
16616                title: "Please wait...",
16617                msg: msg,
16618                width:450,
16619                progress:true,
16620                buttons : false,
16621                closable:false,
16622                modal: false
16623               
16624             });
16625         }
16626         var total = mods.length;
16627         
16628         var _this = this;
16629         var progressRun = function() {
16630             if (!mods.length) {
16631                 Roo.debug && Roo.log('hide?');
16632                 if (!this.hideProgress && Roo.MessageBox) {
16633                     Roo.MessageBox.hide();
16634                 }
16635                 Roo.XComponent.build_from_html = false; // reset, so dialogs will be build from javascript
16636                 
16637                 Roo.XComponent.event.fireEvent('buildcomplete', _this.topModule);
16638                 
16639                 // THE END...
16640                 return false;   
16641             }
16642             
16643             var m = mods.shift();
16644             
16645             
16646             Roo.debug && Roo.log(m);
16647             // not sure if this is supported any more.. - modules that are are just function
16648             if (typeof(m) == 'function') { 
16649                 m.call(this);
16650                 return progressRun.defer(10, _this);
16651             } 
16652             
16653             
16654             msg = "Building Interface " + (total  - mods.length) + 
16655                     " of " + total + 
16656                     (m.name ? (' - ' + m.name) : '');
16657                         Roo.debug && Roo.log(msg);
16658             if (!_this.hideProgress &&  Roo.MessageBox) { 
16659                 Roo.MessageBox.updateProgress(  (total  - mods.length)/total, msg  );
16660             }
16661             
16662          
16663             // is the module disabled?
16664             var disabled = (typeof(m.disabled) == 'function') ?
16665                 m.disabled.call(m.module.disabled) : m.disabled;    
16666             
16667             
16668             if (disabled) {
16669                 return progressRun(); // we do not update the display!
16670             }
16671             
16672             // now build 
16673             
16674                         
16675                         
16676             m.render();
16677             // it's 10 on top level, and 1 on others??? why...
16678             return progressRun.defer(10, _this);
16679              
16680         }
16681         progressRun.defer(1, _this);
16682      
16683         
16684         
16685     },
16686     /**
16687      * Overlay a set of modified strings onto a component
16688      * This is dependant on our builder exporting the strings and 'named strings' elements.
16689      * 
16690      * @param {Object} element to overlay on - eg. Pman.Dialog.Login
16691      * @param {Object} associative array of 'named' string and it's new value.
16692      * 
16693      */
16694         overlayStrings : function( component, strings )
16695     {
16696         if (typeof(component['_named_strings']) == 'undefined') {
16697             throw "ERROR: component does not have _named_strings";
16698         }
16699         for ( var k in strings ) {
16700             var md = typeof(component['_named_strings'][k]) == 'undefined' ? false : component['_named_strings'][k];
16701             if (md !== false) {
16702                 component['_strings'][md] = strings[k];
16703             } else {
16704                 Roo.log('could not find named string: ' + k + ' in');
16705                 Roo.log(component);
16706             }
16707             
16708         }
16709         
16710     },
16711     
16712         
16713         /**
16714          * Event Object.
16715          *
16716          *
16717          */
16718         event: false, 
16719     /**
16720          * wrapper for event.on - aliased later..  
16721          * Typically use to register a event handler for register:
16722          *
16723          * eg. Roo.XComponent.on('register', function(comp) { comp.disable = true } );
16724          *
16725          */
16726     on : false
16727    
16728     
16729     
16730 });
16731
16732 Roo.XComponent.event = new Roo.util.Observable({
16733                 events : { 
16734                         /**
16735                          * @event register
16736                          * Fires when an Component is registered,
16737                          * set the disable property on the Component to stop registration.
16738                          * @param {Roo.XComponent} c the component being registerd.
16739                          * 
16740                          */
16741                         'register' : true,
16742             /**
16743                          * @event beforebuild
16744                          * Fires before each Component is built
16745                          * can be used to apply permissions.
16746                          * @param {Roo.XComponent} c the component being registerd.
16747                          * 
16748                          */
16749                         'beforebuild' : true,
16750                         /**
16751                          * @event buildcomplete
16752                          * Fires on the top level element when all elements have been built
16753                          * @param {Roo.XComponent} the top level component.
16754                          */
16755                         'buildcomplete' : true
16756                         
16757                 }
16758 });
16759
16760 Roo.XComponent.on = Roo.XComponent.event.on.createDelegate(Roo.XComponent.event); 
16761  //
16762  /**
16763  * marked - a markdown parser
16764  * Copyright (c) 2011-2014, Christopher Jeffrey. (MIT Licensed)
16765  * https://github.com/chjj/marked
16766  */
16767
16768
16769 /**
16770  *
16771  * Roo.Markdown - is a very crude wrapper around marked..
16772  *
16773  * usage:
16774  * 
16775  * alert( Roo.Markdown.toHtml("Markdown *rocks*.") );
16776  * 
16777  * Note: move the sample code to the bottom of this
16778  * file before uncommenting it.
16779  *
16780  */
16781
16782 Roo.Markdown = {};
16783 Roo.Markdown.toHtml = function(text) {
16784     
16785     var c = new Roo.Markdown.marked.setOptions({
16786             renderer: new Roo.Markdown.marked.Renderer(),
16787             gfm: true,
16788             tables: true,
16789             breaks: false,
16790             pedantic: false,
16791             sanitize: false,
16792             smartLists: true,
16793             smartypants: false
16794           });
16795     // A FEW HACKS!!?
16796     
16797     text = text.replace(/\\\n/g,' ');
16798     return Roo.Markdown.marked(text);
16799 };
16800 //
16801 // converter
16802 //
16803 // Wraps all "globals" so that the only thing
16804 // exposed is makeHtml().
16805 //
16806 (function() {
16807     
16808      /**
16809          * eval:var:escape
16810          * eval:var:unescape
16811          * eval:var:replace
16812          */
16813       
16814     /**
16815      * Helpers
16816      */
16817     
16818     var escape = function (html, encode) {
16819       return html
16820         .replace(!encode ? /&(?!#?\w+;)/g : /&/g, '&amp;')
16821         .replace(/</g, '&lt;')
16822         .replace(/>/g, '&gt;')
16823         .replace(/"/g, '&quot;')
16824         .replace(/'/g, '&#39;');
16825     }
16826     
16827     var unescape = function (html) {
16828         // explicitly match decimal, hex, and named HTML entities 
16829       return html.replace(/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/g, function(_, n) {
16830         n = n.toLowerCase();
16831         if (n === 'colon') { return ':'; }
16832         if (n.charAt(0) === '#') {
16833           return n.charAt(1) === 'x'
16834             ? String.fromCharCode(parseInt(n.substring(2), 16))
16835             : String.fromCharCode(+n.substring(1));
16836         }
16837         return '';
16838       });
16839     }
16840     
16841     var replace = function (regex, opt) {
16842       regex = regex.source;
16843       opt = opt || '';
16844       return function self(name, val) {
16845         if (!name) { return new RegExp(regex, opt); }
16846         val = val.source || val;
16847         val = val.replace(/(^|[^\[])\^/g, '$1');
16848         regex = regex.replace(name, val);
16849         return self;
16850       };
16851     }
16852     
16853     /**
16854      * Block-Level Grammar
16855      */
16856     
16857     
16858     
16859     
16860     var block = {
16861       newline: /^\n+/,
16862       code: /^( {4}[^\n]+\n*)+/,
16863       fences: noop,
16864       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16865       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16866       nptable: noop,
16867       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16868       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16869       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16870       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16871       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16872       table: noop,
16873       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16874       text: /^[^\n]+/
16875     };
16876     
16877     block.bullet = /(?:[*+-]|\d+\.)/;
16878     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16879     block.item = replace(block.item, 'gm')
16880       (/bull/g, block.bullet)
16881       ();
16882     
16883     block.list = replace(block.list)
16884       (/bull/g, block.bullet)
16885       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16886       ('def', '\\n+(?=' + block.def.source + ')')
16887       ();
16888     
16889     block.blockquote = replace(block.blockquote)
16890       ('def', block.def)
16891       ();
16892     
16893     block._tag = '(?!(?:'
16894       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16895       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16896       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16897     
16898     block.html = replace(block.html)
16899       ('comment', /<!--[\s\S]*?-->/)
16900       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16901       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16902       (/tag/g, block._tag)
16903       ();
16904     
16905     block.paragraph = replace(block.paragraph)
16906       ('hr', block.hr)
16907       ('heading', block.heading)
16908       ('lheading', block.lheading)
16909       ('blockquote', block.blockquote)
16910       ('tag', '<' + block._tag)
16911       ('def', block.def)
16912       ();
16913     
16914     /**
16915      * Normal Block Grammar
16916      */
16917     
16918     block.normal = merge({}, block);
16919     
16920     /**
16921      * GFM Block Grammar
16922      */
16923     
16924     block.gfm = merge({}, block.normal, {
16925       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16926       paragraph: /^/,
16927       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16928     });
16929     
16930     block.gfm.paragraph = replace(block.paragraph)
16931       ('(?!', '(?!'
16932         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16933         + block.list.source.replace('\\1', '\\3') + '|')
16934       ();
16935     
16936     /**
16937      * GFM + Tables Block Grammar
16938      */
16939     
16940     block.tables = merge({}, block.gfm, {
16941       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16942       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16943     });
16944     
16945     /**
16946      * Block Lexer
16947      */
16948     
16949     var Lexer = function (options) {
16950       this.tokens = [];
16951       this.tokens.links = {};
16952       this.options = options || marked.defaults;
16953       this.rules = block.normal;
16954     
16955       if (this.options.gfm) {
16956         if (this.options.tables) {
16957           this.rules = block.tables;
16958         } else {
16959           this.rules = block.gfm;
16960         }
16961       }
16962     }
16963     
16964     /**
16965      * Expose Block Rules
16966      */
16967     
16968     Lexer.rules = block;
16969     
16970     /**
16971      * Static Lex Method
16972      */
16973     
16974     Lexer.lex = function(src, options) {
16975       var lexer = new Lexer(options);
16976       return lexer.lex(src);
16977     };
16978     
16979     /**
16980      * Preprocessing
16981      */
16982     
16983     Lexer.prototype.lex = function(src) {
16984       src = src
16985         .replace(/\r\n|\r/g, '\n')
16986         .replace(/\t/g, '    ')
16987         .replace(/\u00a0/g, ' ')
16988         .replace(/\u2424/g, '\n');
16989     
16990       return this.token(src, true);
16991     };
16992     
16993     /**
16994      * Lexing
16995      */
16996     
16997     Lexer.prototype.token = function(src, top, bq) {
16998       var src = src.replace(/^ +$/gm, '')
16999         , next
17000         , loose
17001         , cap
17002         , bull
17003         , b
17004         , item
17005         , space
17006         , i
17007         , l;
17008     
17009       while (src) {
17010         // newline
17011         if (cap = this.rules.newline.exec(src)) {
17012           src = src.substring(cap[0].length);
17013           if (cap[0].length > 1) {
17014             this.tokens.push({
17015               type: 'space'
17016             });
17017           }
17018         }
17019     
17020         // code
17021         if (cap = this.rules.code.exec(src)) {
17022           src = src.substring(cap[0].length);
17023           cap = cap[0].replace(/^ {4}/gm, '');
17024           this.tokens.push({
17025             type: 'code',
17026             text: !this.options.pedantic
17027               ? cap.replace(/\n+$/, '')
17028               : cap
17029           });
17030           continue;
17031         }
17032     
17033         // fences (gfm)
17034         if (cap = this.rules.fences.exec(src)) {
17035           src = src.substring(cap[0].length);
17036           this.tokens.push({
17037             type: 'code',
17038             lang: cap[2],
17039             text: cap[3] || ''
17040           });
17041           continue;
17042         }
17043     
17044         // heading
17045         if (cap = this.rules.heading.exec(src)) {
17046           src = src.substring(cap[0].length);
17047           this.tokens.push({
17048             type: 'heading',
17049             depth: cap[1].length,
17050             text: cap[2]
17051           });
17052           continue;
17053         }
17054     
17055         // table no leading pipe (gfm)
17056         if (top && (cap = this.rules.nptable.exec(src))) {
17057           src = src.substring(cap[0].length);
17058     
17059           item = {
17060             type: 'table',
17061             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17062             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17063             cells: cap[3].replace(/\n$/, '').split('\n')
17064           };
17065     
17066           for (i = 0; i < item.align.length; i++) {
17067             if (/^ *-+: *$/.test(item.align[i])) {
17068               item.align[i] = 'right';
17069             } else if (/^ *:-+: *$/.test(item.align[i])) {
17070               item.align[i] = 'center';
17071             } else if (/^ *:-+ *$/.test(item.align[i])) {
17072               item.align[i] = 'left';
17073             } else {
17074               item.align[i] = null;
17075             }
17076           }
17077     
17078           for (i = 0; i < item.cells.length; i++) {
17079             item.cells[i] = item.cells[i].split(/ *\| */);
17080           }
17081     
17082           this.tokens.push(item);
17083     
17084           continue;
17085         }
17086     
17087         // lheading
17088         if (cap = this.rules.lheading.exec(src)) {
17089           src = src.substring(cap[0].length);
17090           this.tokens.push({
17091             type: 'heading',
17092             depth: cap[2] === '=' ? 1 : 2,
17093             text: cap[1]
17094           });
17095           continue;
17096         }
17097     
17098         // hr
17099         if (cap = this.rules.hr.exec(src)) {
17100           src = src.substring(cap[0].length);
17101           this.tokens.push({
17102             type: 'hr'
17103           });
17104           continue;
17105         }
17106     
17107         // blockquote
17108         if (cap = this.rules.blockquote.exec(src)) {
17109           src = src.substring(cap[0].length);
17110     
17111           this.tokens.push({
17112             type: 'blockquote_start'
17113           });
17114     
17115           cap = cap[0].replace(/^ *> ?/gm, '');
17116     
17117           // Pass `top` to keep the current
17118           // "toplevel" state. This is exactly
17119           // how markdown.pl works.
17120           this.token(cap, top, true);
17121     
17122           this.tokens.push({
17123             type: 'blockquote_end'
17124           });
17125     
17126           continue;
17127         }
17128     
17129         // list
17130         if (cap = this.rules.list.exec(src)) {
17131           src = src.substring(cap[0].length);
17132           bull = cap[2];
17133     
17134           this.tokens.push({
17135             type: 'list_start',
17136             ordered: bull.length > 1
17137           });
17138     
17139           // Get each top-level item.
17140           cap = cap[0].match(this.rules.item);
17141     
17142           next = false;
17143           l = cap.length;
17144           i = 0;
17145     
17146           for (; i < l; i++) {
17147             item = cap[i];
17148     
17149             // Remove the list item's bullet
17150             // so it is seen as the next token.
17151             space = item.length;
17152             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17153     
17154             // Outdent whatever the
17155             // list item contains. Hacky.
17156             if (~item.indexOf('\n ')) {
17157               space -= item.length;
17158               item = !this.options.pedantic
17159                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17160                 : item.replace(/^ {1,4}/gm, '');
17161             }
17162     
17163             // Determine whether the next list item belongs here.
17164             // Backpedal if it does not belong in this list.
17165             if (this.options.smartLists && i !== l - 1) {
17166               b = block.bullet.exec(cap[i + 1])[0];
17167               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17168                 src = cap.slice(i + 1).join('\n') + src;
17169                 i = l - 1;
17170               }
17171             }
17172     
17173             // Determine whether item is loose or not.
17174             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17175             // for discount behavior.
17176             loose = next || /\n\n(?!\s*$)/.test(item);
17177             if (i !== l - 1) {
17178               next = item.charAt(item.length - 1) === '\n';
17179               if (!loose) { loose = next; }
17180             }
17181     
17182             this.tokens.push({
17183               type: loose
17184                 ? 'loose_item_start'
17185                 : 'list_item_start'
17186             });
17187     
17188             // Recurse.
17189             this.token(item, false, bq);
17190     
17191             this.tokens.push({
17192               type: 'list_item_end'
17193             });
17194           }
17195     
17196           this.tokens.push({
17197             type: 'list_end'
17198           });
17199     
17200           continue;
17201         }
17202     
17203         // html
17204         if (cap = this.rules.html.exec(src)) {
17205           src = src.substring(cap[0].length);
17206           this.tokens.push({
17207             type: this.options.sanitize
17208               ? 'paragraph'
17209               : 'html',
17210             pre: !this.options.sanitizer
17211               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17212             text: cap[0]
17213           });
17214           continue;
17215         }
17216     
17217         // def
17218         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17219           src = src.substring(cap[0].length);
17220           this.tokens.links[cap[1].toLowerCase()] = {
17221             href: cap[2],
17222             title: cap[3]
17223           };
17224           continue;
17225         }
17226     
17227         // table (gfm)
17228         if (top && (cap = this.rules.table.exec(src))) {
17229           src = src.substring(cap[0].length);
17230     
17231           item = {
17232             type: 'table',
17233             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17234             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17235             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17236           };
17237     
17238           for (i = 0; i < item.align.length; i++) {
17239             if (/^ *-+: *$/.test(item.align[i])) {
17240               item.align[i] = 'right';
17241             } else if (/^ *:-+: *$/.test(item.align[i])) {
17242               item.align[i] = 'center';
17243             } else if (/^ *:-+ *$/.test(item.align[i])) {
17244               item.align[i] = 'left';
17245             } else {
17246               item.align[i] = null;
17247             }
17248           }
17249     
17250           for (i = 0; i < item.cells.length; i++) {
17251             item.cells[i] = item.cells[i]
17252               .replace(/^ *\| *| *\| *$/g, '')
17253               .split(/ *\| */);
17254           }
17255     
17256           this.tokens.push(item);
17257     
17258           continue;
17259         }
17260     
17261         // top-level paragraph
17262         if (top && (cap = this.rules.paragraph.exec(src))) {
17263           src = src.substring(cap[0].length);
17264           this.tokens.push({
17265             type: 'paragraph',
17266             text: cap[1].charAt(cap[1].length - 1) === '\n'
17267               ? cap[1].slice(0, -1)
17268               : cap[1]
17269           });
17270           continue;
17271         }
17272     
17273         // text
17274         if (cap = this.rules.text.exec(src)) {
17275           // Top-level should never reach here.
17276           src = src.substring(cap[0].length);
17277           this.tokens.push({
17278             type: 'text',
17279             text: cap[0]
17280           });
17281           continue;
17282         }
17283     
17284         if (src) {
17285           throw new
17286             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17287         }
17288       }
17289     
17290       return this.tokens;
17291     };
17292     
17293     /**
17294      * Inline-Level Grammar
17295      */
17296     
17297     var inline = {
17298       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17299       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17300       url: noop,
17301       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17302       link: /^!?\[(inside)\]\(href\)/,
17303       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17304       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17305       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17306       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17307       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17308       br: /^ {2,}\n(?!\s*$)/,
17309       del: noop,
17310       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17311     };
17312     
17313     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17314     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17315     
17316     inline.link = replace(inline.link)
17317       ('inside', inline._inside)
17318       ('href', inline._href)
17319       ();
17320     
17321     inline.reflink = replace(inline.reflink)
17322       ('inside', inline._inside)
17323       ();
17324     
17325     /**
17326      * Normal Inline Grammar
17327      */
17328     
17329     inline.normal = merge({}, inline);
17330     
17331     /**
17332      * Pedantic Inline Grammar
17333      */
17334     
17335     inline.pedantic = merge({}, inline.normal, {
17336       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17337       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17338     });
17339     
17340     /**
17341      * GFM Inline Grammar
17342      */
17343     
17344     inline.gfm = merge({}, inline.normal, {
17345       escape: replace(inline.escape)('])', '~|])')(),
17346       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17347       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17348       text: replace(inline.text)
17349         (']|', '~]|')
17350         ('|', '|https?://|')
17351         ()
17352     });
17353     
17354     /**
17355      * GFM + Line Breaks Inline Grammar
17356      */
17357     
17358     inline.breaks = merge({}, inline.gfm, {
17359       br: replace(inline.br)('{2,}', '*')(),
17360       text: replace(inline.gfm.text)('{2,}', '*')()
17361     });
17362     
17363     /**
17364      * Inline Lexer & Compiler
17365      */
17366     
17367     var InlineLexer  = function (links, options) {
17368       this.options = options || marked.defaults;
17369       this.links = links;
17370       this.rules = inline.normal;
17371       this.renderer = this.options.renderer || new Renderer;
17372       this.renderer.options = this.options;
17373     
17374       if (!this.links) {
17375         throw new
17376           Error('Tokens array requires a `links` property.');
17377       }
17378     
17379       if (this.options.gfm) {
17380         if (this.options.breaks) {
17381           this.rules = inline.breaks;
17382         } else {
17383           this.rules = inline.gfm;
17384         }
17385       } else if (this.options.pedantic) {
17386         this.rules = inline.pedantic;
17387       }
17388     }
17389     
17390     /**
17391      * Expose Inline Rules
17392      */
17393     
17394     InlineLexer.rules = inline;
17395     
17396     /**
17397      * Static Lexing/Compiling Method
17398      */
17399     
17400     InlineLexer.output = function(src, links, options) {
17401       var inline = new InlineLexer(links, options);
17402       return inline.output(src);
17403     };
17404     
17405     /**
17406      * Lexing/Compiling
17407      */
17408     
17409     InlineLexer.prototype.output = function(src) {
17410       var out = ''
17411         , link
17412         , text
17413         , href
17414         , cap;
17415     
17416       while (src) {
17417         // escape
17418         if (cap = this.rules.escape.exec(src)) {
17419           src = src.substring(cap[0].length);
17420           out += cap[1];
17421           continue;
17422         }
17423     
17424         // autolink
17425         if (cap = this.rules.autolink.exec(src)) {
17426           src = src.substring(cap[0].length);
17427           if (cap[2] === '@') {
17428             text = cap[1].charAt(6) === ':'
17429               ? this.mangle(cap[1].substring(7))
17430               : this.mangle(cap[1]);
17431             href = this.mangle('mailto:') + text;
17432           } else {
17433             text = escape(cap[1]);
17434             href = text;
17435           }
17436           out += this.renderer.link(href, null, text);
17437           continue;
17438         }
17439     
17440         // url (gfm)
17441         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17442           src = src.substring(cap[0].length);
17443           text = escape(cap[1]);
17444           href = text;
17445           out += this.renderer.link(href, null, text);
17446           continue;
17447         }
17448     
17449         // tag
17450         if (cap = this.rules.tag.exec(src)) {
17451           if (!this.inLink && /^<a /i.test(cap[0])) {
17452             this.inLink = true;
17453           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17454             this.inLink = false;
17455           }
17456           src = src.substring(cap[0].length);
17457           out += this.options.sanitize
17458             ? this.options.sanitizer
17459               ? this.options.sanitizer(cap[0])
17460               : escape(cap[0])
17461             : cap[0];
17462           continue;
17463         }
17464     
17465         // link
17466         if (cap = this.rules.link.exec(src)) {
17467           src = src.substring(cap[0].length);
17468           this.inLink = true;
17469           out += this.outputLink(cap, {
17470             href: cap[2],
17471             title: cap[3]
17472           });
17473           this.inLink = false;
17474           continue;
17475         }
17476     
17477         // reflink, nolink
17478         if ((cap = this.rules.reflink.exec(src))
17479             || (cap = this.rules.nolink.exec(src))) {
17480           src = src.substring(cap[0].length);
17481           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17482           link = this.links[link.toLowerCase()];
17483           if (!link || !link.href) {
17484             out += cap[0].charAt(0);
17485             src = cap[0].substring(1) + src;
17486             continue;
17487           }
17488           this.inLink = true;
17489           out += this.outputLink(cap, link);
17490           this.inLink = false;
17491           continue;
17492         }
17493     
17494         // strong
17495         if (cap = this.rules.strong.exec(src)) {
17496           src = src.substring(cap[0].length);
17497           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17498           continue;
17499         }
17500     
17501         // em
17502         if (cap = this.rules.em.exec(src)) {
17503           src = src.substring(cap[0].length);
17504           out += this.renderer.em(this.output(cap[2] || cap[1]));
17505           continue;
17506         }
17507     
17508         // code
17509         if (cap = this.rules.code.exec(src)) {
17510           src = src.substring(cap[0].length);
17511           out += this.renderer.codespan(escape(cap[2], true));
17512           continue;
17513         }
17514     
17515         // br
17516         if (cap = this.rules.br.exec(src)) {
17517           src = src.substring(cap[0].length);
17518           out += this.renderer.br();
17519           continue;
17520         }
17521     
17522         // del (gfm)
17523         if (cap = this.rules.del.exec(src)) {
17524           src = src.substring(cap[0].length);
17525           out += this.renderer.del(this.output(cap[1]));
17526           continue;
17527         }
17528     
17529         // text
17530         if (cap = this.rules.text.exec(src)) {
17531           src = src.substring(cap[0].length);
17532           out += this.renderer.text(escape(this.smartypants(cap[0])));
17533           continue;
17534         }
17535     
17536         if (src) {
17537           throw new
17538             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17539         }
17540       }
17541     
17542       return out;
17543     };
17544     
17545     /**
17546      * Compile Link
17547      */
17548     
17549     InlineLexer.prototype.outputLink = function(cap, link) {
17550       var href = escape(link.href)
17551         , title = link.title ? escape(link.title) : null;
17552     
17553       return cap[0].charAt(0) !== '!'
17554         ? this.renderer.link(href, title, this.output(cap[1]))
17555         : this.renderer.image(href, title, escape(cap[1]));
17556     };
17557     
17558     /**
17559      * Smartypants Transformations
17560      */
17561     
17562     InlineLexer.prototype.smartypants = function(text) {
17563       if (!this.options.smartypants)  { return text; }
17564       return text
17565         // em-dashes
17566         .replace(/---/g, '\u2014')
17567         // en-dashes
17568         .replace(/--/g, '\u2013')
17569         // opening singles
17570         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17571         // closing singles & apostrophes
17572         .replace(/'/g, '\u2019')
17573         // opening doubles
17574         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17575         // closing doubles
17576         .replace(/"/g, '\u201d')
17577         // ellipses
17578         .replace(/\.{3}/g, '\u2026');
17579     };
17580     
17581     /**
17582      * Mangle Links
17583      */
17584     
17585     InlineLexer.prototype.mangle = function(text) {
17586       if (!this.options.mangle) { return text; }
17587       var out = ''
17588         , l = text.length
17589         , i = 0
17590         , ch;
17591     
17592       for (; i < l; i++) {
17593         ch = text.charCodeAt(i);
17594         if (Math.random() > 0.5) {
17595           ch = 'x' + ch.toString(16);
17596         }
17597         out += '&#' + ch + ';';
17598       }
17599     
17600       return out;
17601     };
17602     
17603     /**
17604      * Renderer
17605      */
17606     
17607      /**
17608          * eval:var:Renderer
17609     */
17610     
17611     var Renderer   = function (options) {
17612       this.options = options || {};
17613     }
17614     
17615     Renderer.prototype.code = function(code, lang, escaped) {
17616       if (this.options.highlight) {
17617         var out = this.options.highlight(code, lang);
17618         if (out != null && out !== code) {
17619           escaped = true;
17620           code = out;
17621         }
17622       } else {
17623             // hack!!! - it's already escapeD?
17624             escaped = true;
17625       }
17626     
17627       if (!lang) {
17628         return '<pre><code>'
17629           + (escaped ? code : escape(code, true))
17630           + '\n</code></pre>';
17631       }
17632     
17633       return '<pre><code class="'
17634         + this.options.langPrefix
17635         + escape(lang, true)
17636         + '">'
17637         + (escaped ? code : escape(code, true))
17638         + '\n</code></pre>\n';
17639     };
17640     
17641     Renderer.prototype.blockquote = function(quote) {
17642       return '<blockquote>\n' + quote + '</blockquote>\n';
17643     };
17644     
17645     Renderer.prototype.html = function(html) {
17646       return html;
17647     };
17648     
17649     Renderer.prototype.heading = function(text, level, raw) {
17650       return '<h'
17651         + level
17652         + ' id="'
17653         + this.options.headerPrefix
17654         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17655         + '">'
17656         + text
17657         + '</h'
17658         + level
17659         + '>\n';
17660     };
17661     
17662     Renderer.prototype.hr = function() {
17663       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17664     };
17665     
17666     Renderer.prototype.list = function(body, ordered) {
17667       var type = ordered ? 'ol' : 'ul';
17668       return '<' + type + '>\n' + body + '</' + type + '>\n';
17669     };
17670     
17671     Renderer.prototype.listitem = function(text) {
17672       return '<li>' + text + '</li>\n';
17673     };
17674     
17675     Renderer.prototype.paragraph = function(text) {
17676       return '<p>' + text + '</p>\n';
17677     };
17678     
17679     Renderer.prototype.table = function(header, body) {
17680       return '<table class="table table-striped">\n'
17681         + '<thead>\n'
17682         + header
17683         + '</thead>\n'
17684         + '<tbody>\n'
17685         + body
17686         + '</tbody>\n'
17687         + '</table>\n';
17688     };
17689     
17690     Renderer.prototype.tablerow = function(content) {
17691       return '<tr>\n' + content + '</tr>\n';
17692     };
17693     
17694     Renderer.prototype.tablecell = function(content, flags) {
17695       var type = flags.header ? 'th' : 'td';
17696       var tag = flags.align
17697         ? '<' + type + ' style="text-align:' + flags.align + '">'
17698         : '<' + type + '>';
17699       return tag + content + '</' + type + '>\n';
17700     };
17701     
17702     // span level renderer
17703     Renderer.prototype.strong = function(text) {
17704       return '<strong>' + text + '</strong>';
17705     };
17706     
17707     Renderer.prototype.em = function(text) {
17708       return '<em>' + text + '</em>';
17709     };
17710     
17711     Renderer.prototype.codespan = function(text) {
17712       return '<code>' + text + '</code>';
17713     };
17714     
17715     Renderer.prototype.br = function() {
17716       return this.options.xhtml ? '<br/>' : '<br>';
17717     };
17718     
17719     Renderer.prototype.del = function(text) {
17720       return '<del>' + text + '</del>';
17721     };
17722     
17723     Renderer.prototype.link = function(href, title, text) {
17724       if (this.options.sanitize) {
17725         try {
17726           var prot = decodeURIComponent(unescape(href))
17727             .replace(/[^\w:]/g, '')
17728             .toLowerCase();
17729         } catch (e) {
17730           return '';
17731         }
17732         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17733           return '';
17734         }
17735       }
17736       var out = '<a href="' + href + '"';
17737       if (title) {
17738         out += ' title="' + title + '"';
17739       }
17740       out += '>' + text + '</a>';
17741       return out;
17742     };
17743     
17744     Renderer.prototype.image = function(href, title, text) {
17745       var out = '<img src="' + href + '" alt="' + text + '"';
17746       if (title) {
17747         out += ' title="' + title + '"';
17748       }
17749       out += this.options.xhtml ? '/>' : '>';
17750       return out;
17751     };
17752     
17753     Renderer.prototype.text = function(text) {
17754       return text;
17755     };
17756     
17757     /**
17758      * Parsing & Compiling
17759      */
17760          /**
17761          * eval:var:Parser
17762     */
17763     
17764     var Parser= function (options) {
17765       this.tokens = [];
17766       this.token = null;
17767       this.options = options || marked.defaults;
17768       this.options.renderer = this.options.renderer || new Renderer;
17769       this.renderer = this.options.renderer;
17770       this.renderer.options = this.options;
17771     }
17772     
17773     /**
17774      * Static Parse Method
17775      */
17776     
17777     Parser.parse = function(src, options, renderer) {
17778       var parser = new Parser(options, renderer);
17779       return parser.parse(src);
17780     };
17781     
17782     /**
17783      * Parse Loop
17784      */
17785     
17786     Parser.prototype.parse = function(src) {
17787       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17788       this.tokens = src.reverse();
17789     
17790       var out = '';
17791       while (this.next()) {
17792         out += this.tok();
17793       }
17794     
17795       return out;
17796     };
17797     
17798     /**
17799      * Next Token
17800      */
17801     
17802     Parser.prototype.next = function() {
17803       return this.token = this.tokens.pop();
17804     };
17805     
17806     /**
17807      * Preview Next Token
17808      */
17809     
17810     Parser.prototype.peek = function() {
17811       return this.tokens[this.tokens.length - 1] || 0;
17812     };
17813     
17814     /**
17815      * Parse Text Tokens
17816      */
17817     
17818     Parser.prototype.parseText = function() {
17819       var body = this.token.text;
17820     
17821       while (this.peek().type === 'text') {
17822         body += '\n' + this.next().text;
17823       }
17824     
17825       return this.inline.output(body);
17826     };
17827     
17828     /**
17829      * Parse Current Token
17830      */
17831     
17832     Parser.prototype.tok = function() {
17833       switch (this.token.type) {
17834         case 'space': {
17835           return '';
17836         }
17837         case 'hr': {
17838           return this.renderer.hr();
17839         }
17840         case 'heading': {
17841           return this.renderer.heading(
17842             this.inline.output(this.token.text),
17843             this.token.depth,
17844             this.token.text);
17845         }
17846         case 'code': {
17847           return this.renderer.code(this.token.text,
17848             this.token.lang,
17849             this.token.escaped);
17850         }
17851         case 'table': {
17852           var header = ''
17853             , body = ''
17854             , i
17855             , row
17856             , cell
17857             , flags
17858             , j;
17859     
17860           // header
17861           cell = '';
17862           for (i = 0; i < this.token.header.length; i++) {
17863             flags = { header: true, align: this.token.align[i] };
17864             cell += this.renderer.tablecell(
17865               this.inline.output(this.token.header[i]),
17866               { header: true, align: this.token.align[i] }
17867             );
17868           }
17869           header += this.renderer.tablerow(cell);
17870     
17871           for (i = 0; i < this.token.cells.length; i++) {
17872             row = this.token.cells[i];
17873     
17874             cell = '';
17875             for (j = 0; j < row.length; j++) {
17876               cell += this.renderer.tablecell(
17877                 this.inline.output(row[j]),
17878                 { header: false, align: this.token.align[j] }
17879               );
17880             }
17881     
17882             body += this.renderer.tablerow(cell);
17883           }
17884           return this.renderer.table(header, body);
17885         }
17886         case 'blockquote_start': {
17887           var body = '';
17888     
17889           while (this.next().type !== 'blockquote_end') {
17890             body += this.tok();
17891           }
17892     
17893           return this.renderer.blockquote(body);
17894         }
17895         case 'list_start': {
17896           var body = ''
17897             , ordered = this.token.ordered;
17898     
17899           while (this.next().type !== 'list_end') {
17900             body += this.tok();
17901           }
17902     
17903           return this.renderer.list(body, ordered);
17904         }
17905         case 'list_item_start': {
17906           var body = '';
17907     
17908           while (this.next().type !== 'list_item_end') {
17909             body += this.token.type === 'text'
17910               ? this.parseText()
17911               : this.tok();
17912           }
17913     
17914           return this.renderer.listitem(body);
17915         }
17916         case 'loose_item_start': {
17917           var body = '';
17918     
17919           while (this.next().type !== 'list_item_end') {
17920             body += this.tok();
17921           }
17922     
17923           return this.renderer.listitem(body);
17924         }
17925         case 'html': {
17926           var html = !this.token.pre && !this.options.pedantic
17927             ? this.inline.output(this.token.text)
17928             : this.token.text;
17929           return this.renderer.html(html);
17930         }
17931         case 'paragraph': {
17932           return this.renderer.paragraph(this.inline.output(this.token.text));
17933         }
17934         case 'text': {
17935           return this.renderer.paragraph(this.parseText());
17936         }
17937       }
17938     };
17939   
17940          /**
17941          * eval:var:noop
17942     */
17943     var noop = function () {}
17944     noop.exec = noop;
17945     
17946          /**
17947          * eval:var:merge
17948     */
17949     var merge = function (obj) {
17950       var i = 1
17951         , target
17952         , key;
17953     
17954       for (; i < arguments.length; i++) {
17955         target = arguments[i];
17956         for (key in target) {
17957           if (Object.prototype.hasOwnProperty.call(target, key)) {
17958             obj[key] = target[key];
17959           }
17960         }
17961       }
17962     
17963       return obj;
17964     }
17965     
17966     
17967     /**
17968      * Marked
17969      */
17970     
17971     var marked = function (src, opt, callback) {
17972       if (callback || typeof opt === 'function') {
17973         if (!callback) {
17974           callback = opt;
17975           opt = null;
17976         }
17977     
17978         opt = merge({}, marked.defaults, opt || {});
17979     
17980         var highlight = opt.highlight
17981           , tokens
17982           , pending
17983           , i = 0;
17984     
17985         try {
17986           tokens = Lexer.lex(src, opt)
17987         } catch (e) {
17988           return callback(e);
17989         }
17990     
17991         pending = tokens.length;
17992     
17993         var done = function(err) {
17994           if (err) {
17995             opt.highlight = highlight;
17996             return callback(err);
17997           }
17998     
17999           var out;
18000     
18001           try {
18002             out = Parser.parse(tokens, opt);
18003           } catch (e) {
18004             err = e;
18005           }
18006     
18007           opt.highlight = highlight;
18008     
18009           return err
18010             ? callback(err)
18011             : callback(null, out);
18012         };
18013     
18014         if (!highlight || highlight.length < 3) {
18015           return done();
18016         }
18017     
18018         delete opt.highlight;
18019     
18020         if (!pending) { return done(); }
18021     
18022         for (; i < tokens.length; i++) {
18023           (function(token) {
18024             if (token.type !== 'code') {
18025               return --pending || done();
18026             }
18027             return highlight(token.text, token.lang, function(err, code) {
18028               if (err) { return done(err); }
18029               if (code == null || code === token.text) {
18030                 return --pending || done();
18031               }
18032               token.text = code;
18033               token.escaped = true;
18034               --pending || done();
18035             });
18036           })(tokens[i]);
18037         }
18038     
18039         return;
18040       }
18041       try {
18042         if (opt) { opt = merge({}, marked.defaults, opt); }
18043         return Parser.parse(Lexer.lex(src, opt), opt);
18044       } catch (e) {
18045         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18046         if ((opt || marked.defaults).silent) {
18047           return '<p>An error occured:</p><pre>'
18048             + escape(e.message + '', true)
18049             + '</pre>';
18050         }
18051         throw e;
18052       }
18053     }
18054     
18055     /**
18056      * Options
18057      */
18058     
18059     marked.options =
18060     marked.setOptions = function(opt) {
18061       merge(marked.defaults, opt);
18062       return marked;
18063     };
18064     
18065     marked.defaults = {
18066       gfm: true,
18067       tables: true,
18068       breaks: false,
18069       pedantic: false,
18070       sanitize: false,
18071       sanitizer: null,
18072       mangle: true,
18073       smartLists: false,
18074       silent: false,
18075       highlight: null,
18076       langPrefix: 'lang-',
18077       smartypants: false,
18078       headerPrefix: '',
18079       renderer: new Renderer,
18080       xhtml: false
18081     };
18082     
18083     /**
18084      * Expose
18085      */
18086     
18087     marked.Parser = Parser;
18088     marked.parser = Parser.parse;
18089     
18090     marked.Renderer = Renderer;
18091     
18092     marked.Lexer = Lexer;
18093     marked.lexer = Lexer.lex;
18094     
18095     marked.InlineLexer = InlineLexer;
18096     marked.inlineLexer = InlineLexer.output;
18097     
18098     marked.parse = marked;
18099     
18100     Roo.Markdown.marked = marked;
18101
18102 })();/*
18103  * Based on:
18104  * Ext JS Library 1.1.1
18105  * Copyright(c) 2006-2007, Ext JS, LLC.
18106  *
18107  * Originally Released Under LGPL - original licence link has changed is not relivant.
18108  *
18109  * Fork - LGPL
18110  * <script type="text/javascript">
18111  */
18112
18113
18114
18115 /*
18116  * These classes are derivatives of the similarly named classes in the YUI Library.
18117  * The original license:
18118  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18119  * Code licensed under the BSD License:
18120  * http://developer.yahoo.net/yui/license.txt
18121  */
18122
18123 (function() {
18124
18125 var Event=Roo.EventManager;
18126 var Dom=Roo.lib.Dom;
18127
18128 /**
18129  * @class Roo.dd.DragDrop
18130  * @extends Roo.util.Observable
18131  * Defines the interface and base operation of items that that can be
18132  * dragged or can be drop targets.  It was designed to be extended, overriding
18133  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18134  * Up to three html elements can be associated with a DragDrop instance:
18135  * <ul>
18136  * <li>linked element: the element that is passed into the constructor.
18137  * This is the element which defines the boundaries for interaction with
18138  * other DragDrop objects.</li>
18139  * <li>handle element(s): The drag operation only occurs if the element that
18140  * was clicked matches a handle element.  By default this is the linked
18141  * element, but there are times that you will want only a portion of the
18142  * linked element to initiate the drag operation, and the setHandleElId()
18143  * method provides a way to define this.</li>
18144  * <li>drag element: this represents the element that would be moved along
18145  * with the cursor during a drag operation.  By default, this is the linked
18146  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18147  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18148  * </li>
18149  * </ul>
18150  * This class should not be instantiated until the onload event to ensure that
18151  * the associated elements are available.
18152  * The following would define a DragDrop obj that would interact with any
18153  * other DragDrop obj in the "group1" group:
18154  * <pre>
18155  *  dd = new Roo.dd.DragDrop("div1", "group1");
18156  * </pre>
18157  * Since none of the event handlers have been implemented, nothing would
18158  * actually happen if you were to run the code above.  Normally you would
18159  * override this class or one of the default implementations, but you can
18160  * also override the methods you want on an instance of the class...
18161  * <pre>
18162  *  dd.onDragDrop = function(e, id) {
18163  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18164  *  }
18165  * </pre>
18166  * @constructor
18167  * @param {String} id of the element that is linked to this instance
18168  * @param {String} sGroup the group of related DragDrop objects
18169  * @param {object} config an object containing configurable attributes
18170  *                Valid properties for DragDrop:
18171  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18172  */
18173 Roo.dd.DragDrop = function(id, sGroup, config) {
18174     if (id) {
18175         this.init(id, sGroup, config);
18176     }
18177     
18178 };
18179
18180 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18181
18182     /**
18183      * The id of the element associated with this object.  This is what we
18184      * refer to as the "linked element" because the size and position of
18185      * this element is used to determine when the drag and drop objects have
18186      * interacted.
18187      * @property id
18188      * @type String
18189      */
18190     id: null,
18191
18192     /**
18193      * Configuration attributes passed into the constructor
18194      * @property config
18195      * @type object
18196      */
18197     config: null,
18198
18199     /**
18200      * The id of the element that will be dragged.  By default this is same
18201      * as the linked element , but could be changed to another element. Ex:
18202      * Roo.dd.DDProxy
18203      * @property dragElId
18204      * @type String
18205      * @private
18206      */
18207     dragElId: null,
18208
18209     /**
18210      * the id of the element that initiates the drag operation.  By default
18211      * this is the linked element, but could be changed to be a child of this
18212      * element.  This lets us do things like only starting the drag when the
18213      * header element within the linked html element is clicked.
18214      * @property handleElId
18215      * @type String
18216      * @private
18217      */
18218     handleElId: null,
18219
18220     /**
18221      * An associative array of HTML tags that will be ignored if clicked.
18222      * @property invalidHandleTypes
18223      * @type {string: string}
18224      */
18225     invalidHandleTypes: null,
18226
18227     /**
18228      * An associative array of ids for elements that will be ignored if clicked
18229      * @property invalidHandleIds
18230      * @type {string: string}
18231      */
18232     invalidHandleIds: null,
18233
18234     /**
18235      * An indexted array of css class names for elements that will be ignored
18236      * if clicked.
18237      * @property invalidHandleClasses
18238      * @type string[]
18239      */
18240     invalidHandleClasses: null,
18241
18242     /**
18243      * The linked element's absolute X position at the time the drag was
18244      * started
18245      * @property startPageX
18246      * @type int
18247      * @private
18248      */
18249     startPageX: 0,
18250
18251     /**
18252      * The linked element's absolute X position at the time the drag was
18253      * started
18254      * @property startPageY
18255      * @type int
18256      * @private
18257      */
18258     startPageY: 0,
18259
18260     /**
18261      * The group defines a logical collection of DragDrop objects that are
18262      * related.  Instances only get events when interacting with other
18263      * DragDrop object in the same group.  This lets us define multiple
18264      * groups using a single DragDrop subclass if we want.
18265      * @property groups
18266      * @type {string: string}
18267      */
18268     groups: null,
18269
18270     /**
18271      * Individual drag/drop instances can be locked.  This will prevent
18272      * onmousedown start drag.
18273      * @property locked
18274      * @type boolean
18275      * @private
18276      */
18277     locked: false,
18278
18279     /**
18280      * Lock this instance
18281      * @method lock
18282      */
18283     lock: function() { this.locked = true; },
18284
18285     /**
18286      * Unlock this instace
18287      * @method unlock
18288      */
18289     unlock: function() { this.locked = false; },
18290
18291     /**
18292      * By default, all insances can be a drop target.  This can be disabled by
18293      * setting isTarget to false.
18294      * @method isTarget
18295      * @type boolean
18296      */
18297     isTarget: true,
18298
18299     /**
18300      * The padding configured for this drag and drop object for calculating
18301      * the drop zone intersection with this object.
18302      * @method padding
18303      * @type int[]
18304      */
18305     padding: null,
18306
18307     /**
18308      * Cached reference to the linked element
18309      * @property _domRef
18310      * @private
18311      */
18312     _domRef: null,
18313
18314     /**
18315      * Internal typeof flag
18316      * @property __ygDragDrop
18317      * @private
18318      */
18319     __ygDragDrop: true,
18320
18321     /**
18322      * Set to true when horizontal contraints are applied
18323      * @property constrainX
18324      * @type boolean
18325      * @private
18326      */
18327     constrainX: false,
18328
18329     /**
18330      * Set to true when vertical contraints are applied
18331      * @property constrainY
18332      * @type boolean
18333      * @private
18334      */
18335     constrainY: false,
18336
18337     /**
18338      * The left constraint
18339      * @property minX
18340      * @type int
18341      * @private
18342      */
18343     minX: 0,
18344
18345     /**
18346      * The right constraint
18347      * @property maxX
18348      * @type int
18349      * @private
18350      */
18351     maxX: 0,
18352
18353     /**
18354      * The up constraint
18355      * @property minY
18356      * @type int
18357      * @type int
18358      * @private
18359      */
18360     minY: 0,
18361
18362     /**
18363      * The down constraint
18364      * @property maxY
18365      * @type int
18366      * @private
18367      */
18368     maxY: 0,
18369
18370     /**
18371      * Maintain offsets when we resetconstraints.  Set to true when you want
18372      * the position of the element relative to its parent to stay the same
18373      * when the page changes
18374      *
18375      * @property maintainOffset
18376      * @type boolean
18377      */
18378     maintainOffset: false,
18379
18380     /**
18381      * Array of pixel locations the element will snap to if we specified a
18382      * horizontal graduation/interval.  This array is generated automatically
18383      * when you define a tick interval.
18384      * @property xTicks
18385      * @type int[]
18386      */
18387     xTicks: null,
18388
18389     /**
18390      * Array of pixel locations the element will snap to if we specified a
18391      * vertical graduation/interval.  This array is generated automatically
18392      * when you define a tick interval.
18393      * @property yTicks
18394      * @type int[]
18395      */
18396     yTicks: null,
18397
18398     /**
18399      * By default the drag and drop instance will only respond to the primary
18400      * button click (left button for a right-handed mouse).  Set to true to
18401      * allow drag and drop to start with any mouse click that is propogated
18402      * by the browser
18403      * @property primaryButtonOnly
18404      * @type boolean
18405      */
18406     primaryButtonOnly: true,
18407
18408     /**
18409      * The availabe property is false until the linked dom element is accessible.
18410      * @property available
18411      * @type boolean
18412      */
18413     available: false,
18414
18415     /**
18416      * By default, drags can only be initiated if the mousedown occurs in the
18417      * region the linked element is.  This is done in part to work around a
18418      * bug in some browsers that mis-report the mousedown if the previous
18419      * mouseup happened outside of the window.  This property is set to true
18420      * if outer handles are defined.
18421      *
18422      * @property hasOuterHandles
18423      * @type boolean
18424      * @default false
18425      */
18426     hasOuterHandles: false,
18427
18428     /**
18429      * Code that executes immediately before the startDrag event
18430      * @method b4StartDrag
18431      * @private
18432      */
18433     b4StartDrag: function(x, y) { },
18434
18435     /**
18436      * Abstract method called after a drag/drop object is clicked
18437      * and the drag or mousedown time thresholds have beeen met.
18438      * @method startDrag
18439      * @param {int} X click location
18440      * @param {int} Y click location
18441      */
18442     startDrag: function(x, y) { /* override this */ },
18443
18444     /**
18445      * Code that executes immediately before the onDrag event
18446      * @method b4Drag
18447      * @private
18448      */
18449     b4Drag: function(e) { },
18450
18451     /**
18452      * Abstract method called during the onMouseMove event while dragging an
18453      * object.
18454      * @method onDrag
18455      * @param {Event} e the mousemove event
18456      */
18457     onDrag: function(e) { /* override this */ },
18458
18459     /**
18460      * Abstract method called when this element fist begins hovering over
18461      * another DragDrop obj
18462      * @method onDragEnter
18463      * @param {Event} e the mousemove event
18464      * @param {String|DragDrop[]} id In POINT mode, the element
18465      * id this is hovering over.  In INTERSECT mode, an array of one or more
18466      * dragdrop items being hovered over.
18467      */
18468     onDragEnter: function(e, id) { /* override this */ },
18469
18470     /**
18471      * Code that executes immediately before the onDragOver event
18472      * @method b4DragOver
18473      * @private
18474      */
18475     b4DragOver: function(e) { },
18476
18477     /**
18478      * Abstract method called when this element is hovering over another
18479      * DragDrop obj
18480      * @method onDragOver
18481      * @param {Event} e the mousemove event
18482      * @param {String|DragDrop[]} id In POINT mode, the element
18483      * id this is hovering over.  In INTERSECT mode, an array of dd items
18484      * being hovered over.
18485      */
18486     onDragOver: function(e, id) { /* override this */ },
18487
18488     /**
18489      * Code that executes immediately before the onDragOut event
18490      * @method b4DragOut
18491      * @private
18492      */
18493     b4DragOut: function(e) { },
18494
18495     /**
18496      * Abstract method called when we are no longer hovering over an element
18497      * @method onDragOut
18498      * @param {Event} e the mousemove event
18499      * @param {String|DragDrop[]} id In POINT mode, the element
18500      * id this was hovering over.  In INTERSECT mode, an array of dd items
18501      * that the mouse is no longer over.
18502      */
18503     onDragOut: function(e, id) { /* override this */ },
18504
18505     /**
18506      * Code that executes immediately before the onDragDrop event
18507      * @method b4DragDrop
18508      * @private
18509      */
18510     b4DragDrop: function(e) { },
18511
18512     /**
18513      * Abstract method called when this item is dropped on another DragDrop
18514      * obj
18515      * @method onDragDrop
18516      * @param {Event} e the mouseup event
18517      * @param {String|DragDrop[]} id In POINT mode, the element
18518      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18519      * was dropped on.
18520      */
18521     onDragDrop: function(e, id) { /* override this */ },
18522
18523     /**
18524      * Abstract method called when this item is dropped on an area with no
18525      * drop target
18526      * @method onInvalidDrop
18527      * @param {Event} e the mouseup event
18528      */
18529     onInvalidDrop: function(e) { /* override this */ },
18530
18531     /**
18532      * Code that executes immediately before the endDrag event
18533      * @method b4EndDrag
18534      * @private
18535      */
18536     b4EndDrag: function(e) { },
18537
18538     /**
18539      * Fired when we are done dragging the object
18540      * @method endDrag
18541      * @param {Event} e the mouseup event
18542      */
18543     endDrag: function(e) { /* override this */ },
18544
18545     /**
18546      * Code executed immediately before the onMouseDown event
18547      * @method b4MouseDown
18548      * @param {Event} e the mousedown event
18549      * @private
18550      */
18551     b4MouseDown: function(e) {  },
18552
18553     /**
18554      * Event handler that fires when a drag/drop obj gets a mousedown
18555      * @method onMouseDown
18556      * @param {Event} e the mousedown event
18557      */
18558     onMouseDown: function(e) { /* override this */ },
18559
18560     /**
18561      * Event handler that fires when a drag/drop obj gets a mouseup
18562      * @method onMouseUp
18563      * @param {Event} e the mouseup event
18564      */
18565     onMouseUp: function(e) { /* override this */ },
18566
18567     /**
18568      * Override the onAvailable method to do what is needed after the initial
18569      * position was determined.
18570      * @method onAvailable
18571      */
18572     onAvailable: function () {
18573     },
18574
18575     /*
18576      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18577      * @type Object
18578      */
18579     defaultPadding : {left:0, right:0, top:0, bottom:0},
18580
18581     /*
18582      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18583  *
18584  * Usage:
18585  <pre><code>
18586  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18587                 { dragElId: "existingProxyDiv" });
18588  dd.startDrag = function(){
18589      this.constrainTo("parent-id");
18590  };
18591  </code></pre>
18592  * Or you can initalize it using the {@link Roo.Element} object:
18593  <pre><code>
18594  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18595      startDrag : function(){
18596          this.constrainTo("parent-id");
18597      }
18598  });
18599  </code></pre>
18600      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18601      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18602      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18603      * an object containing the sides to pad. For example: {right:10, bottom:10}
18604      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18605      */
18606     constrainTo : function(constrainTo, pad, inContent){
18607         if(typeof pad == "number"){
18608             pad = {left: pad, right:pad, top:pad, bottom:pad};
18609         }
18610         pad = pad || this.defaultPadding;
18611         var b = Roo.get(this.getEl()).getBox();
18612         var ce = Roo.get(constrainTo);
18613         var s = ce.getScroll();
18614         var c, cd = ce.dom;
18615         if(cd == document.body){
18616             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18617         }else{
18618             xy = ce.getXY();
18619             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18620         }
18621
18622
18623         var topSpace = b.y - c.y;
18624         var leftSpace = b.x - c.x;
18625
18626         this.resetConstraints();
18627         this.setXConstraint(leftSpace - (pad.left||0), // left
18628                 c.width - leftSpace - b.width - (pad.right||0) //right
18629         );
18630         this.setYConstraint(topSpace - (pad.top||0), //top
18631                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18632         );
18633     },
18634
18635     /**
18636      * Returns a reference to the linked element
18637      * @method getEl
18638      * @return {HTMLElement} the html element
18639      */
18640     getEl: function() {
18641         if (!this._domRef) {
18642             this._domRef = Roo.getDom(this.id);
18643         }
18644
18645         return this._domRef;
18646     },
18647
18648     /**
18649      * Returns a reference to the actual element to drag.  By default this is
18650      * the same as the html element, but it can be assigned to another
18651      * element. An example of this can be found in Roo.dd.DDProxy
18652      * @method getDragEl
18653      * @return {HTMLElement} the html element
18654      */
18655     getDragEl: function() {
18656         return Roo.getDom(this.dragElId);
18657     },
18658
18659     /**
18660      * Sets up the DragDrop object.  Must be called in the constructor of any
18661      * Roo.dd.DragDrop subclass
18662      * @method init
18663      * @param id the id of the linked element
18664      * @param {String} sGroup the group of related items
18665      * @param {object} config configuration attributes
18666      */
18667     init: function(id, sGroup, config) {
18668         this.initTarget(id, sGroup, config);
18669         if (!Roo.isTouch) {
18670             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18671         }
18672         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18673         // Event.on(this.id, "selectstart", Event.preventDefault);
18674     },
18675
18676     /**
18677      * Initializes Targeting functionality only... the object does not
18678      * get a mousedown handler.
18679      * @method initTarget
18680      * @param id the id of the linked element
18681      * @param {String} sGroup the group of related items
18682      * @param {object} config configuration attributes
18683      */
18684     initTarget: function(id, sGroup, config) {
18685
18686         // configuration attributes
18687         this.config = config || {};
18688
18689         // create a local reference to the drag and drop manager
18690         this.DDM = Roo.dd.DDM;
18691         // initialize the groups array
18692         this.groups = {};
18693
18694         // assume that we have an element reference instead of an id if the
18695         // parameter is not a string
18696         if (typeof id !== "string") {
18697             id = Roo.id(id);
18698         }
18699
18700         // set the id
18701         this.id = id;
18702
18703         // add to an interaction group
18704         this.addToGroup((sGroup) ? sGroup : "default");
18705
18706         // We don't want to register this as the handle with the manager
18707         // so we just set the id rather than calling the setter.
18708         this.handleElId = id;
18709
18710         // the linked element is the element that gets dragged by default
18711         this.setDragElId(id);
18712
18713         // by default, clicked anchors will not start drag operations.
18714         this.invalidHandleTypes = { A: "A" };
18715         this.invalidHandleIds = {};
18716         this.invalidHandleClasses = [];
18717
18718         this.applyConfig();
18719
18720         this.handleOnAvailable();
18721     },
18722
18723     /**
18724      * Applies the configuration parameters that were passed into the constructor.
18725      * This is supposed to happen at each level through the inheritance chain.  So
18726      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18727      * DragDrop in order to get all of the parameters that are available in
18728      * each object.
18729      * @method applyConfig
18730      */
18731     applyConfig: function() {
18732
18733         // configurable properties:
18734         //    padding, isTarget, maintainOffset, primaryButtonOnly
18735         this.padding           = this.config.padding || [0, 0, 0, 0];
18736         this.isTarget          = (this.config.isTarget !== false);
18737         this.maintainOffset    = (this.config.maintainOffset);
18738         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18739
18740     },
18741
18742     /**
18743      * Executed when the linked element is available
18744      * @method handleOnAvailable
18745      * @private
18746      */
18747     handleOnAvailable: function() {
18748         this.available = true;
18749         this.resetConstraints();
18750         this.onAvailable();
18751     },
18752
18753      /**
18754      * Configures the padding for the target zone in px.  Effectively expands
18755      * (or reduces) the virtual object size for targeting calculations.
18756      * Supports css-style shorthand; if only one parameter is passed, all sides
18757      * will have that padding, and if only two are passed, the top and bottom
18758      * will have the first param, the left and right the second.
18759      * @method setPadding
18760      * @param {int} iTop    Top pad
18761      * @param {int} iRight  Right pad
18762      * @param {int} iBot    Bot pad
18763      * @param {int} iLeft   Left pad
18764      */
18765     setPadding: function(iTop, iRight, iBot, iLeft) {
18766         // this.padding = [iLeft, iRight, iTop, iBot];
18767         if (!iRight && 0 !== iRight) {
18768             this.padding = [iTop, iTop, iTop, iTop];
18769         } else if (!iBot && 0 !== iBot) {
18770             this.padding = [iTop, iRight, iTop, iRight];
18771         } else {
18772             this.padding = [iTop, iRight, iBot, iLeft];
18773         }
18774     },
18775
18776     /**
18777      * Stores the initial placement of the linked element.
18778      * @method setInitialPosition
18779      * @param {int} diffX   the X offset, default 0
18780      * @param {int} diffY   the Y offset, default 0
18781      */
18782     setInitPosition: function(diffX, diffY) {
18783         var el = this.getEl();
18784
18785         if (!this.DDM.verifyEl(el)) {
18786             return;
18787         }
18788
18789         var dx = diffX || 0;
18790         var dy = diffY || 0;
18791
18792         var p = Dom.getXY( el );
18793
18794         this.initPageX = p[0] - dx;
18795         this.initPageY = p[1] - dy;
18796
18797         this.lastPageX = p[0];
18798         this.lastPageY = p[1];
18799
18800
18801         this.setStartPosition(p);
18802     },
18803
18804     /**
18805      * Sets the start position of the element.  This is set when the obj
18806      * is initialized, the reset when a drag is started.
18807      * @method setStartPosition
18808      * @param pos current position (from previous lookup)
18809      * @private
18810      */
18811     setStartPosition: function(pos) {
18812         var p = pos || Dom.getXY( this.getEl() );
18813         this.deltaSetXY = null;
18814
18815         this.startPageX = p[0];
18816         this.startPageY = p[1];
18817     },
18818
18819     /**
18820      * Add this instance to a group of related drag/drop objects.  All
18821      * instances belong to at least one group, and can belong to as many
18822      * groups as needed.
18823      * @method addToGroup
18824      * @param sGroup {string} the name of the group
18825      */
18826     addToGroup: function(sGroup) {
18827         this.groups[sGroup] = true;
18828         this.DDM.regDragDrop(this, sGroup);
18829     },
18830
18831     /**
18832      * Remove's this instance from the supplied interaction group
18833      * @method removeFromGroup
18834      * @param {string}  sGroup  The group to drop
18835      */
18836     removeFromGroup: function(sGroup) {
18837         if (this.groups[sGroup]) {
18838             delete this.groups[sGroup];
18839         }
18840
18841         this.DDM.removeDDFromGroup(this, sGroup);
18842     },
18843
18844     /**
18845      * Allows you to specify that an element other than the linked element
18846      * will be moved with the cursor during a drag
18847      * @method setDragElId
18848      * @param id {string} the id of the element that will be used to initiate the drag
18849      */
18850     setDragElId: function(id) {
18851         this.dragElId = id;
18852     },
18853
18854     /**
18855      * Allows you to specify a child of the linked element that should be
18856      * used to initiate the drag operation.  An example of this would be if
18857      * you have a content div with text and links.  Clicking anywhere in the
18858      * content area would normally start the drag operation.  Use this method
18859      * to specify that an element inside of the content div is the element
18860      * that starts the drag operation.
18861      * @method setHandleElId
18862      * @param id {string} the id of the element that will be used to
18863      * initiate the drag.
18864      */
18865     setHandleElId: function(id) {
18866         if (typeof id !== "string") {
18867             id = Roo.id(id);
18868         }
18869         this.handleElId = id;
18870         this.DDM.regHandle(this.id, id);
18871     },
18872
18873     /**
18874      * Allows you to set an element outside of the linked element as a drag
18875      * handle
18876      * @method setOuterHandleElId
18877      * @param id the id of the element that will be used to initiate the drag
18878      */
18879     setOuterHandleElId: function(id) {
18880         if (typeof id !== "string") {
18881             id = Roo.id(id);
18882         }
18883         Event.on(id, "mousedown",
18884                 this.handleMouseDown, this);
18885         this.setHandleElId(id);
18886
18887         this.hasOuterHandles = true;
18888     },
18889
18890     /**
18891      * Remove all drag and drop hooks for this element
18892      * @method unreg
18893      */
18894     unreg: function() {
18895         Event.un(this.id, "mousedown",
18896                 this.handleMouseDown);
18897         Event.un(this.id, "touchstart",
18898                 this.handleMouseDown);
18899         this._domRef = null;
18900         this.DDM._remove(this);
18901     },
18902
18903     destroy : function(){
18904         this.unreg();
18905     },
18906
18907     /**
18908      * Returns true if this instance is locked, or the drag drop mgr is locked
18909      * (meaning that all drag/drop is disabled on the page.)
18910      * @method isLocked
18911      * @return {boolean} true if this obj or all drag/drop is locked, else
18912      * false
18913      */
18914     isLocked: function() {
18915         return (this.DDM.isLocked() || this.locked);
18916     },
18917
18918     /**
18919      * Fired when this object is clicked
18920      * @method handleMouseDown
18921      * @param {Event} e
18922      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18923      * @private
18924      */
18925     handleMouseDown: function(e, oDD){
18926      
18927         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18928             //Roo.log('not touch/ button !=0');
18929             return;
18930         }
18931         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18932             return; // double touch..
18933         }
18934         
18935
18936         if (this.isLocked()) {
18937             //Roo.log('locked');
18938             return;
18939         }
18940
18941         this.DDM.refreshCache(this.groups);
18942 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18943         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18944         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18945             //Roo.log('no outer handes or not over target');
18946                 // do nothing.
18947         } else {
18948 //            Roo.log('check validator');
18949             if (this.clickValidator(e)) {
18950 //                Roo.log('validate success');
18951                 // set the initial element position
18952                 this.setStartPosition();
18953
18954
18955                 this.b4MouseDown(e);
18956                 this.onMouseDown(e);
18957
18958                 this.DDM.handleMouseDown(e, this);
18959
18960                 this.DDM.stopEvent(e);
18961             } else {
18962
18963
18964             }
18965         }
18966     },
18967
18968     clickValidator: function(e) {
18969         var target = e.getTarget();
18970         return ( this.isValidHandleChild(target) &&
18971                     (this.id == this.handleElId ||
18972                         this.DDM.handleWasClicked(target, this.id)) );
18973     },
18974
18975     /**
18976      * Allows you to specify a tag name that should not start a drag operation
18977      * when clicked.  This is designed to facilitate embedding links within a
18978      * drag handle that do something other than start the drag.
18979      * @method addInvalidHandleType
18980      * @param {string} tagName the type of element to exclude
18981      */
18982     addInvalidHandleType: function(tagName) {
18983         var type = tagName.toUpperCase();
18984         this.invalidHandleTypes[type] = type;
18985     },
18986
18987     /**
18988      * Lets you to specify an element id for a child of a drag handle
18989      * that should not initiate a drag
18990      * @method addInvalidHandleId
18991      * @param {string} id the element id of the element you wish to ignore
18992      */
18993     addInvalidHandleId: function(id) {
18994         if (typeof id !== "string") {
18995             id = Roo.id(id);
18996         }
18997         this.invalidHandleIds[id] = id;
18998     },
18999
19000     /**
19001      * Lets you specify a css class of elements that will not initiate a drag
19002      * @method addInvalidHandleClass
19003      * @param {string} cssClass the class of the elements you wish to ignore
19004      */
19005     addInvalidHandleClass: function(cssClass) {
19006         this.invalidHandleClasses.push(cssClass);
19007     },
19008
19009     /**
19010      * Unsets an excluded tag name set by addInvalidHandleType
19011      * @method removeInvalidHandleType
19012      * @param {string} tagName the type of element to unexclude
19013      */
19014     removeInvalidHandleType: function(tagName) {
19015         var type = tagName.toUpperCase();
19016         // this.invalidHandleTypes[type] = null;
19017         delete this.invalidHandleTypes[type];
19018     },
19019
19020     /**
19021      * Unsets an invalid handle id
19022      * @method removeInvalidHandleId
19023      * @param {string} id the id of the element to re-enable
19024      */
19025     removeInvalidHandleId: function(id) {
19026         if (typeof id !== "string") {
19027             id = Roo.id(id);
19028         }
19029         delete this.invalidHandleIds[id];
19030     },
19031
19032     /**
19033      * Unsets an invalid css class
19034      * @method removeInvalidHandleClass
19035      * @param {string} cssClass the class of the element(s) you wish to
19036      * re-enable
19037      */
19038     removeInvalidHandleClass: function(cssClass) {
19039         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19040             if (this.invalidHandleClasses[i] == cssClass) {
19041                 delete this.invalidHandleClasses[i];
19042             }
19043         }
19044     },
19045
19046     /**
19047      * Checks the tag exclusion list to see if this click should be ignored
19048      * @method isValidHandleChild
19049      * @param {HTMLElement} node the HTMLElement to evaluate
19050      * @return {boolean} true if this is a valid tag type, false if not
19051      */
19052     isValidHandleChild: function(node) {
19053
19054         var valid = true;
19055         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19056         var nodeName;
19057         try {
19058             nodeName = node.nodeName.toUpperCase();
19059         } catch(e) {
19060             nodeName = node.nodeName;
19061         }
19062         valid = valid && !this.invalidHandleTypes[nodeName];
19063         valid = valid && !this.invalidHandleIds[node.id];
19064
19065         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19066             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19067         }
19068
19069
19070         return valid;
19071
19072     },
19073
19074     /**
19075      * Create the array of horizontal tick marks if an interval was specified
19076      * in setXConstraint().
19077      * @method setXTicks
19078      * @private
19079      */
19080     setXTicks: function(iStartX, iTickSize) {
19081         this.xTicks = [];
19082         this.xTickSize = iTickSize;
19083
19084         var tickMap = {};
19085
19086         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19087             if (!tickMap[i]) {
19088                 this.xTicks[this.xTicks.length] = i;
19089                 tickMap[i] = true;
19090             }
19091         }
19092
19093         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19094             if (!tickMap[i]) {
19095                 this.xTicks[this.xTicks.length] = i;
19096                 tickMap[i] = true;
19097             }
19098         }
19099
19100         this.xTicks.sort(this.DDM.numericSort) ;
19101     },
19102
19103     /**
19104      * Create the array of vertical tick marks if an interval was specified in
19105      * setYConstraint().
19106      * @method setYTicks
19107      * @private
19108      */
19109     setYTicks: function(iStartY, iTickSize) {
19110         this.yTicks = [];
19111         this.yTickSize = iTickSize;
19112
19113         var tickMap = {};
19114
19115         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19116             if (!tickMap[i]) {
19117                 this.yTicks[this.yTicks.length] = i;
19118                 tickMap[i] = true;
19119             }
19120         }
19121
19122         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19123             if (!tickMap[i]) {
19124                 this.yTicks[this.yTicks.length] = i;
19125                 tickMap[i] = true;
19126             }
19127         }
19128
19129         this.yTicks.sort(this.DDM.numericSort) ;
19130     },
19131
19132     /**
19133      * By default, the element can be dragged any place on the screen.  Use
19134      * this method to limit the horizontal travel of the element.  Pass in
19135      * 0,0 for the parameters if you want to lock the drag to the y axis.
19136      * @method setXConstraint
19137      * @param {int} iLeft the number of pixels the element can move to the left
19138      * @param {int} iRight the number of pixels the element can move to the
19139      * right
19140      * @param {int} iTickSize optional parameter for specifying that the
19141      * element
19142      * should move iTickSize pixels at a time.
19143      */
19144     setXConstraint: function(iLeft, iRight, iTickSize) {
19145         this.leftConstraint = iLeft;
19146         this.rightConstraint = iRight;
19147
19148         this.minX = this.initPageX - iLeft;
19149         this.maxX = this.initPageX + iRight;
19150         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19151
19152         this.constrainX = true;
19153     },
19154
19155     /**
19156      * Clears any constraints applied to this instance.  Also clears ticks
19157      * since they can't exist independent of a constraint at this time.
19158      * @method clearConstraints
19159      */
19160     clearConstraints: function() {
19161         this.constrainX = false;
19162         this.constrainY = false;
19163         this.clearTicks();
19164     },
19165
19166     /**
19167      * Clears any tick interval defined for this instance
19168      * @method clearTicks
19169      */
19170     clearTicks: function() {
19171         this.xTicks = null;
19172         this.yTicks = null;
19173         this.xTickSize = 0;
19174         this.yTickSize = 0;
19175     },
19176
19177     /**
19178      * By default, the element can be dragged any place on the screen.  Set
19179      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19180      * parameters if you want to lock the drag to the x axis.
19181      * @method setYConstraint
19182      * @param {int} iUp the number of pixels the element can move up
19183      * @param {int} iDown the number of pixels the element can move down
19184      * @param {int} iTickSize optional parameter for specifying that the
19185      * element should move iTickSize pixels at a time.
19186      */
19187     setYConstraint: function(iUp, iDown, iTickSize) {
19188         this.topConstraint = iUp;
19189         this.bottomConstraint = iDown;
19190
19191         this.minY = this.initPageY - iUp;
19192         this.maxY = this.initPageY + iDown;
19193         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19194
19195         this.constrainY = true;
19196
19197     },
19198
19199     /**
19200      * resetConstraints must be called if you manually reposition a dd element.
19201      * @method resetConstraints
19202      * @param {boolean} maintainOffset
19203      */
19204     resetConstraints: function() {
19205
19206
19207         // Maintain offsets if necessary
19208         if (this.initPageX || this.initPageX === 0) {
19209             // figure out how much this thing has moved
19210             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19211             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19212
19213             this.setInitPosition(dx, dy);
19214
19215         // This is the first time we have detected the element's position
19216         } else {
19217             this.setInitPosition();
19218         }
19219
19220         if (this.constrainX) {
19221             this.setXConstraint( this.leftConstraint,
19222                                  this.rightConstraint,
19223                                  this.xTickSize        );
19224         }
19225
19226         if (this.constrainY) {
19227             this.setYConstraint( this.topConstraint,
19228                                  this.bottomConstraint,
19229                                  this.yTickSize         );
19230         }
19231     },
19232
19233     /**
19234      * Normally the drag element is moved pixel by pixel, but we can specify
19235      * that it move a number of pixels at a time.  This method resolves the
19236      * location when we have it set up like this.
19237      * @method getTick
19238      * @param {int} val where we want to place the object
19239      * @param {int[]} tickArray sorted array of valid points
19240      * @return {int} the closest tick
19241      * @private
19242      */
19243     getTick: function(val, tickArray) {
19244
19245         if (!tickArray) {
19246             // If tick interval is not defined, it is effectively 1 pixel,
19247             // so we return the value passed to us.
19248             return val;
19249         } else if (tickArray[0] >= val) {
19250             // The value is lower than the first tick, so we return the first
19251             // tick.
19252             return tickArray[0];
19253         } else {
19254             for (var i=0, len=tickArray.length; i<len; ++i) {
19255                 var next = i + 1;
19256                 if (tickArray[next] && tickArray[next] >= val) {
19257                     var diff1 = val - tickArray[i];
19258                     var diff2 = tickArray[next] - val;
19259                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19260                 }
19261             }
19262
19263             // The value is larger than the last tick, so we return the last
19264             // tick.
19265             return tickArray[tickArray.length - 1];
19266         }
19267     },
19268
19269     /**
19270      * toString method
19271      * @method toString
19272      * @return {string} string representation of the dd obj
19273      */
19274     toString: function() {
19275         return ("DragDrop " + this.id);
19276     }
19277
19278 });
19279
19280 })();
19281 /*
19282  * Based on:
19283  * Ext JS Library 1.1.1
19284  * Copyright(c) 2006-2007, Ext JS, LLC.
19285  *
19286  * Originally Released Under LGPL - original licence link has changed is not relivant.
19287  *
19288  * Fork - LGPL
19289  * <script type="text/javascript">
19290  */
19291
19292
19293 /**
19294  * The drag and drop utility provides a framework for building drag and drop
19295  * applications.  In addition to enabling drag and drop for specific elements,
19296  * the drag and drop elements are tracked by the manager class, and the
19297  * interactions between the various elements are tracked during the drag and
19298  * the implementing code is notified about these important moments.
19299  */
19300
19301 // Only load the library once.  Rewriting the manager class would orphan
19302 // existing drag and drop instances.
19303 if (!Roo.dd.DragDropMgr) {
19304
19305 /**
19306  * @class Roo.dd.DragDropMgr
19307  * DragDropMgr is a singleton that tracks the element interaction for
19308  * all DragDrop items in the window.  Generally, you will not call
19309  * this class directly, but it does have helper methods that could
19310  * be useful in your DragDrop implementations.
19311  * @singleton
19312  */
19313 Roo.dd.DragDropMgr = function() {
19314
19315     var Event = Roo.EventManager;
19316
19317     return {
19318
19319         /**
19320          * Two dimensional Array of registered DragDrop objects.  The first
19321          * dimension is the DragDrop item group, the second the DragDrop
19322          * object.
19323          * @property ids
19324          * @type {string: string}
19325          * @private
19326          * @static
19327          */
19328         ids: {},
19329
19330         /**
19331          * Array of element ids defined as drag handles.  Used to determine
19332          * if the element that generated the mousedown event is actually the
19333          * handle and not the html element itself.
19334          * @property handleIds
19335          * @type {string: string}
19336          * @private
19337          * @static
19338          */
19339         handleIds: {},
19340
19341         /**
19342          * the DragDrop object that is currently being dragged
19343          * @property dragCurrent
19344          * @type DragDrop
19345          * @private
19346          * @static
19347          **/
19348         dragCurrent: null,
19349
19350         /**
19351          * the DragDrop object(s) that are being hovered over
19352          * @property dragOvers
19353          * @type Array
19354          * @private
19355          * @static
19356          */
19357         dragOvers: {},
19358
19359         /**
19360          * the X distance between the cursor and the object being dragged
19361          * @property deltaX
19362          * @type int
19363          * @private
19364          * @static
19365          */
19366         deltaX: 0,
19367
19368         /**
19369          * the Y distance between the cursor and the object being dragged
19370          * @property deltaY
19371          * @type int
19372          * @private
19373          * @static
19374          */
19375         deltaY: 0,
19376
19377         /**
19378          * Flag to determine if we should prevent the default behavior of the
19379          * events we define. By default this is true, but this can be set to
19380          * false if you need the default behavior (not recommended)
19381          * @property preventDefault
19382          * @type boolean
19383          * @static
19384          */
19385         preventDefault: true,
19386
19387         /**
19388          * Flag to determine if we should stop the propagation of the events
19389          * we generate. This is true by default but you may want to set it to
19390          * false if the html element contains other features that require the
19391          * mouse click.
19392          * @property stopPropagation
19393          * @type boolean
19394          * @static
19395          */
19396         stopPropagation: true,
19397
19398         /**
19399          * Internal flag that is set to true when drag and drop has been
19400          * intialized
19401          * @property initialized
19402          * @private
19403          * @static
19404          */
19405         initalized: false,
19406
19407         /**
19408          * All drag and drop can be disabled.
19409          * @property locked
19410          * @private
19411          * @static
19412          */
19413         locked: false,
19414
19415         /**
19416          * Called the first time an element is registered.
19417          * @method init
19418          * @private
19419          * @static
19420          */
19421         init: function() {
19422             this.initialized = true;
19423         },
19424
19425         /**
19426          * In point mode, drag and drop interaction is defined by the
19427          * location of the cursor during the drag/drop
19428          * @property POINT
19429          * @type int
19430          * @static
19431          */
19432         POINT: 0,
19433
19434         /**
19435          * In intersect mode, drag and drop interactio nis defined by the
19436          * overlap of two or more drag and drop objects.
19437          * @property INTERSECT
19438          * @type int
19439          * @static
19440          */
19441         INTERSECT: 1,
19442
19443         /**
19444          * The current drag and drop mode.  Default: POINT
19445          * @property mode
19446          * @type int
19447          * @static
19448          */
19449         mode: 0,
19450
19451         /**
19452          * Runs method on all drag and drop objects
19453          * @method _execOnAll
19454          * @private
19455          * @static
19456          */
19457         _execOnAll: function(sMethod, args) {
19458             for (var i in this.ids) {
19459                 for (var j in this.ids[i]) {
19460                     var oDD = this.ids[i][j];
19461                     if (! this.isTypeOfDD(oDD)) {
19462                         continue;
19463                     }
19464                     oDD[sMethod].apply(oDD, args);
19465                 }
19466             }
19467         },
19468
19469         /**
19470          * Drag and drop initialization.  Sets up the global event handlers
19471          * @method _onLoad
19472          * @private
19473          * @static
19474          */
19475         _onLoad: function() {
19476
19477             this.init();
19478
19479             if (!Roo.isTouch) {
19480                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19481                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19482             }
19483             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19484             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19485             
19486             Event.on(window,   "unload",    this._onUnload, this, true);
19487             Event.on(window,   "resize",    this._onResize, this, true);
19488             // Event.on(window,   "mouseout",    this._test);
19489
19490         },
19491
19492         /**
19493          * Reset constraints on all drag and drop objs
19494          * @method _onResize
19495          * @private
19496          * @static
19497          */
19498         _onResize: function(e) {
19499             this._execOnAll("resetConstraints", []);
19500         },
19501
19502         /**
19503          * Lock all drag and drop functionality
19504          * @method lock
19505          * @static
19506          */
19507         lock: function() { this.locked = true; },
19508
19509         /**
19510          * Unlock all drag and drop functionality
19511          * @method unlock
19512          * @static
19513          */
19514         unlock: function() { this.locked = false; },
19515
19516         /**
19517          * Is drag and drop locked?
19518          * @method isLocked
19519          * @return {boolean} True if drag and drop is locked, false otherwise.
19520          * @static
19521          */
19522         isLocked: function() { return this.locked; },
19523
19524         /**
19525          * Location cache that is set for all drag drop objects when a drag is
19526          * initiated, cleared when the drag is finished.
19527          * @property locationCache
19528          * @private
19529          * @static
19530          */
19531         locationCache: {},
19532
19533         /**
19534          * Set useCache to false if you want to force object the lookup of each
19535          * drag and drop linked element constantly during a drag.
19536          * @property useCache
19537          * @type boolean
19538          * @static
19539          */
19540         useCache: true,
19541
19542         /**
19543          * The number of pixels that the mouse needs to move after the
19544          * mousedown before the drag is initiated.  Default=3;
19545          * @property clickPixelThresh
19546          * @type int
19547          * @static
19548          */
19549         clickPixelThresh: 3,
19550
19551         /**
19552          * The number of milliseconds after the mousedown event to initiate the
19553          * drag if we don't get a mouseup event. Default=1000
19554          * @property clickTimeThresh
19555          * @type int
19556          * @static
19557          */
19558         clickTimeThresh: 350,
19559
19560         /**
19561          * Flag that indicates that either the drag pixel threshold or the
19562          * mousdown time threshold has been met
19563          * @property dragThreshMet
19564          * @type boolean
19565          * @private
19566          * @static
19567          */
19568         dragThreshMet: false,
19569
19570         /**
19571          * Timeout used for the click time threshold
19572          * @property clickTimeout
19573          * @type Object
19574          * @private
19575          * @static
19576          */
19577         clickTimeout: null,
19578
19579         /**
19580          * The X position of the mousedown event stored for later use when a
19581          * drag threshold is met.
19582          * @property startX
19583          * @type int
19584          * @private
19585          * @static
19586          */
19587         startX: 0,
19588
19589         /**
19590          * The Y position of the mousedown event stored for later use when a
19591          * drag threshold is met.
19592          * @property startY
19593          * @type int
19594          * @private
19595          * @static
19596          */
19597         startY: 0,
19598
19599         /**
19600          * Each DragDrop instance must be registered with the DragDropMgr.
19601          * This is executed in DragDrop.init()
19602          * @method regDragDrop
19603          * @param {DragDrop} oDD the DragDrop object to register
19604          * @param {String} sGroup the name of the group this element belongs to
19605          * @static
19606          */
19607         regDragDrop: function(oDD, sGroup) {
19608             if (!this.initialized) { this.init(); }
19609
19610             if (!this.ids[sGroup]) {
19611                 this.ids[sGroup] = {};
19612             }
19613             this.ids[sGroup][oDD.id] = oDD;
19614         },
19615
19616         /**
19617          * Removes the supplied dd instance from the supplied group. Executed
19618          * by DragDrop.removeFromGroup, so don't call this function directly.
19619          * @method removeDDFromGroup
19620          * @private
19621          * @static
19622          */
19623         removeDDFromGroup: function(oDD, sGroup) {
19624             if (!this.ids[sGroup]) {
19625                 this.ids[sGroup] = {};
19626             }
19627
19628             var obj = this.ids[sGroup];
19629             if (obj && obj[oDD.id]) {
19630                 delete obj[oDD.id];
19631             }
19632         },
19633
19634         /**
19635          * Unregisters a drag and drop item.  This is executed in
19636          * DragDrop.unreg, use that method instead of calling this directly.
19637          * @method _remove
19638          * @private
19639          * @static
19640          */
19641         _remove: function(oDD) {
19642             for (var g in oDD.groups) {
19643                 if (g && this.ids[g][oDD.id]) {
19644                     delete this.ids[g][oDD.id];
19645                 }
19646             }
19647             delete this.handleIds[oDD.id];
19648         },
19649
19650         /**
19651          * Each DragDrop handle element must be registered.  This is done
19652          * automatically when executing DragDrop.setHandleElId()
19653          * @method regHandle
19654          * @param {String} sDDId the DragDrop id this element is a handle for
19655          * @param {String} sHandleId the id of the element that is the drag
19656          * handle
19657          * @static
19658          */
19659         regHandle: function(sDDId, sHandleId) {
19660             if (!this.handleIds[sDDId]) {
19661                 this.handleIds[sDDId] = {};
19662             }
19663             this.handleIds[sDDId][sHandleId] = sHandleId;
19664         },
19665
19666         /**
19667          * Utility function to determine if a given element has been
19668          * registered as a drag drop item.
19669          * @method isDragDrop
19670          * @param {String} id the element id to check
19671          * @return {boolean} true if this element is a DragDrop item,
19672          * false otherwise
19673          * @static
19674          */
19675         isDragDrop: function(id) {
19676             return ( this.getDDById(id) ) ? true : false;
19677         },
19678
19679         /**
19680          * Returns the drag and drop instances that are in all groups the
19681          * passed in instance belongs to.
19682          * @method getRelated
19683          * @param {DragDrop} p_oDD the obj to get related data for
19684          * @param {boolean} bTargetsOnly if true, only return targetable objs
19685          * @return {DragDrop[]} the related instances
19686          * @static
19687          */
19688         getRelated: function(p_oDD, bTargetsOnly) {
19689             var oDDs = [];
19690             for (var i in p_oDD.groups) {
19691                 for (j in this.ids[i]) {
19692                     var dd = this.ids[i][j];
19693                     if (! this.isTypeOfDD(dd)) {
19694                         continue;
19695                     }
19696                     if (!bTargetsOnly || dd.isTarget) {
19697                         oDDs[oDDs.length] = dd;
19698                     }
19699                 }
19700             }
19701
19702             return oDDs;
19703         },
19704
19705         /**
19706          * Returns true if the specified dd target is a legal target for
19707          * the specifice drag obj
19708          * @method isLegalTarget
19709          * @param {DragDrop} the drag obj
19710          * @param {DragDrop} the target
19711          * @return {boolean} true if the target is a legal target for the
19712          * dd obj
19713          * @static
19714          */
19715         isLegalTarget: function (oDD, oTargetDD) {
19716             var targets = this.getRelated(oDD, true);
19717             for (var i=0, len=targets.length;i<len;++i) {
19718                 if (targets[i].id == oTargetDD.id) {
19719                     return true;
19720                 }
19721             }
19722
19723             return false;
19724         },
19725
19726         /**
19727          * My goal is to be able to transparently determine if an object is
19728          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19729          * returns "object", oDD.constructor.toString() always returns
19730          * "DragDrop" and not the name of the subclass.  So for now it just
19731          * evaluates a well-known variable in DragDrop.
19732          * @method isTypeOfDD
19733          * @param {Object} the object to evaluate
19734          * @return {boolean} true if typeof oDD = DragDrop
19735          * @static
19736          */
19737         isTypeOfDD: function (oDD) {
19738             return (oDD && oDD.__ygDragDrop);
19739         },
19740
19741         /**
19742          * Utility function to determine if a given element has been
19743          * registered as a drag drop handle for the given Drag Drop object.
19744          * @method isHandle
19745          * @param {String} id the element id to check
19746          * @return {boolean} true if this element is a DragDrop handle, false
19747          * otherwise
19748          * @static
19749          */
19750         isHandle: function(sDDId, sHandleId) {
19751             return ( this.handleIds[sDDId] &&
19752                             this.handleIds[sDDId][sHandleId] );
19753         },
19754
19755         /**
19756          * Returns the DragDrop instance for a given id
19757          * @method getDDById
19758          * @param {String} id the id of the DragDrop object
19759          * @return {DragDrop} the drag drop object, null if it is not found
19760          * @static
19761          */
19762         getDDById: function(id) {
19763             for (var i in this.ids) {
19764                 if (this.ids[i][id]) {
19765                     return this.ids[i][id];
19766                 }
19767             }
19768             return null;
19769         },
19770
19771         /**
19772          * Fired after a registered DragDrop object gets the mousedown event.
19773          * Sets up the events required to track the object being dragged
19774          * @method handleMouseDown
19775          * @param {Event} e the event
19776          * @param oDD the DragDrop object being dragged
19777          * @private
19778          * @static
19779          */
19780         handleMouseDown: function(e, oDD) {
19781             if(Roo.QuickTips){
19782                 Roo.QuickTips.disable();
19783             }
19784             this.currentTarget = e.getTarget();
19785
19786             this.dragCurrent = oDD;
19787
19788             var el = oDD.getEl();
19789
19790             // track start position
19791             this.startX = e.getPageX();
19792             this.startY = e.getPageY();
19793
19794             this.deltaX = this.startX - el.offsetLeft;
19795             this.deltaY = this.startY - el.offsetTop;
19796
19797             this.dragThreshMet = false;
19798
19799             this.clickTimeout = setTimeout(
19800                     function() {
19801                         var DDM = Roo.dd.DDM;
19802                         DDM.startDrag(DDM.startX, DDM.startY);
19803                     },
19804                     this.clickTimeThresh );
19805         },
19806
19807         /**
19808          * Fired when either the drag pixel threshol or the mousedown hold
19809          * time threshold has been met.
19810          * @method startDrag
19811          * @param x {int} the X position of the original mousedown
19812          * @param y {int} the Y position of the original mousedown
19813          * @static
19814          */
19815         startDrag: function(x, y) {
19816             clearTimeout(this.clickTimeout);
19817             if (this.dragCurrent) {
19818                 this.dragCurrent.b4StartDrag(x, y);
19819                 this.dragCurrent.startDrag(x, y);
19820             }
19821             this.dragThreshMet = true;
19822         },
19823
19824         /**
19825          * Internal function to handle the mouseup event.  Will be invoked
19826          * from the context of the document.
19827          * @method handleMouseUp
19828          * @param {Event} e the event
19829          * @private
19830          * @static
19831          */
19832         handleMouseUp: function(e) {
19833
19834             if(Roo.QuickTips){
19835                 Roo.QuickTips.enable();
19836             }
19837             if (! this.dragCurrent) {
19838                 return;
19839             }
19840
19841             clearTimeout(this.clickTimeout);
19842
19843             if (this.dragThreshMet) {
19844                 this.fireEvents(e, true);
19845             } else {
19846             }
19847
19848             this.stopDrag(e);
19849
19850             this.stopEvent(e);
19851         },
19852
19853         /**
19854          * Utility to stop event propagation and event default, if these
19855          * features are turned on.
19856          * @method stopEvent
19857          * @param {Event} e the event as returned by this.getEvent()
19858          * @static
19859          */
19860         stopEvent: function(e){
19861             if(this.stopPropagation) {
19862                 e.stopPropagation();
19863             }
19864
19865             if (this.preventDefault) {
19866                 e.preventDefault();
19867             }
19868         },
19869
19870         /**
19871          * Internal function to clean up event handlers after the drag
19872          * operation is complete
19873          * @method stopDrag
19874          * @param {Event} e the event
19875          * @private
19876          * @static
19877          */
19878         stopDrag: function(e) {
19879             // Fire the drag end event for the item that was dragged
19880             if (this.dragCurrent) {
19881                 if (this.dragThreshMet) {
19882                     this.dragCurrent.b4EndDrag(e);
19883                     this.dragCurrent.endDrag(e);
19884                 }
19885
19886                 this.dragCurrent.onMouseUp(e);
19887             }
19888
19889             this.dragCurrent = null;
19890             this.dragOvers = {};
19891         },
19892
19893         /**
19894          * Internal function to handle the mousemove event.  Will be invoked
19895          * from the context of the html element.
19896          *
19897          * @TODO figure out what we can do about mouse events lost when the
19898          * user drags objects beyond the window boundary.  Currently we can
19899          * detect this in internet explorer by verifying that the mouse is
19900          * down during the mousemove event.  Firefox doesn't give us the
19901          * button state on the mousemove event.
19902          * @method handleMouseMove
19903          * @param {Event} e the event
19904          * @private
19905          * @static
19906          */
19907         handleMouseMove: function(e) {
19908             if (! this.dragCurrent) {
19909                 return true;
19910             }
19911
19912             // var button = e.which || e.button;
19913
19914             // check for IE mouseup outside of page boundary
19915             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19916                 this.stopEvent(e);
19917                 return this.handleMouseUp(e);
19918             }
19919
19920             if (!this.dragThreshMet) {
19921                 var diffX = Math.abs(this.startX - e.getPageX());
19922                 var diffY = Math.abs(this.startY - e.getPageY());
19923                 if (diffX > this.clickPixelThresh ||
19924                             diffY > this.clickPixelThresh) {
19925                     this.startDrag(this.startX, this.startY);
19926                 }
19927             }
19928
19929             if (this.dragThreshMet) {
19930                 this.dragCurrent.b4Drag(e);
19931                 this.dragCurrent.onDrag(e);
19932                 if(!this.dragCurrent.moveOnly){
19933                     this.fireEvents(e, false);
19934                 }
19935             }
19936
19937             this.stopEvent(e);
19938
19939             return true;
19940         },
19941
19942         /**
19943          * Iterates over all of the DragDrop elements to find ones we are
19944          * hovering over or dropping on
19945          * @method fireEvents
19946          * @param {Event} e the event
19947          * @param {boolean} isDrop is this a drop op or a mouseover op?
19948          * @private
19949          * @static
19950          */
19951         fireEvents: function(e, isDrop) {
19952             var dc = this.dragCurrent;
19953
19954             // If the user did the mouse up outside of the window, we could
19955             // get here even though we have ended the drag.
19956             if (!dc || dc.isLocked()) {
19957                 return;
19958             }
19959
19960             var pt = e.getPoint();
19961
19962             // cache the previous dragOver array
19963             var oldOvers = [];
19964
19965             var outEvts   = [];
19966             var overEvts  = [];
19967             var dropEvts  = [];
19968             var enterEvts = [];
19969
19970             // Check to see if the object(s) we were hovering over is no longer
19971             // being hovered over so we can fire the onDragOut event
19972             for (var i in this.dragOvers) {
19973
19974                 var ddo = this.dragOvers[i];
19975
19976                 if (! this.isTypeOfDD(ddo)) {
19977                     continue;
19978                 }
19979
19980                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19981                     outEvts.push( ddo );
19982                 }
19983
19984                 oldOvers[i] = true;
19985                 delete this.dragOvers[i];
19986             }
19987
19988             for (var sGroup in dc.groups) {
19989
19990                 if ("string" != typeof sGroup) {
19991                     continue;
19992                 }
19993
19994                 for (i in this.ids[sGroup]) {
19995                     var oDD = this.ids[sGroup][i];
19996                     if (! this.isTypeOfDD(oDD)) {
19997                         continue;
19998                     }
19999
20000                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20001                         if (this.isOverTarget(pt, oDD, this.mode)) {
20002                             // look for drop interactions
20003                             if (isDrop) {
20004                                 dropEvts.push( oDD );
20005                             // look for drag enter and drag over interactions
20006                             } else {
20007
20008                                 // initial drag over: dragEnter fires
20009                                 if (!oldOvers[oDD.id]) {
20010                                     enterEvts.push( oDD );
20011                                 // subsequent drag overs: dragOver fires
20012                                 } else {
20013                                     overEvts.push( oDD );
20014                                 }
20015
20016                                 this.dragOvers[oDD.id] = oDD;
20017                             }
20018                         }
20019                     }
20020                 }
20021             }
20022
20023             if (this.mode) {
20024                 if (outEvts.length) {
20025                     dc.b4DragOut(e, outEvts);
20026                     dc.onDragOut(e, outEvts);
20027                 }
20028
20029                 if (enterEvts.length) {
20030                     dc.onDragEnter(e, enterEvts);
20031                 }
20032
20033                 if (overEvts.length) {
20034                     dc.b4DragOver(e, overEvts);
20035                     dc.onDragOver(e, overEvts);
20036                 }
20037
20038                 if (dropEvts.length) {
20039                     dc.b4DragDrop(e, dropEvts);
20040                     dc.onDragDrop(e, dropEvts);
20041                 }
20042
20043             } else {
20044                 // fire dragout events
20045                 var len = 0;
20046                 for (i=0, len=outEvts.length; i<len; ++i) {
20047                     dc.b4DragOut(e, outEvts[i].id);
20048                     dc.onDragOut(e, outEvts[i].id);
20049                 }
20050
20051                 // fire enter events
20052                 for (i=0,len=enterEvts.length; i<len; ++i) {
20053                     // dc.b4DragEnter(e, oDD.id);
20054                     dc.onDragEnter(e, enterEvts[i].id);
20055                 }
20056
20057                 // fire over events
20058                 for (i=0,len=overEvts.length; i<len; ++i) {
20059                     dc.b4DragOver(e, overEvts[i].id);
20060                     dc.onDragOver(e, overEvts[i].id);
20061                 }
20062
20063                 // fire drop events
20064                 for (i=0, len=dropEvts.length; i<len; ++i) {
20065                     dc.b4DragDrop(e, dropEvts[i].id);
20066                     dc.onDragDrop(e, dropEvts[i].id);
20067                 }
20068
20069             }
20070
20071             // notify about a drop that did not find a target
20072             if (isDrop && !dropEvts.length) {
20073                 dc.onInvalidDrop(e);
20074             }
20075
20076         },
20077
20078         /**
20079          * Helper function for getting the best match from the list of drag
20080          * and drop objects returned by the drag and drop events when we are
20081          * in INTERSECT mode.  It returns either the first object that the
20082          * cursor is over, or the object that has the greatest overlap with
20083          * the dragged element.
20084          * @method getBestMatch
20085          * @param  {DragDrop[]} dds The array of drag and drop objects
20086          * targeted
20087          * @return {DragDrop}       The best single match
20088          * @static
20089          */
20090         getBestMatch: function(dds) {
20091             var winner = null;
20092             // Return null if the input is not what we expect
20093             //if (!dds || !dds.length || dds.length == 0) {
20094                // winner = null;
20095             // If there is only one item, it wins
20096             //} else if (dds.length == 1) {
20097
20098             var len = dds.length;
20099
20100             if (len == 1) {
20101                 winner = dds[0];
20102             } else {
20103                 // Loop through the targeted items
20104                 for (var i=0; i<len; ++i) {
20105                     var dd = dds[i];
20106                     // If the cursor is over the object, it wins.  If the
20107                     // cursor is over multiple matches, the first one we come
20108                     // to wins.
20109                     if (dd.cursorIsOver) {
20110                         winner = dd;
20111                         break;
20112                     // Otherwise the object with the most overlap wins
20113                     } else {
20114                         if (!winner ||
20115                             winner.overlap.getArea() < dd.overlap.getArea()) {
20116                             winner = dd;
20117                         }
20118                     }
20119                 }
20120             }
20121
20122             return winner;
20123         },
20124
20125         /**
20126          * Refreshes the cache of the top-left and bottom-right points of the
20127          * drag and drop objects in the specified group(s).  This is in the
20128          * format that is stored in the drag and drop instance, so typical
20129          * usage is:
20130          * <code>
20131          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20132          * </code>
20133          * Alternatively:
20134          * <code>
20135          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20136          * </code>
20137          * @TODO this really should be an indexed array.  Alternatively this
20138          * method could accept both.
20139          * @method refreshCache
20140          * @param {Object} groups an associative array of groups to refresh
20141          * @static
20142          */
20143         refreshCache: function(groups) {
20144             for (var sGroup in groups) {
20145                 if ("string" != typeof sGroup) {
20146                     continue;
20147                 }
20148                 for (var i in this.ids[sGroup]) {
20149                     var oDD = this.ids[sGroup][i];
20150
20151                     if (this.isTypeOfDD(oDD)) {
20152                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20153                         var loc = this.getLocation(oDD);
20154                         if (loc) {
20155                             this.locationCache[oDD.id] = loc;
20156                         } else {
20157                             delete this.locationCache[oDD.id];
20158                             // this will unregister the drag and drop object if
20159                             // the element is not in a usable state
20160                             // oDD.unreg();
20161                         }
20162                     }
20163                 }
20164             }
20165         },
20166
20167         /**
20168          * This checks to make sure an element exists and is in the DOM.  The
20169          * main purpose is to handle cases where innerHTML is used to remove
20170          * drag and drop objects from the DOM.  IE provides an 'unspecified
20171          * error' when trying to access the offsetParent of such an element
20172          * @method verifyEl
20173          * @param {HTMLElement} el the element to check
20174          * @return {boolean} true if the element looks usable
20175          * @static
20176          */
20177         verifyEl: function(el) {
20178             if (el) {
20179                 var parent;
20180                 if(Roo.isIE){
20181                     try{
20182                         parent = el.offsetParent;
20183                     }catch(e){}
20184                 }else{
20185                     parent = el.offsetParent;
20186                 }
20187                 if (parent) {
20188                     return true;
20189                 }
20190             }
20191
20192             return false;
20193         },
20194
20195         /**
20196          * Returns a Region object containing the drag and drop element's position
20197          * and size, including the padding configured for it
20198          * @method getLocation
20199          * @param {DragDrop} oDD the drag and drop object to get the
20200          *                       location for
20201          * @return {Roo.lib.Region} a Region object representing the total area
20202          *                             the element occupies, including any padding
20203          *                             the instance is configured for.
20204          * @static
20205          */
20206         getLocation: function(oDD) {
20207             if (! this.isTypeOfDD(oDD)) {
20208                 return null;
20209             }
20210
20211             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20212
20213             try {
20214                 pos= Roo.lib.Dom.getXY(el);
20215             } catch (e) { }
20216
20217             if (!pos) {
20218                 return null;
20219             }
20220
20221             x1 = pos[0];
20222             x2 = x1 + el.offsetWidth;
20223             y1 = pos[1];
20224             y2 = y1 + el.offsetHeight;
20225
20226             t = y1 - oDD.padding[0];
20227             r = x2 + oDD.padding[1];
20228             b = y2 + oDD.padding[2];
20229             l = x1 - oDD.padding[3];
20230
20231             return new Roo.lib.Region( t, r, b, l );
20232         },
20233
20234         /**
20235          * Checks the cursor location to see if it over the target
20236          * @method isOverTarget
20237          * @param {Roo.lib.Point} pt The point to evaluate
20238          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20239          * @return {boolean} true if the mouse is over the target
20240          * @private
20241          * @static
20242          */
20243         isOverTarget: function(pt, oTarget, intersect) {
20244             // use cache if available
20245             var loc = this.locationCache[oTarget.id];
20246             if (!loc || !this.useCache) {
20247                 loc = this.getLocation(oTarget);
20248                 this.locationCache[oTarget.id] = loc;
20249
20250             }
20251
20252             if (!loc) {
20253                 return false;
20254             }
20255
20256             oTarget.cursorIsOver = loc.contains( pt );
20257
20258             // DragDrop is using this as a sanity check for the initial mousedown
20259             // in this case we are done.  In POINT mode, if the drag obj has no
20260             // contraints, we are also done. Otherwise we need to evaluate the
20261             // location of the target as related to the actual location of the
20262             // dragged element.
20263             var dc = this.dragCurrent;
20264             if (!dc || !dc.getTargetCoord ||
20265                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20266                 return oTarget.cursorIsOver;
20267             }
20268
20269             oTarget.overlap = null;
20270
20271             // Get the current location of the drag element, this is the
20272             // location of the mouse event less the delta that represents
20273             // where the original mousedown happened on the element.  We
20274             // need to consider constraints and ticks as well.
20275             var pos = dc.getTargetCoord(pt.x, pt.y);
20276
20277             var el = dc.getDragEl();
20278             var curRegion = new Roo.lib.Region( pos.y,
20279                                                    pos.x + el.offsetWidth,
20280                                                    pos.y + el.offsetHeight,
20281                                                    pos.x );
20282
20283             var overlap = curRegion.intersect(loc);
20284
20285             if (overlap) {
20286                 oTarget.overlap = overlap;
20287                 return (intersect) ? true : oTarget.cursorIsOver;
20288             } else {
20289                 return false;
20290             }
20291         },
20292
20293         /**
20294          * unload event handler
20295          * @method _onUnload
20296          * @private
20297          * @static
20298          */
20299         _onUnload: function(e, me) {
20300             Roo.dd.DragDropMgr.unregAll();
20301         },
20302
20303         /**
20304          * Cleans up the drag and drop events and objects.
20305          * @method unregAll
20306          * @private
20307          * @static
20308          */
20309         unregAll: function() {
20310
20311             if (this.dragCurrent) {
20312                 this.stopDrag();
20313                 this.dragCurrent = null;
20314             }
20315
20316             this._execOnAll("unreg", []);
20317
20318             for (i in this.elementCache) {
20319                 delete this.elementCache[i];
20320             }
20321
20322             this.elementCache = {};
20323             this.ids = {};
20324         },
20325
20326         /**
20327          * A cache of DOM elements
20328          * @property elementCache
20329          * @private
20330          * @static
20331          */
20332         elementCache: {},
20333
20334         /**
20335          * Get the wrapper for the DOM element specified
20336          * @method getElWrapper
20337          * @param {String} id the id of the element to get
20338          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20339          * @private
20340          * @deprecated This wrapper isn't that useful
20341          * @static
20342          */
20343         getElWrapper: function(id) {
20344             var oWrapper = this.elementCache[id];
20345             if (!oWrapper || !oWrapper.el) {
20346                 oWrapper = this.elementCache[id] =
20347                     new this.ElementWrapper(Roo.getDom(id));
20348             }
20349             return oWrapper;
20350         },
20351
20352         /**
20353          * Returns the actual DOM element
20354          * @method getElement
20355          * @param {String} id the id of the elment to get
20356          * @return {Object} The element
20357          * @deprecated use Roo.getDom instead
20358          * @static
20359          */
20360         getElement: function(id) {
20361             return Roo.getDom(id);
20362         },
20363
20364         /**
20365          * Returns the style property for the DOM element (i.e.,
20366          * document.getElById(id).style)
20367          * @method getCss
20368          * @param {String} id the id of the elment to get
20369          * @return {Object} The style property of the element
20370          * @deprecated use Roo.getDom instead
20371          * @static
20372          */
20373         getCss: function(id) {
20374             var el = Roo.getDom(id);
20375             return (el) ? el.style : null;
20376         },
20377
20378         /**
20379          * Inner class for cached elements
20380          * @class DragDropMgr.ElementWrapper
20381          * @for DragDropMgr
20382          * @private
20383          * @deprecated
20384          */
20385         ElementWrapper: function(el) {
20386                 /**
20387                  * The element
20388                  * @property el
20389                  */
20390                 this.el = el || null;
20391                 /**
20392                  * The element id
20393                  * @property id
20394                  */
20395                 this.id = this.el && el.id;
20396                 /**
20397                  * A reference to the style property
20398                  * @property css
20399                  */
20400                 this.css = this.el && el.style;
20401             },
20402
20403         /**
20404          * Returns the X position of an html element
20405          * @method getPosX
20406          * @param el the element for which to get the position
20407          * @return {int} the X coordinate
20408          * @for DragDropMgr
20409          * @deprecated use Roo.lib.Dom.getX instead
20410          * @static
20411          */
20412         getPosX: function(el) {
20413             return Roo.lib.Dom.getX(el);
20414         },
20415
20416         /**
20417          * Returns the Y position of an html element
20418          * @method getPosY
20419          * @param el the element for which to get the position
20420          * @return {int} the Y coordinate
20421          * @deprecated use Roo.lib.Dom.getY instead
20422          * @static
20423          */
20424         getPosY: function(el) {
20425             return Roo.lib.Dom.getY(el);
20426         },
20427
20428         /**
20429          * Swap two nodes.  In IE, we use the native method, for others we
20430          * emulate the IE behavior
20431          * @method swapNode
20432          * @param n1 the first node to swap
20433          * @param n2 the other node to swap
20434          * @static
20435          */
20436         swapNode: function(n1, n2) {
20437             if (n1.swapNode) {
20438                 n1.swapNode(n2);
20439             } else {
20440                 var p = n2.parentNode;
20441                 var s = n2.nextSibling;
20442
20443                 if (s == n1) {
20444                     p.insertBefore(n1, n2);
20445                 } else if (n2 == n1.nextSibling) {
20446                     p.insertBefore(n2, n1);
20447                 } else {
20448                     n1.parentNode.replaceChild(n2, n1);
20449                     p.insertBefore(n1, s);
20450                 }
20451             }
20452         },
20453
20454         /**
20455          * Returns the current scroll position
20456          * @method getScroll
20457          * @private
20458          * @static
20459          */
20460         getScroll: function () {
20461             var t, l, dde=document.documentElement, db=document.body;
20462             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20463                 t = dde.scrollTop;
20464                 l = dde.scrollLeft;
20465             } else if (db) {
20466                 t = db.scrollTop;
20467                 l = db.scrollLeft;
20468             } else {
20469
20470             }
20471             return { top: t, left: l };
20472         },
20473
20474         /**
20475          * Returns the specified element style property
20476          * @method getStyle
20477          * @param {HTMLElement} el          the element
20478          * @param {string}      styleProp   the style property
20479          * @return {string} The value of the style property
20480          * @deprecated use Roo.lib.Dom.getStyle
20481          * @static
20482          */
20483         getStyle: function(el, styleProp) {
20484             return Roo.fly(el).getStyle(styleProp);
20485         },
20486
20487         /**
20488          * Gets the scrollTop
20489          * @method getScrollTop
20490          * @return {int} the document's scrollTop
20491          * @static
20492          */
20493         getScrollTop: function () { return this.getScroll().top; },
20494
20495         /**
20496          * Gets the scrollLeft
20497          * @method getScrollLeft
20498          * @return {int} the document's scrollTop
20499          * @static
20500          */
20501         getScrollLeft: function () { return this.getScroll().left; },
20502
20503         /**
20504          * Sets the x/y position of an element to the location of the
20505          * target element.
20506          * @method moveToEl
20507          * @param {HTMLElement} moveEl      The element to move
20508          * @param {HTMLElement} targetEl    The position reference element
20509          * @static
20510          */
20511         moveToEl: function (moveEl, targetEl) {
20512             var aCoord = Roo.lib.Dom.getXY(targetEl);
20513             Roo.lib.Dom.setXY(moveEl, aCoord);
20514         },
20515
20516         /**
20517          * Numeric array sort function
20518          * @method numericSort
20519          * @static
20520          */
20521         numericSort: function(a, b) { return (a - b); },
20522
20523         /**
20524          * Internal counter
20525          * @property _timeoutCount
20526          * @private
20527          * @static
20528          */
20529         _timeoutCount: 0,
20530
20531         /**
20532          * Trying to make the load order less important.  Without this we get
20533          * an error if this file is loaded before the Event Utility.
20534          * @method _addListeners
20535          * @private
20536          * @static
20537          */
20538         _addListeners: function() {
20539             var DDM = Roo.dd.DDM;
20540             if ( Roo.lib.Event && document ) {
20541                 DDM._onLoad();
20542             } else {
20543                 if (DDM._timeoutCount > 2000) {
20544                 } else {
20545                     setTimeout(DDM._addListeners, 10);
20546                     if (document && document.body) {
20547                         DDM._timeoutCount += 1;
20548                     }
20549                 }
20550             }
20551         },
20552
20553         /**
20554          * Recursively searches the immediate parent and all child nodes for
20555          * the handle element in order to determine wheter or not it was
20556          * clicked.
20557          * @method handleWasClicked
20558          * @param node the html element to inspect
20559          * @static
20560          */
20561         handleWasClicked: function(node, id) {
20562             if (this.isHandle(id, node.id)) {
20563                 return true;
20564             } else {
20565                 // check to see if this is a text node child of the one we want
20566                 var p = node.parentNode;
20567
20568                 while (p) {
20569                     if (this.isHandle(id, p.id)) {
20570                         return true;
20571                     } else {
20572                         p = p.parentNode;
20573                     }
20574                 }
20575             }
20576
20577             return false;
20578         }
20579
20580     };
20581
20582 }();
20583
20584 // shorter alias, save a few bytes
20585 Roo.dd.DDM = Roo.dd.DragDropMgr;
20586 Roo.dd.DDM._addListeners();
20587
20588 }/*
20589  * Based on:
20590  * Ext JS Library 1.1.1
20591  * Copyright(c) 2006-2007, Ext JS, LLC.
20592  *
20593  * Originally Released Under LGPL - original licence link has changed is not relivant.
20594  *
20595  * Fork - LGPL
20596  * <script type="text/javascript">
20597  */
20598
20599 /**
20600  * @class Roo.dd.DD
20601  * A DragDrop implementation where the linked element follows the
20602  * mouse cursor during a drag.
20603  * @extends Roo.dd.DragDrop
20604  * @constructor
20605  * @param {String} id the id of the linked element
20606  * @param {String} sGroup the group of related DragDrop items
20607  * @param {object} config an object containing configurable attributes
20608  *                Valid properties for DD:
20609  *                    scroll
20610  */
20611 Roo.dd.DD = function(id, sGroup, config) {
20612     if (id) {
20613         this.init(id, sGroup, config);
20614     }
20615 };
20616
20617 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20618
20619     /**
20620      * When set to true, the utility automatically tries to scroll the browser
20621      * window wehn a drag and drop element is dragged near the viewport boundary.
20622      * Defaults to true.
20623      * @property scroll
20624      * @type boolean
20625      */
20626     scroll: true,
20627
20628     /**
20629      * Sets the pointer offset to the distance between the linked element's top
20630      * left corner and the location the element was clicked
20631      * @method autoOffset
20632      * @param {int} iPageX the X coordinate of the click
20633      * @param {int} iPageY the Y coordinate of the click
20634      */
20635     autoOffset: function(iPageX, iPageY) {
20636         var x = iPageX - this.startPageX;
20637         var y = iPageY - this.startPageY;
20638         this.setDelta(x, y);
20639     },
20640
20641     /**
20642      * Sets the pointer offset.  You can call this directly to force the
20643      * offset to be in a particular location (e.g., pass in 0,0 to set it
20644      * to the center of the object)
20645      * @method setDelta
20646      * @param {int} iDeltaX the distance from the left
20647      * @param {int} iDeltaY the distance from the top
20648      */
20649     setDelta: function(iDeltaX, iDeltaY) {
20650         this.deltaX = iDeltaX;
20651         this.deltaY = iDeltaY;
20652     },
20653
20654     /**
20655      * Sets the drag element to the location of the mousedown or click event,
20656      * maintaining the cursor location relative to the location on the element
20657      * that was clicked.  Override this if you want to place the element in a
20658      * location other than where the cursor is.
20659      * @method setDragElPos
20660      * @param {int} iPageX the X coordinate of the mousedown or drag event
20661      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20662      */
20663     setDragElPos: function(iPageX, iPageY) {
20664         // the first time we do this, we are going to check to make sure
20665         // the element has css positioning
20666
20667         var el = this.getDragEl();
20668         this.alignElWithMouse(el, iPageX, iPageY);
20669     },
20670
20671     /**
20672      * Sets the element to the location of the mousedown or click event,
20673      * maintaining the cursor location relative to the location on the element
20674      * that was clicked.  Override this if you want to place the element in a
20675      * location other than where the cursor is.
20676      * @method alignElWithMouse
20677      * @param {HTMLElement} el the element to move
20678      * @param {int} iPageX the X coordinate of the mousedown or drag event
20679      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20680      */
20681     alignElWithMouse: function(el, iPageX, iPageY) {
20682         var oCoord = this.getTargetCoord(iPageX, iPageY);
20683         var fly = el.dom ? el : Roo.fly(el);
20684         if (!this.deltaSetXY) {
20685             var aCoord = [oCoord.x, oCoord.y];
20686             fly.setXY(aCoord);
20687             var newLeft = fly.getLeft(true);
20688             var newTop  = fly.getTop(true);
20689             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20690         } else {
20691             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20692         }
20693
20694         this.cachePosition(oCoord.x, oCoord.y);
20695         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20696         return oCoord;
20697     },
20698
20699     /**
20700      * Saves the most recent position so that we can reset the constraints and
20701      * tick marks on-demand.  We need to know this so that we can calculate the
20702      * number of pixels the element is offset from its original position.
20703      * @method cachePosition
20704      * @param iPageX the current x position (optional, this just makes it so we
20705      * don't have to look it up again)
20706      * @param iPageY the current y position (optional, this just makes it so we
20707      * don't have to look it up again)
20708      */
20709     cachePosition: function(iPageX, iPageY) {
20710         if (iPageX) {
20711             this.lastPageX = iPageX;
20712             this.lastPageY = iPageY;
20713         } else {
20714             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20715             this.lastPageX = aCoord[0];
20716             this.lastPageY = aCoord[1];
20717         }
20718     },
20719
20720     /**
20721      * Auto-scroll the window if the dragged object has been moved beyond the
20722      * visible window boundary.
20723      * @method autoScroll
20724      * @param {int} x the drag element's x position
20725      * @param {int} y the drag element's y position
20726      * @param {int} h the height of the drag element
20727      * @param {int} w the width of the drag element
20728      * @private
20729      */
20730     autoScroll: function(x, y, h, w) {
20731
20732         if (this.scroll) {
20733             // The client height
20734             var clientH = Roo.lib.Dom.getViewWidth();
20735
20736             // The client width
20737             var clientW = Roo.lib.Dom.getViewHeight();
20738
20739             // The amt scrolled down
20740             var st = this.DDM.getScrollTop();
20741
20742             // The amt scrolled right
20743             var sl = this.DDM.getScrollLeft();
20744
20745             // Location of the bottom of the element
20746             var bot = h + y;
20747
20748             // Location of the right of the element
20749             var right = w + x;
20750
20751             // The distance from the cursor to the bottom of the visible area,
20752             // adjusted so that we don't scroll if the cursor is beyond the
20753             // element drag constraints
20754             var toBot = (clientH + st - y - this.deltaY);
20755
20756             // The distance from the cursor to the right of the visible area
20757             var toRight = (clientW + sl - x - this.deltaX);
20758
20759
20760             // How close to the edge the cursor must be before we scroll
20761             // var thresh = (document.all) ? 100 : 40;
20762             var thresh = 40;
20763
20764             // How many pixels to scroll per autoscroll op.  This helps to reduce
20765             // clunky scrolling. IE is more sensitive about this ... it needs this
20766             // value to be higher.
20767             var scrAmt = (document.all) ? 80 : 30;
20768
20769             // Scroll down if we are near the bottom of the visible page and the
20770             // obj extends below the crease
20771             if ( bot > clientH && toBot < thresh ) {
20772                 window.scrollTo(sl, st + scrAmt);
20773             }
20774
20775             // Scroll up if the window is scrolled down and the top of the object
20776             // goes above the top border
20777             if ( y < st && st > 0 && y - st < thresh ) {
20778                 window.scrollTo(sl, st - scrAmt);
20779             }
20780
20781             // Scroll right if the obj is beyond the right border and the cursor is
20782             // near the border.
20783             if ( right > clientW && toRight < thresh ) {
20784                 window.scrollTo(sl + scrAmt, st);
20785             }
20786
20787             // Scroll left if the window has been scrolled to the right and the obj
20788             // extends past the left border
20789             if ( x < sl && sl > 0 && x - sl < thresh ) {
20790                 window.scrollTo(sl - scrAmt, st);
20791             }
20792         }
20793     },
20794
20795     /**
20796      * Finds the location the element should be placed if we want to move
20797      * it to where the mouse location less the click offset would place us.
20798      * @method getTargetCoord
20799      * @param {int} iPageX the X coordinate of the click
20800      * @param {int} iPageY the Y coordinate of the click
20801      * @return an object that contains the coordinates (Object.x and Object.y)
20802      * @private
20803      */
20804     getTargetCoord: function(iPageX, iPageY) {
20805
20806
20807         var x = iPageX - this.deltaX;
20808         var y = iPageY - this.deltaY;
20809
20810         if (this.constrainX) {
20811             if (x < this.minX) { x = this.minX; }
20812             if (x > this.maxX) { x = this.maxX; }
20813         }
20814
20815         if (this.constrainY) {
20816             if (y < this.minY) { y = this.minY; }
20817             if (y > this.maxY) { y = this.maxY; }
20818         }
20819
20820         x = this.getTick(x, this.xTicks);
20821         y = this.getTick(y, this.yTicks);
20822
20823
20824         return {x:x, y:y};
20825     },
20826
20827     /*
20828      * Sets up config options specific to this class. Overrides
20829      * Roo.dd.DragDrop, but all versions of this method through the
20830      * inheritance chain are called
20831      */
20832     applyConfig: function() {
20833         Roo.dd.DD.superclass.applyConfig.call(this);
20834         this.scroll = (this.config.scroll !== false);
20835     },
20836
20837     /*
20838      * Event that fires prior to the onMouseDown event.  Overrides
20839      * Roo.dd.DragDrop.
20840      */
20841     b4MouseDown: function(e) {
20842         // this.resetConstraints();
20843         this.autoOffset(e.getPageX(),
20844                             e.getPageY());
20845     },
20846
20847     /*
20848      * Event that fires prior to the onDrag event.  Overrides
20849      * Roo.dd.DragDrop.
20850      */
20851     b4Drag: function(e) {
20852         this.setDragElPos(e.getPageX(),
20853                             e.getPageY());
20854     },
20855
20856     toString: function() {
20857         return ("DD " + this.id);
20858     }
20859
20860     //////////////////////////////////////////////////////////////////////////
20861     // Debugging ygDragDrop events that can be overridden
20862     //////////////////////////////////////////////////////////////////////////
20863     /*
20864     startDrag: function(x, y) {
20865     },
20866
20867     onDrag: function(e) {
20868     },
20869
20870     onDragEnter: function(e, id) {
20871     },
20872
20873     onDragOver: function(e, id) {
20874     },
20875
20876     onDragOut: function(e, id) {
20877     },
20878
20879     onDragDrop: function(e, id) {
20880     },
20881
20882     endDrag: function(e) {
20883     }
20884
20885     */
20886
20887 });/*
20888  * Based on:
20889  * Ext JS Library 1.1.1
20890  * Copyright(c) 2006-2007, Ext JS, LLC.
20891  *
20892  * Originally Released Under LGPL - original licence link has changed is not relivant.
20893  *
20894  * Fork - LGPL
20895  * <script type="text/javascript">
20896  */
20897
20898 /**
20899  * @class Roo.dd.DDProxy
20900  * A DragDrop implementation that inserts an empty, bordered div into
20901  * the document that follows the cursor during drag operations.  At the time of
20902  * the click, the frame div is resized to the dimensions of the linked html
20903  * element, and moved to the exact location of the linked element.
20904  *
20905  * References to the "frame" element refer to the single proxy element that
20906  * was created to be dragged in place of all DDProxy elements on the
20907  * page.
20908  *
20909  * @extends Roo.dd.DD
20910  * @constructor
20911  * @param {String} id the id of the linked html element
20912  * @param {String} sGroup the group of related DragDrop objects
20913  * @param {object} config an object containing configurable attributes
20914  *                Valid properties for DDProxy in addition to those in DragDrop:
20915  *                   resizeFrame, centerFrame, dragElId
20916  */
20917 Roo.dd.DDProxy = function(id, sGroup, config) {
20918     if (id) {
20919         this.init(id, sGroup, config);
20920         this.initFrame();
20921     }
20922 };
20923
20924 /**
20925  * The default drag frame div id
20926  * @property Roo.dd.DDProxy.dragElId
20927  * @type String
20928  * @static
20929  */
20930 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20931
20932 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20933
20934     /**
20935      * By default we resize the drag frame to be the same size as the element
20936      * we want to drag (this is to get the frame effect).  We can turn it off
20937      * if we want a different behavior.
20938      * @property resizeFrame
20939      * @type boolean
20940      */
20941     resizeFrame: true,
20942
20943     /**
20944      * By default the frame is positioned exactly where the drag element is, so
20945      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20946      * you do not have constraints on the obj is to have the drag frame centered
20947      * around the cursor.  Set centerFrame to true for this effect.
20948      * @property centerFrame
20949      * @type boolean
20950      */
20951     centerFrame: false,
20952
20953     /**
20954      * Creates the proxy element if it does not yet exist
20955      * @method createFrame
20956      */
20957     createFrame: function() {
20958         var self = this;
20959         var body = document.body;
20960
20961         if (!body || !body.firstChild) {
20962             setTimeout( function() { self.createFrame(); }, 50 );
20963             return;
20964         }
20965
20966         var div = this.getDragEl();
20967
20968         if (!div) {
20969             div    = document.createElement("div");
20970             div.id = this.dragElId;
20971             var s  = div.style;
20972
20973             s.position   = "absolute";
20974             s.visibility = "hidden";
20975             s.cursor     = "move";
20976             s.border     = "2px solid #aaa";
20977             s.zIndex     = 999;
20978
20979             // appendChild can blow up IE if invoked prior to the window load event
20980             // while rendering a table.  It is possible there are other scenarios
20981             // that would cause this to happen as well.
20982             body.insertBefore(div, body.firstChild);
20983         }
20984     },
20985
20986     /**
20987      * Initialization for the drag frame element.  Must be called in the
20988      * constructor of all subclasses
20989      * @method initFrame
20990      */
20991     initFrame: function() {
20992         this.createFrame();
20993     },
20994
20995     applyConfig: function() {
20996         Roo.dd.DDProxy.superclass.applyConfig.call(this);
20997
20998         this.resizeFrame = (this.config.resizeFrame !== false);
20999         this.centerFrame = (this.config.centerFrame);
21000         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21001     },
21002
21003     /**
21004      * Resizes the drag frame to the dimensions of the clicked object, positions
21005      * it over the object, and finally displays it
21006      * @method showFrame
21007      * @param {int} iPageX X click position
21008      * @param {int} iPageY Y click position
21009      * @private
21010      */
21011     showFrame: function(iPageX, iPageY) {
21012         var el = this.getEl();
21013         var dragEl = this.getDragEl();
21014         var s = dragEl.style;
21015
21016         this._resizeProxy();
21017
21018         if (this.centerFrame) {
21019             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21020                            Math.round(parseInt(s.height, 10)/2) );
21021         }
21022
21023         this.setDragElPos(iPageX, iPageY);
21024
21025         Roo.fly(dragEl).show();
21026     },
21027
21028     /**
21029      * The proxy is automatically resized to the dimensions of the linked
21030      * element when a drag is initiated, unless resizeFrame is set to false
21031      * @method _resizeProxy
21032      * @private
21033      */
21034     _resizeProxy: function() {
21035         if (this.resizeFrame) {
21036             var el = this.getEl();
21037             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21038         }
21039     },
21040
21041     // overrides Roo.dd.DragDrop
21042     b4MouseDown: function(e) {
21043         var x = e.getPageX();
21044         var y = e.getPageY();
21045         this.autoOffset(x, y);
21046         this.setDragElPos(x, y);
21047     },
21048
21049     // overrides Roo.dd.DragDrop
21050     b4StartDrag: function(x, y) {
21051         // show the drag frame
21052         this.showFrame(x, y);
21053     },
21054
21055     // overrides Roo.dd.DragDrop
21056     b4EndDrag: function(e) {
21057         Roo.fly(this.getDragEl()).hide();
21058     },
21059
21060     // overrides Roo.dd.DragDrop
21061     // By default we try to move the element to the last location of the frame.
21062     // This is so that the default behavior mirrors that of Roo.dd.DD.
21063     endDrag: function(e) {
21064
21065         var lel = this.getEl();
21066         var del = this.getDragEl();
21067
21068         // Show the drag frame briefly so we can get its position
21069         del.style.visibility = "";
21070
21071         this.beforeMove();
21072         // Hide the linked element before the move to get around a Safari
21073         // rendering bug.
21074         lel.style.visibility = "hidden";
21075         Roo.dd.DDM.moveToEl(lel, del);
21076         del.style.visibility = "hidden";
21077         lel.style.visibility = "";
21078
21079         this.afterDrag();
21080     },
21081
21082     beforeMove : function(){
21083
21084     },
21085
21086     afterDrag : function(){
21087
21088     },
21089
21090     toString: function() {
21091         return ("DDProxy " + this.id);
21092     }
21093
21094 });
21095 /*
21096  * Based on:
21097  * Ext JS Library 1.1.1
21098  * Copyright(c) 2006-2007, Ext JS, LLC.
21099  *
21100  * Originally Released Under LGPL - original licence link has changed is not relivant.
21101  *
21102  * Fork - LGPL
21103  * <script type="text/javascript">
21104  */
21105
21106  /**
21107  * @class Roo.dd.DDTarget
21108  * A DragDrop implementation that does not move, but can be a drop
21109  * target.  You would get the same result by simply omitting implementation
21110  * for the event callbacks, but this way we reduce the processing cost of the
21111  * event listener and the callbacks.
21112  * @extends Roo.dd.DragDrop
21113  * @constructor
21114  * @param {String} id the id of the element that is a drop target
21115  * @param {String} sGroup the group of related DragDrop objects
21116  * @param {object} config an object containing configurable attributes
21117  *                 Valid properties for DDTarget in addition to those in
21118  *                 DragDrop:
21119  *                    none
21120  */
21121 Roo.dd.DDTarget = function(id, sGroup, config) {
21122     if (id) {
21123         this.initTarget(id, sGroup, config);
21124     }
21125     if (config && (config.listeners || config.events)) { 
21126         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21127             listeners : config.listeners || {}, 
21128             events : config.events || {} 
21129         });    
21130     }
21131 };
21132
21133 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21134 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21135     toString: function() {
21136         return ("DDTarget " + this.id);
21137     }
21138 });
21139 /*
21140  * Based on:
21141  * Ext JS Library 1.1.1
21142  * Copyright(c) 2006-2007, Ext JS, LLC.
21143  *
21144  * Originally Released Under LGPL - original licence link has changed is not relivant.
21145  *
21146  * Fork - LGPL
21147  * <script type="text/javascript">
21148  */
21149  
21150
21151 /**
21152  * @class Roo.dd.ScrollManager
21153  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21154  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21155  * @singleton
21156  */
21157 Roo.dd.ScrollManager = function(){
21158     var ddm = Roo.dd.DragDropMgr;
21159     var els = {};
21160     var dragEl = null;
21161     var proc = {};
21162     
21163     
21164     
21165     var onStop = function(e){
21166         dragEl = null;
21167         clearProc();
21168     };
21169     
21170     var triggerRefresh = function(){
21171         if(ddm.dragCurrent){
21172              ddm.refreshCache(ddm.dragCurrent.groups);
21173         }
21174     };
21175     
21176     var doScroll = function(){
21177         if(ddm.dragCurrent){
21178             var dds = Roo.dd.ScrollManager;
21179             if(!dds.animate){
21180                 if(proc.el.scroll(proc.dir, dds.increment)){
21181                     triggerRefresh();
21182                 }
21183             }else{
21184                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21185             }
21186         }
21187     };
21188     
21189     var clearProc = function(){
21190         if(proc.id){
21191             clearInterval(proc.id);
21192         }
21193         proc.id = 0;
21194         proc.el = null;
21195         proc.dir = "";
21196     };
21197     
21198     var startProc = function(el, dir){
21199          Roo.log('scroll startproc');
21200         clearProc();
21201         proc.el = el;
21202         proc.dir = dir;
21203         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21204     };
21205     
21206     var onFire = function(e, isDrop){
21207        
21208         if(isDrop || !ddm.dragCurrent){ return; }
21209         var dds = Roo.dd.ScrollManager;
21210         if(!dragEl || dragEl != ddm.dragCurrent){
21211             dragEl = ddm.dragCurrent;
21212             // refresh regions on drag start
21213             dds.refreshCache();
21214         }
21215         
21216         var xy = Roo.lib.Event.getXY(e);
21217         var pt = new Roo.lib.Point(xy[0], xy[1]);
21218         for(var id in els){
21219             var el = els[id], r = el._region;
21220             if(r && r.contains(pt) && el.isScrollable()){
21221                 if(r.bottom - pt.y <= dds.thresh){
21222                     if(proc.el != el){
21223                         startProc(el, "down");
21224                     }
21225                     return;
21226                 }else if(r.right - pt.x <= dds.thresh){
21227                     if(proc.el != el){
21228                         startProc(el, "left");
21229                     }
21230                     return;
21231                 }else if(pt.y - r.top <= dds.thresh){
21232                     if(proc.el != el){
21233                         startProc(el, "up");
21234                     }
21235                     return;
21236                 }else if(pt.x - r.left <= dds.thresh){
21237                     if(proc.el != el){
21238                         startProc(el, "right");
21239                     }
21240                     return;
21241                 }
21242             }
21243         }
21244         clearProc();
21245     };
21246     
21247     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21248     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21249     
21250     return {
21251         /**
21252          * Registers new overflow element(s) to auto scroll
21253          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21254          */
21255         register : function(el){
21256             if(el instanceof Array){
21257                 for(var i = 0, len = el.length; i < len; i++) {
21258                         this.register(el[i]);
21259                 }
21260             }else{
21261                 el = Roo.get(el);
21262                 els[el.id] = el;
21263             }
21264             Roo.dd.ScrollManager.els = els;
21265         },
21266         
21267         /**
21268          * Unregisters overflow element(s) so they are no longer scrolled
21269          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21270          */
21271         unregister : function(el){
21272             if(el instanceof Array){
21273                 for(var i = 0, len = el.length; i < len; i++) {
21274                         this.unregister(el[i]);
21275                 }
21276             }else{
21277                 el = Roo.get(el);
21278                 delete els[el.id];
21279             }
21280         },
21281         
21282         /**
21283          * The number of pixels from the edge of a container the pointer needs to be to 
21284          * trigger scrolling (defaults to 25)
21285          * @type Number
21286          */
21287         thresh : 25,
21288         
21289         /**
21290          * The number of pixels to scroll in each scroll increment (defaults to 50)
21291          * @type Number
21292          */
21293         increment : 100,
21294         
21295         /**
21296          * The frequency of scrolls in milliseconds (defaults to 500)
21297          * @type Number
21298          */
21299         frequency : 500,
21300         
21301         /**
21302          * True to animate the scroll (defaults to true)
21303          * @type Boolean
21304          */
21305         animate: true,
21306         
21307         /**
21308          * The animation duration in seconds - 
21309          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21310          * @type Number
21311          */
21312         animDuration: .4,
21313         
21314         /**
21315          * Manually trigger a cache refresh.
21316          */
21317         refreshCache : function(){
21318             for(var id in els){
21319                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21320                     els[id]._region = els[id].getRegion();
21321                 }
21322             }
21323         }
21324     };
21325 }();/*
21326  * Based on:
21327  * Ext JS Library 1.1.1
21328  * Copyright(c) 2006-2007, Ext JS, LLC.
21329  *
21330  * Originally Released Under LGPL - original licence link has changed is not relivant.
21331  *
21332  * Fork - LGPL
21333  * <script type="text/javascript">
21334  */
21335  
21336
21337 /**
21338  * @class Roo.dd.Registry
21339  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21340  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21341  * @singleton
21342  */
21343 Roo.dd.Registry = function(){
21344     var elements = {}; 
21345     var handles = {}; 
21346     var autoIdSeed = 0;
21347
21348     var getId = function(el, autogen){
21349         if(typeof el == "string"){
21350             return el;
21351         }
21352         var id = el.id;
21353         if(!id && autogen !== false){
21354             id = "roodd-" + (++autoIdSeed);
21355             el.id = id;
21356         }
21357         return id;
21358     };
21359     
21360     return {
21361     /**
21362      * Register a drag drop element
21363      * @param {String|HTMLElement} element The id or DOM node to register
21364      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21365      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21366      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21367      * populated in the data object (if applicable):
21368      * <pre>
21369 Value      Description<br />
21370 ---------  ------------------------------------------<br />
21371 handles    Array of DOM nodes that trigger dragging<br />
21372            for the element being registered<br />
21373 isHandle   True if the element passed in triggers<br />
21374            dragging itself, else false
21375 </pre>
21376      */
21377         register : function(el, data){
21378             data = data || {};
21379             if(typeof el == "string"){
21380                 el = document.getElementById(el);
21381             }
21382             data.ddel = el;
21383             elements[getId(el)] = data;
21384             if(data.isHandle !== false){
21385                 handles[data.ddel.id] = data;
21386             }
21387             if(data.handles){
21388                 var hs = data.handles;
21389                 for(var i = 0, len = hs.length; i < len; i++){
21390                         handles[getId(hs[i])] = data;
21391                 }
21392             }
21393         },
21394
21395     /**
21396      * Unregister a drag drop element
21397      * @param {String|HTMLElement}  element The id or DOM node to unregister
21398      */
21399         unregister : function(el){
21400             var id = getId(el, false);
21401             var data = elements[id];
21402             if(data){
21403                 delete elements[id];
21404                 if(data.handles){
21405                     var hs = data.handles;
21406                     for(var i = 0, len = hs.length; i < len; i++){
21407                         delete handles[getId(hs[i], false)];
21408                     }
21409                 }
21410             }
21411         },
21412
21413     /**
21414      * Returns the handle registered for a DOM Node by id
21415      * @param {String|HTMLElement} id The DOM node or id to look up
21416      * @return {Object} handle The custom handle data
21417      */
21418         getHandle : function(id){
21419             if(typeof id != "string"){ // must be element?
21420                 id = id.id;
21421             }
21422             return handles[id];
21423         },
21424
21425     /**
21426      * Returns the handle that is registered for the DOM node that is the target of the event
21427      * @param {Event} e The event
21428      * @return {Object} handle The custom handle data
21429      */
21430         getHandleFromEvent : function(e){
21431             var t = Roo.lib.Event.getTarget(e);
21432             return t ? handles[t.id] : null;
21433         },
21434
21435     /**
21436      * Returns a custom data object that is registered for a DOM node by id
21437      * @param {String|HTMLElement} id The DOM node or id to look up
21438      * @return {Object} data The custom data
21439      */
21440         getTarget : function(id){
21441             if(typeof id != "string"){ // must be element?
21442                 id = id.id;
21443             }
21444             return elements[id];
21445         },
21446
21447     /**
21448      * Returns a custom data object that is registered for the DOM node that is the target of the event
21449      * @param {Event} e The event
21450      * @return {Object} data The custom data
21451      */
21452         getTargetFromEvent : function(e){
21453             var t = Roo.lib.Event.getTarget(e);
21454             return t ? elements[t.id] || handles[t.id] : null;
21455         }
21456     };
21457 }();/*
21458  * Based on:
21459  * Ext JS Library 1.1.1
21460  * Copyright(c) 2006-2007, Ext JS, LLC.
21461  *
21462  * Originally Released Under LGPL - original licence link has changed is not relivant.
21463  *
21464  * Fork - LGPL
21465  * <script type="text/javascript">
21466  */
21467  
21468
21469 /**
21470  * @class Roo.dd.StatusProxy
21471  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21472  * default drag proxy used by all Roo.dd components.
21473  * @constructor
21474  * @param {Object} config
21475  */
21476 Roo.dd.StatusProxy = function(config){
21477     Roo.apply(this, config);
21478     this.id = this.id || Roo.id();
21479     this.el = new Roo.Layer({
21480         dh: {
21481             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21482                 {tag: "div", cls: "x-dd-drop-icon"},
21483                 {tag: "div", cls: "x-dd-drag-ghost"}
21484             ]
21485         }, 
21486         shadow: !config || config.shadow !== false
21487     });
21488     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21489     this.dropStatus = this.dropNotAllowed;
21490 };
21491
21492 Roo.dd.StatusProxy.prototype = {
21493     /**
21494      * @cfg {String} dropAllowed
21495      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21496      */
21497     dropAllowed : "x-dd-drop-ok",
21498     /**
21499      * @cfg {String} dropNotAllowed
21500      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21501      */
21502     dropNotAllowed : "x-dd-drop-nodrop",
21503
21504     /**
21505      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21506      * over the current target element.
21507      * @param {String} cssClass The css class for the new drop status indicator image
21508      */
21509     setStatus : function(cssClass){
21510         cssClass = cssClass || this.dropNotAllowed;
21511         if(this.dropStatus != cssClass){
21512             this.el.replaceClass(this.dropStatus, cssClass);
21513             this.dropStatus = cssClass;
21514         }
21515     },
21516
21517     /**
21518      * Resets the status indicator to the default dropNotAllowed value
21519      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21520      */
21521     reset : function(clearGhost){
21522         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21523         this.dropStatus = this.dropNotAllowed;
21524         if(clearGhost){
21525             this.ghost.update("");
21526         }
21527     },
21528
21529     /**
21530      * Updates the contents of the ghost element
21531      * @param {String} html The html that will replace the current innerHTML of the ghost element
21532      */
21533     update : function(html){
21534         if(typeof html == "string"){
21535             this.ghost.update(html);
21536         }else{
21537             this.ghost.update("");
21538             html.style.margin = "0";
21539             this.ghost.dom.appendChild(html);
21540         }
21541         // ensure float = none set?? cant remember why though.
21542         var el = this.ghost.dom.firstChild;
21543                 if(el){
21544                         Roo.fly(el).setStyle('float', 'none');
21545                 }
21546     },
21547     
21548     /**
21549      * Returns the underlying proxy {@link Roo.Layer}
21550      * @return {Roo.Layer} el
21551     */
21552     getEl : function(){
21553         return this.el;
21554     },
21555
21556     /**
21557      * Returns the ghost element
21558      * @return {Roo.Element} el
21559      */
21560     getGhost : function(){
21561         return this.ghost;
21562     },
21563
21564     /**
21565      * Hides the proxy
21566      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21567      */
21568     hide : function(clear){
21569         this.el.hide();
21570         if(clear){
21571             this.reset(true);
21572         }
21573     },
21574
21575     /**
21576      * Stops the repair animation if it's currently running
21577      */
21578     stop : function(){
21579         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21580             this.anim.stop();
21581         }
21582     },
21583
21584     /**
21585      * Displays this proxy
21586      */
21587     show : function(){
21588         this.el.show();
21589     },
21590
21591     /**
21592      * Force the Layer to sync its shadow and shim positions to the element
21593      */
21594     sync : function(){
21595         this.el.sync();
21596     },
21597
21598     /**
21599      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21600      * invalid drop operation by the item being dragged.
21601      * @param {Array} xy The XY position of the element ([x, y])
21602      * @param {Function} callback The function to call after the repair is complete
21603      * @param {Object} scope The scope in which to execute the callback
21604      */
21605     repair : function(xy, callback, scope){
21606         this.callback = callback;
21607         this.scope = scope;
21608         if(xy && this.animRepair !== false){
21609             this.el.addClass("x-dd-drag-repair");
21610             this.el.hideUnders(true);
21611             this.anim = this.el.shift({
21612                 duration: this.repairDuration || .5,
21613                 easing: 'easeOut',
21614                 xy: xy,
21615                 stopFx: true,
21616                 callback: this.afterRepair,
21617                 scope: this
21618             });
21619         }else{
21620             this.afterRepair();
21621         }
21622     },
21623
21624     // private
21625     afterRepair : function(){
21626         this.hide(true);
21627         if(typeof this.callback == "function"){
21628             this.callback.call(this.scope || this);
21629         }
21630         this.callback = null;
21631         this.scope = null;
21632     }
21633 };/*
21634  * Based on:
21635  * Ext JS Library 1.1.1
21636  * Copyright(c) 2006-2007, Ext JS, LLC.
21637  *
21638  * Originally Released Under LGPL - original licence link has changed is not relivant.
21639  *
21640  * Fork - LGPL
21641  * <script type="text/javascript">
21642  */
21643
21644 /**
21645  * @class Roo.dd.DragSource
21646  * @extends Roo.dd.DDProxy
21647  * A simple class that provides the basic implementation needed to make any element draggable.
21648  * @constructor
21649  * @param {String/HTMLElement/Element} el The container element
21650  * @param {Object} config
21651  */
21652 Roo.dd.DragSource = function(el, config){
21653     this.el = Roo.get(el);
21654     this.dragData = {};
21655     
21656     Roo.apply(this, config);
21657     
21658     if(!this.proxy){
21659         this.proxy = new Roo.dd.StatusProxy();
21660     }
21661
21662     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21663           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21664     
21665     this.dragging = false;
21666 };
21667
21668 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21669     /**
21670      * @cfg {String} dropAllowed
21671      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21672      */
21673     dropAllowed : "x-dd-drop-ok",
21674     /**
21675      * @cfg {String} dropNotAllowed
21676      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21677      */
21678     dropNotAllowed : "x-dd-drop-nodrop",
21679
21680     /**
21681      * Returns the data object associated with this drag source
21682      * @return {Object} data An object containing arbitrary data
21683      */
21684     getDragData : function(e){
21685         return this.dragData;
21686     },
21687
21688     // private
21689     onDragEnter : function(e, id){
21690         var target = Roo.dd.DragDropMgr.getDDById(id);
21691         this.cachedTarget = target;
21692         if(this.beforeDragEnter(target, e, id) !== false){
21693             if(target.isNotifyTarget){
21694                 var status = target.notifyEnter(this, e, this.dragData);
21695                 this.proxy.setStatus(status);
21696             }else{
21697                 this.proxy.setStatus(this.dropAllowed);
21698             }
21699             
21700             if(this.afterDragEnter){
21701                 /**
21702                  * An empty function by default, but provided so that you can perform a custom action
21703                  * when the dragged item enters the drop target by providing an implementation.
21704                  * @param {Roo.dd.DragDrop} target The drop target
21705                  * @param {Event} e The event object
21706                  * @param {String} id The id of the dragged element
21707                  * @method afterDragEnter
21708                  */
21709                 this.afterDragEnter(target, e, id);
21710             }
21711         }
21712     },
21713
21714     /**
21715      * An empty function by default, but provided so that you can perform a custom action
21716      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
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      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21721      */
21722     beforeDragEnter : function(target, e, id){
21723         return true;
21724     },
21725
21726     // private
21727     alignElWithMouse: function() {
21728         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21729         this.proxy.sync();
21730     },
21731
21732     // private
21733     onDragOver : function(e, id){
21734         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21735         if(this.beforeDragOver(target, e, id) !== false){
21736             if(target.isNotifyTarget){
21737                 var status = target.notifyOver(this, e, this.dragData);
21738                 this.proxy.setStatus(status);
21739             }
21740
21741             if(this.afterDragOver){
21742                 /**
21743                  * An empty function by default, but provided so that you can perform a custom action
21744                  * while the dragged item is over the drop target by providing an implementation.
21745                  * @param {Roo.dd.DragDrop} target The drop target
21746                  * @param {Event} e The event object
21747                  * @param {String} id The id of the dragged element
21748                  * @method afterDragOver
21749                  */
21750                 this.afterDragOver(target, e, id);
21751             }
21752         }
21753     },
21754
21755     /**
21756      * An empty function by default, but provided so that you can perform a custom action
21757      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21758      * @param {Roo.dd.DragDrop} target The drop target
21759      * @param {Event} e The event object
21760      * @param {String} id The id of the dragged element
21761      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21762      */
21763     beforeDragOver : function(target, e, id){
21764         return true;
21765     },
21766
21767     // private
21768     onDragOut : function(e, id){
21769         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21770         if(this.beforeDragOut(target, e, id) !== false){
21771             if(target.isNotifyTarget){
21772                 target.notifyOut(this, e, this.dragData);
21773             }
21774             this.proxy.reset();
21775             if(this.afterDragOut){
21776                 /**
21777                  * An empty function by default, but provided so that you can perform a custom action
21778                  * after the dragged item is dragged out of the target without dropping.
21779                  * @param {Roo.dd.DragDrop} target The drop target
21780                  * @param {Event} e The event object
21781                  * @param {String} id The id of the dragged element
21782                  * @method afterDragOut
21783                  */
21784                 this.afterDragOut(target, e, id);
21785             }
21786         }
21787         this.cachedTarget = null;
21788     },
21789
21790     /**
21791      * An empty function by default, but provided so that you can perform a custom action before the dragged
21792      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21793      * @param {Roo.dd.DragDrop} target The drop target
21794      * @param {Event} e The event object
21795      * @param {String} id The id of the dragged element
21796      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21797      */
21798     beforeDragOut : function(target, e, id){
21799         return true;
21800     },
21801     
21802     // private
21803     onDragDrop : function(e, id){
21804         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21805         if(this.beforeDragDrop(target, e, id) !== false){
21806             if(target.isNotifyTarget){
21807                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21808                     this.onValidDrop(target, e, id);
21809                 }else{
21810                     this.onInvalidDrop(target, e, id);
21811                 }
21812             }else{
21813                 this.onValidDrop(target, e, id);
21814             }
21815             
21816             if(this.afterDragDrop){
21817                 /**
21818                  * An empty function by default, but provided so that you can perform a custom action
21819                  * after a valid drag drop has occurred by providing an implementation.
21820                  * @param {Roo.dd.DragDrop} target The drop target
21821                  * @param {Event} e The event object
21822                  * @param {String} id The id of the dropped element
21823                  * @method afterDragDrop
21824                  */
21825                 this.afterDragDrop(target, e, id);
21826             }
21827         }
21828         delete this.cachedTarget;
21829     },
21830
21831     /**
21832      * An empty function by default, but provided so that you can perform a custom action before the dragged
21833      * item is dropped onto the target and optionally cancel the onDragDrop.
21834      * @param {Roo.dd.DragDrop} target The drop target
21835      * @param {Event} e The event object
21836      * @param {String} id The id of the dragged element
21837      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21838      */
21839     beforeDragDrop : function(target, e, id){
21840         return true;
21841     },
21842
21843     // private
21844     onValidDrop : function(target, e, id){
21845         this.hideProxy();
21846         if(this.afterValidDrop){
21847             /**
21848              * An empty function by default, but provided so that you can perform a custom action
21849              * after a valid drop has occurred by providing an implementation.
21850              * @param {Object} target The target DD 
21851              * @param {Event} e The event object
21852              * @param {String} id The id of the dropped element
21853              * @method afterInvalidDrop
21854              */
21855             this.afterValidDrop(target, e, id);
21856         }
21857     },
21858
21859     // private
21860     getRepairXY : function(e, data){
21861         return this.el.getXY();  
21862     },
21863
21864     // private
21865     onInvalidDrop : function(target, e, id){
21866         this.beforeInvalidDrop(target, e, id);
21867         if(this.cachedTarget){
21868             if(this.cachedTarget.isNotifyTarget){
21869                 this.cachedTarget.notifyOut(this, e, this.dragData);
21870             }
21871             this.cacheTarget = null;
21872         }
21873         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21874
21875         if(this.afterInvalidDrop){
21876             /**
21877              * An empty function by default, but provided so that you can perform a custom action
21878              * after an invalid drop has occurred by providing an implementation.
21879              * @param {Event} e The event object
21880              * @param {String} id The id of the dropped element
21881              * @method afterInvalidDrop
21882              */
21883             this.afterInvalidDrop(e, id);
21884         }
21885     },
21886
21887     // private
21888     afterRepair : function(){
21889         if(Roo.enableFx){
21890             this.el.highlight(this.hlColor || "c3daf9");
21891         }
21892         this.dragging = false;
21893     },
21894
21895     /**
21896      * An empty function by default, but provided so that you can perform a custom action after an invalid
21897      * drop has occurred.
21898      * @param {Roo.dd.DragDrop} target The drop target
21899      * @param {Event} e The event object
21900      * @param {String} id The id of the dragged element
21901      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21902      */
21903     beforeInvalidDrop : function(target, e, id){
21904         return true;
21905     },
21906
21907     // private
21908     handleMouseDown : function(e){
21909         if(this.dragging) {
21910             return;
21911         }
21912         var data = this.getDragData(e);
21913         if(data && this.onBeforeDrag(data, e) !== false){
21914             this.dragData = data;
21915             this.proxy.stop();
21916             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21917         } 
21918     },
21919
21920     /**
21921      * An empty function by default, but provided so that you can perform a custom action before the initial
21922      * drag event begins and optionally cancel it.
21923      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21924      * @param {Event} e The event object
21925      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21926      */
21927     onBeforeDrag : function(data, e){
21928         return true;
21929     },
21930
21931     /**
21932      * An empty function by default, but provided so that you can perform a custom action once the initial
21933      * drag event has begun.  The drag cannot be canceled from this function.
21934      * @param {Number} x The x position of the click on the dragged object
21935      * @param {Number} y The y position of the click on the dragged object
21936      */
21937     onStartDrag : Roo.emptyFn,
21938
21939     // private - YUI override
21940     startDrag : function(x, y){
21941         this.proxy.reset();
21942         this.dragging = true;
21943         this.proxy.update("");
21944         this.onInitDrag(x, y);
21945         this.proxy.show();
21946     },
21947
21948     // private
21949     onInitDrag : function(x, y){
21950         var clone = this.el.dom.cloneNode(true);
21951         clone.id = Roo.id(); // prevent duplicate ids
21952         this.proxy.update(clone);
21953         this.onStartDrag(x, y);
21954         return true;
21955     },
21956
21957     /**
21958      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21959      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21960      */
21961     getProxy : function(){
21962         return this.proxy;  
21963     },
21964
21965     /**
21966      * Hides the drag source's {@link Roo.dd.StatusProxy}
21967      */
21968     hideProxy : function(){
21969         this.proxy.hide();  
21970         this.proxy.reset(true);
21971         this.dragging = false;
21972     },
21973
21974     // private
21975     triggerCacheRefresh : function(){
21976         Roo.dd.DDM.refreshCache(this.groups);
21977     },
21978
21979     // private - override to prevent hiding
21980     b4EndDrag: function(e) {
21981     },
21982
21983     // private - override to prevent moving
21984     endDrag : function(e){
21985         this.onEndDrag(this.dragData, e);
21986     },
21987
21988     // private
21989     onEndDrag : function(data, e){
21990     },
21991     
21992     // private - pin to cursor
21993     autoOffset : function(x, y) {
21994         this.setDelta(-12, -20);
21995     }    
21996 });/*
21997  * Based on:
21998  * Ext JS Library 1.1.1
21999  * Copyright(c) 2006-2007, Ext JS, LLC.
22000  *
22001  * Originally Released Under LGPL - original licence link has changed is not relivant.
22002  *
22003  * Fork - LGPL
22004  * <script type="text/javascript">
22005  */
22006
22007
22008 /**
22009  * @class Roo.dd.DropTarget
22010  * @extends Roo.dd.DDTarget
22011  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22012  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22013  * @constructor
22014  * @param {String/HTMLElement/Element} el The container element
22015  * @param {Object} config
22016  */
22017 Roo.dd.DropTarget = function(el, config){
22018     this.el = Roo.get(el);
22019     
22020     var listeners = false; ;
22021     if (config && config.listeners) {
22022         listeners= config.listeners;
22023         delete config.listeners;
22024     }
22025     Roo.apply(this, config);
22026     
22027     if(this.containerScroll){
22028         Roo.dd.ScrollManager.register(this.el);
22029     }
22030     this.addEvents( {
22031          /**
22032          * @scope Roo.dd.DropTarget
22033          */
22034          
22035          /**
22036          * @event enter
22037          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22038          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22039          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22040          * 
22041          * IMPORTANT : it should set this.overClass and this.dropAllowed
22042          * 
22043          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22044          * @param {Event} e The event
22045          * @param {Object} data An object containing arbitrary data supplied by the drag source
22046          */
22047         "enter" : true,
22048         
22049          /**
22050          * @event over
22051          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22052          * This method will be called on every mouse movement while the drag source is over the drop target.
22053          * This default implementation simply returns the dropAllowed config value.
22054          * 
22055          * IMPORTANT : it should set this.dropAllowed
22056          * 
22057          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22058          * @param {Event} e The event
22059          * @param {Object} data An object containing arbitrary data supplied by the drag source
22060          
22061          */
22062         "over" : true,
22063         /**
22064          * @event out
22065          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22066          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22067          * overClass (if any) from the drop element.
22068          * 
22069          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22070          * @param {Event} e The event
22071          * @param {Object} data An object containing arbitrary data supplied by the drag source
22072          */
22073          "out" : true,
22074          
22075         /**
22076          * @event drop
22077          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22078          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22079          * implementation that does something to process the drop event and returns true so that the drag source's
22080          * repair action does not run.
22081          * 
22082          * IMPORTANT : it should set this.success
22083          * 
22084          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22085          * @param {Event} e The event
22086          * @param {Object} data An object containing arbitrary data supplied by the drag source
22087         */
22088          "drop" : true
22089     });
22090             
22091      
22092     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22093         this.el.dom, 
22094         this.ddGroup || this.group,
22095         {
22096             isTarget: true,
22097             listeners : listeners || {} 
22098            
22099         
22100         }
22101     );
22102
22103 };
22104
22105 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22106     /**
22107      * @cfg {String} overClass
22108      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22109      */
22110      /**
22111      * @cfg {String} ddGroup
22112      * The drag drop group to handle drop events for
22113      */
22114      
22115     /**
22116      * @cfg {String} dropAllowed
22117      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22118      */
22119     dropAllowed : "x-dd-drop-ok",
22120     /**
22121      * @cfg {String} dropNotAllowed
22122      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22123      */
22124     dropNotAllowed : "x-dd-drop-nodrop",
22125     /**
22126      * @cfg {boolean} success
22127      * set this after drop listener.. 
22128      */
22129     success : false,
22130     /**
22131      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22132      * if the drop point is valid for over/enter..
22133      */
22134     valid : false,
22135     // private
22136     isTarget : true,
22137
22138     // private
22139     isNotifyTarget : true,
22140     
22141     /**
22142      * @hide
22143      */
22144     notifyEnter : function(dd, e, data)
22145     {
22146         this.valid = true;
22147         this.fireEvent('enter', dd, e, data);
22148         if(this.overClass){
22149             this.el.addClass(this.overClass);
22150         }
22151         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22152             this.valid ? this.dropAllowed : this.dropNotAllowed
22153         );
22154     },
22155
22156     /**
22157      * @hide
22158      */
22159     notifyOver : function(dd, e, data)
22160     {
22161         this.valid = true;
22162         this.fireEvent('over', dd, e, data);
22163         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22164             this.valid ? this.dropAllowed : this.dropNotAllowed
22165         );
22166     },
22167
22168     /**
22169      * @hide
22170      */
22171     notifyOut : function(dd, e, data)
22172     {
22173         this.fireEvent('out', dd, e, data);
22174         if(this.overClass){
22175             this.el.removeClass(this.overClass);
22176         }
22177     },
22178
22179     /**
22180      * @hide
22181      */
22182     notifyDrop : function(dd, e, data)
22183     {
22184         this.success = false;
22185         this.fireEvent('drop', dd, e, data);
22186         return this.success;
22187     }
22188 });/*
22189  * Based on:
22190  * Ext JS Library 1.1.1
22191  * Copyright(c) 2006-2007, Ext JS, LLC.
22192  *
22193  * Originally Released Under LGPL - original licence link has changed is not relivant.
22194  *
22195  * Fork - LGPL
22196  * <script type="text/javascript">
22197  */
22198
22199
22200 /**
22201  * @class Roo.dd.DragZone
22202  * @extends Roo.dd.DragSource
22203  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22204  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22205  * @constructor
22206  * @param {String/HTMLElement/Element} el The container element
22207  * @param {Object} config
22208  */
22209 Roo.dd.DragZone = function(el, config){
22210     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22211     if(this.containerScroll){
22212         Roo.dd.ScrollManager.register(this.el);
22213     }
22214 };
22215
22216 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22217     /**
22218      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22219      * for auto scrolling during drag operations.
22220      */
22221     /**
22222      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22223      * method after a failed drop (defaults to "c3daf9" - light blue)
22224      */
22225
22226     /**
22227      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22228      * for a valid target to drag based on the mouse down. Override this method
22229      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22230      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22231      * @param {EventObject} e The mouse down event
22232      * @return {Object} The dragData
22233      */
22234     getDragData : function(e){
22235         return Roo.dd.Registry.getHandleFromEvent(e);
22236     },
22237     
22238     /**
22239      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22240      * this.dragData.ddel
22241      * @param {Number} x The x position of the click on the dragged object
22242      * @param {Number} y The y position of the click on the dragged object
22243      * @return {Boolean} true to continue the drag, false to cancel
22244      */
22245     onInitDrag : function(x, y){
22246         this.proxy.update(this.dragData.ddel.cloneNode(true));
22247         this.onStartDrag(x, y);
22248         return true;
22249     },
22250     
22251     /**
22252      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22253      */
22254     afterRepair : function(){
22255         if(Roo.enableFx){
22256             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22257         }
22258         this.dragging = false;
22259     },
22260
22261     /**
22262      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22263      * the XY of this.dragData.ddel
22264      * @param {EventObject} e The mouse up event
22265      * @return {Array} The xy location (e.g. [100, 200])
22266      */
22267     getRepairXY : function(e){
22268         return Roo.Element.fly(this.dragData.ddel).getXY();  
22269     }
22270 });/*
22271  * Based on:
22272  * Ext JS Library 1.1.1
22273  * Copyright(c) 2006-2007, Ext JS, LLC.
22274  *
22275  * Originally Released Under LGPL - original licence link has changed is not relivant.
22276  *
22277  * Fork - LGPL
22278  * <script type="text/javascript">
22279  */
22280 /**
22281  * @class Roo.dd.DropZone
22282  * @extends Roo.dd.DropTarget
22283  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22284  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22285  * @constructor
22286  * @param {String/HTMLElement/Element} el The container element
22287  * @param {Object} config
22288  */
22289 Roo.dd.DropZone = function(el, config){
22290     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22291 };
22292
22293 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22294     /**
22295      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22296      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22297      * provide your own custom lookup.
22298      * @param {Event} e The event
22299      * @return {Object} data The custom data
22300      */
22301     getTargetFromEvent : function(e){
22302         return Roo.dd.Registry.getTargetFromEvent(e);
22303     },
22304
22305     /**
22306      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22307      * that it has registered.  This method has no default implementation and should be overridden to provide
22308      * node-specific processing if necessary.
22309      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22310      * {@link #getTargetFromEvent} for this node)
22311      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22312      * @param {Event} e The event
22313      * @param {Object} data An object containing arbitrary data supplied by the drag source
22314      */
22315     onNodeEnter : function(n, dd, e, data){
22316         
22317     },
22318
22319     /**
22320      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22321      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22322      * overridden to provide the proper feedback.
22323      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22324      * {@link #getTargetFromEvent} for this node)
22325      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22326      * @param {Event} e The event
22327      * @param {Object} data An object containing arbitrary data supplied by the drag source
22328      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22329      * underlying {@link Roo.dd.StatusProxy} can be updated
22330      */
22331     onNodeOver : function(n, dd, e, data){
22332         return this.dropAllowed;
22333     },
22334
22335     /**
22336      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22337      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22338      * node-specific processing if necessary.
22339      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22340      * {@link #getTargetFromEvent} for this node)
22341      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22342      * @param {Event} e The event
22343      * @param {Object} data An object containing arbitrary data supplied by the drag source
22344      */
22345     onNodeOut : function(n, dd, e, data){
22346         
22347     },
22348
22349     /**
22350      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22351      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22352      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22353      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22354      * {@link #getTargetFromEvent} for this node)
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     onNodeDrop : function(n, dd, e, data){
22361         return false;
22362     },
22363
22364     /**
22365      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22366      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22367      * it should be overridden to provide the proper feedback if necessary.
22368      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22369      * @param {Event} e The event
22370      * @param {Object} data An object containing arbitrary data supplied by the drag source
22371      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22372      * underlying {@link Roo.dd.StatusProxy} can be updated
22373      */
22374     onContainerOver : function(dd, e, data){
22375         return this.dropNotAllowed;
22376     },
22377
22378     /**
22379      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22380      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22381      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22382      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22383      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22384      * @param {Event} e The event
22385      * @param {Object} data An object containing arbitrary data supplied by the drag source
22386      * @return {Boolean} True if the drop was valid, else false
22387      */
22388     onContainerDrop : function(dd, e, data){
22389         return false;
22390     },
22391
22392     /**
22393      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22394      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22395      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22396      * you should override this method and provide a custom implementation.
22397      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22398      * @param {Event} e The event
22399      * @param {Object} data An object containing arbitrary data supplied by the drag source
22400      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22401      * underlying {@link Roo.dd.StatusProxy} can be updated
22402      */
22403     notifyEnter : function(dd, e, data){
22404         return this.dropNotAllowed;
22405     },
22406
22407     /**
22408      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22409      * This method will be called on every mouse movement while the drag source is over the drop zone.
22410      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22411      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22412      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22413      * registered node, it will call {@link #onContainerOver}.
22414      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22415      * @param {Event} e The event
22416      * @param {Object} data An object containing arbitrary data supplied by the drag source
22417      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22418      * underlying {@link Roo.dd.StatusProxy} can be updated
22419      */
22420     notifyOver : function(dd, e, data){
22421         var n = this.getTargetFromEvent(e);
22422         if(!n){ // not over valid drop target
22423             if(this.lastOverNode){
22424                 this.onNodeOut(this.lastOverNode, dd, e, data);
22425                 this.lastOverNode = null;
22426             }
22427             return this.onContainerOver(dd, e, data);
22428         }
22429         if(this.lastOverNode != n){
22430             if(this.lastOverNode){
22431                 this.onNodeOut(this.lastOverNode, dd, e, data);
22432             }
22433             this.onNodeEnter(n, dd, e, data);
22434             this.lastOverNode = n;
22435         }
22436         return this.onNodeOver(n, dd, e, data);
22437     },
22438
22439     /**
22440      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22441      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22442      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22443      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22444      * @param {Event} e The event
22445      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22446      */
22447     notifyOut : function(dd, e, data){
22448         if(this.lastOverNode){
22449             this.onNodeOut(this.lastOverNode, dd, e, data);
22450             this.lastOverNode = null;
22451         }
22452     },
22453
22454     /**
22455      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22456      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22457      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22458      * otherwise it will call {@link #onContainerDrop}.
22459      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22460      * @param {Event} e The event
22461      * @param {Object} data An object containing arbitrary data supplied by the drag source
22462      * @return {Boolean} True if the drop was valid, else false
22463      */
22464     notifyDrop : function(dd, e, data){
22465         if(this.lastOverNode){
22466             this.onNodeOut(this.lastOverNode, dd, e, data);
22467             this.lastOverNode = null;
22468         }
22469         var n = this.getTargetFromEvent(e);
22470         return n ?
22471             this.onNodeDrop(n, dd, e, data) :
22472             this.onContainerDrop(dd, e, data);
22473     },
22474
22475     // private
22476     triggerCacheRefresh : function(){
22477         Roo.dd.DDM.refreshCache(this.groups);
22478     }  
22479 });/*
22480  * Based on:
22481  * Ext JS Library 1.1.1
22482  * Copyright(c) 2006-2007, Ext JS, LLC.
22483  *
22484  * Originally Released Under LGPL - original licence link has changed is not relivant.
22485  *
22486  * Fork - LGPL
22487  * <script type="text/javascript">
22488  */
22489
22490
22491 /**
22492  * @class Roo.data.SortTypes
22493  * @singleton
22494  * Defines the default sorting (casting?) comparison functions used when sorting data.
22495  */
22496 Roo.data.SortTypes = {
22497     /**
22498      * Default sort that does nothing
22499      * @param {Mixed} s The value being converted
22500      * @return {Mixed} The comparison value
22501      */
22502     none : function(s){
22503         return s;
22504     },
22505     
22506     /**
22507      * The regular expression used to strip tags
22508      * @type {RegExp}
22509      * @property
22510      */
22511     stripTagsRE : /<\/?[^>]+>/gi,
22512     
22513     /**
22514      * Strips all HTML tags to sort on text only
22515      * @param {Mixed} s The value being converted
22516      * @return {String} The comparison value
22517      */
22518     asText : function(s){
22519         return String(s).replace(this.stripTagsRE, "");
22520     },
22521     
22522     /**
22523      * Strips all HTML tags to sort on text only - Case insensitive
22524      * @param {Mixed} s The value being converted
22525      * @return {String} The comparison value
22526      */
22527     asUCText : function(s){
22528         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22529     },
22530     
22531     /**
22532      * Case insensitive string
22533      * @param {Mixed} s The value being converted
22534      * @return {String} The comparison value
22535      */
22536     asUCString : function(s) {
22537         return String(s).toUpperCase();
22538     },
22539     
22540     /**
22541      * Date sorting
22542      * @param {Mixed} s The value being converted
22543      * @return {Number} The comparison value
22544      */
22545     asDate : function(s) {
22546         if(!s){
22547             return 0;
22548         }
22549         if(s instanceof Date){
22550             return s.getTime();
22551         }
22552         return Date.parse(String(s));
22553     },
22554     
22555     /**
22556      * Float sorting
22557      * @param {Mixed} s The value being converted
22558      * @return {Float} The comparison value
22559      */
22560     asFloat : function(s) {
22561         var val = parseFloat(String(s).replace(/,/g, ""));
22562         if(isNaN(val)) {
22563             val = 0;
22564         }
22565         return val;
22566     },
22567     
22568     /**
22569      * Integer sorting
22570      * @param {Mixed} s The value being converted
22571      * @return {Number} The comparison value
22572      */
22573     asInt : function(s) {
22574         var val = parseInt(String(s).replace(/,/g, ""));
22575         if(isNaN(val)) {
22576             val = 0;
22577         }
22578         return val;
22579     }
22580 };/*
22581  * Based on:
22582  * Ext JS Library 1.1.1
22583  * Copyright(c) 2006-2007, Ext JS, LLC.
22584  *
22585  * Originally Released Under LGPL - original licence link has changed is not relivant.
22586  *
22587  * Fork - LGPL
22588  * <script type="text/javascript">
22589  */
22590
22591 /**
22592 * @class Roo.data.Record
22593  * Instances of this class encapsulate both record <em>definition</em> information, and record
22594  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22595  * to access Records cached in an {@link Roo.data.Store} object.<br>
22596  * <p>
22597  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22598  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22599  * objects.<br>
22600  * <p>
22601  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22602  * @constructor
22603  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22604  * {@link #create}. The parameters are the same.
22605  * @param {Array} data An associative Array of data values keyed by the field name.
22606  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22607  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22608  * not specified an integer id is generated.
22609  */
22610 Roo.data.Record = function(data, id){
22611     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22612     this.data = data;
22613 };
22614
22615 /**
22616  * Generate a constructor for a specific record layout.
22617  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22618  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22619  * Each field definition object may contain the following properties: <ul>
22620  * <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,
22621  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22622  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22623  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22624  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22625  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22626  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22627  * this may be omitted.</p></li>
22628  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22629  * <ul><li>auto (Default, implies no conversion)</li>
22630  * <li>string</li>
22631  * <li>int</li>
22632  * <li>float</li>
22633  * <li>boolean</li>
22634  * <li>date</li></ul></p></li>
22635  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22636  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22637  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22638  * by the Reader into an object that will be stored in the Record. It is passed the
22639  * following parameters:<ul>
22640  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22641  * </ul></p></li>
22642  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22643  * </ul>
22644  * <br>usage:<br><pre><code>
22645 var TopicRecord = Roo.data.Record.create(
22646     {name: 'title', mapping: 'topic_title'},
22647     {name: 'author', mapping: 'username'},
22648     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22649     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22650     {name: 'lastPoster', mapping: 'user2'},
22651     {name: 'excerpt', mapping: 'post_text'}
22652 );
22653
22654 var myNewRecord = new TopicRecord({
22655     title: 'Do my job please',
22656     author: 'noobie',
22657     totalPosts: 1,
22658     lastPost: new Date(),
22659     lastPoster: 'Animal',
22660     excerpt: 'No way dude!'
22661 });
22662 myStore.add(myNewRecord);
22663 </code></pre>
22664  * @method create
22665  * @static
22666  */
22667 Roo.data.Record.create = function(o){
22668     var f = function(){
22669         f.superclass.constructor.apply(this, arguments);
22670     };
22671     Roo.extend(f, Roo.data.Record);
22672     var p = f.prototype;
22673     p.fields = new Roo.util.MixedCollection(false, function(field){
22674         return field.name;
22675     });
22676     for(var i = 0, len = o.length; i < len; i++){
22677         p.fields.add(new Roo.data.Field(o[i]));
22678     }
22679     f.getField = function(name){
22680         return p.fields.get(name);  
22681     };
22682     return f;
22683 };
22684
22685 Roo.data.Record.AUTO_ID = 1000;
22686 Roo.data.Record.EDIT = 'edit';
22687 Roo.data.Record.REJECT = 'reject';
22688 Roo.data.Record.COMMIT = 'commit';
22689
22690 Roo.data.Record.prototype = {
22691     /**
22692      * Readonly flag - true if this record has been modified.
22693      * @type Boolean
22694      */
22695     dirty : false,
22696     editing : false,
22697     error: null,
22698     modified: null,
22699
22700     // private
22701     join : function(store){
22702         this.store = store;
22703     },
22704
22705     /**
22706      * Set the named field to the specified value.
22707      * @param {String} name The name of the field to set.
22708      * @param {Object} value The value to set the field to.
22709      */
22710     set : function(name, value){
22711         if(this.data[name] == value){
22712             return;
22713         }
22714         this.dirty = true;
22715         if(!this.modified){
22716             this.modified = {};
22717         }
22718         if(typeof this.modified[name] == 'undefined'){
22719             this.modified[name] = this.data[name];
22720         }
22721         this.data[name] = value;
22722         if(!this.editing && this.store){
22723             this.store.afterEdit(this);
22724         }       
22725     },
22726
22727     /**
22728      * Get the value of the named field.
22729      * @param {String} name The name of the field to get the value of.
22730      * @return {Object} The value of the field.
22731      */
22732     get : function(name){
22733         return this.data[name]; 
22734     },
22735
22736     // private
22737     beginEdit : function(){
22738         this.editing = true;
22739         this.modified = {}; 
22740     },
22741
22742     // private
22743     cancelEdit : function(){
22744         this.editing = false;
22745         delete this.modified;
22746     },
22747
22748     // private
22749     endEdit : function(){
22750         this.editing = false;
22751         if(this.dirty && this.store){
22752             this.store.afterEdit(this);
22753         }
22754     },
22755
22756     /**
22757      * Usually called by the {@link Roo.data.Store} which owns the Record.
22758      * Rejects all changes made to the Record since either creation, or the last commit operation.
22759      * Modified fields are reverted to their original values.
22760      * <p>
22761      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22762      * of reject operations.
22763      */
22764     reject : function(){
22765         var m = this.modified;
22766         for(var n in m){
22767             if(typeof m[n] != "function"){
22768                 this.data[n] = m[n];
22769             }
22770         }
22771         this.dirty = false;
22772         delete this.modified;
22773         this.editing = false;
22774         if(this.store){
22775             this.store.afterReject(this);
22776         }
22777     },
22778
22779     /**
22780      * Usually called by the {@link Roo.data.Store} which owns the Record.
22781      * Commits all changes made to the Record since either creation, or the last commit operation.
22782      * <p>
22783      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22784      * of commit operations.
22785      */
22786     commit : function(){
22787         this.dirty = false;
22788         delete this.modified;
22789         this.editing = false;
22790         if(this.store){
22791             this.store.afterCommit(this);
22792         }
22793     },
22794
22795     // private
22796     hasError : function(){
22797         return this.error != null;
22798     },
22799
22800     // private
22801     clearError : function(){
22802         this.error = null;
22803     },
22804
22805     /**
22806      * Creates a copy of this record.
22807      * @param {String} id (optional) A new record id if you don't want to use this record's id
22808      * @return {Record}
22809      */
22810     copy : function(newId) {
22811         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22812     }
22813 };/*
22814  * Based on:
22815  * Ext JS Library 1.1.1
22816  * Copyright(c) 2006-2007, Ext JS, LLC.
22817  *
22818  * Originally Released Under LGPL - original licence link has changed is not relivant.
22819  *
22820  * Fork - LGPL
22821  * <script type="text/javascript">
22822  */
22823
22824
22825
22826 /**
22827  * @class Roo.data.Store
22828  * @extends Roo.util.Observable
22829  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22830  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22831  * <p>
22832  * 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
22833  * has no knowledge of the format of the data returned by the Proxy.<br>
22834  * <p>
22835  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22836  * instances from the data object. These records are cached and made available through accessor functions.
22837  * @constructor
22838  * Creates a new Store.
22839  * @param {Object} config A config object containing the objects needed for the Store to access data,
22840  * and read the data into Records.
22841  */
22842 Roo.data.Store = function(config){
22843     this.data = new Roo.util.MixedCollection(false);
22844     this.data.getKey = function(o){
22845         return o.id;
22846     };
22847     this.baseParams = {};
22848     // private
22849     this.paramNames = {
22850         "start" : "start",
22851         "limit" : "limit",
22852         "sort" : "sort",
22853         "dir" : "dir",
22854         "multisort" : "_multisort"
22855     };
22856
22857     if(config && config.data){
22858         this.inlineData = config.data;
22859         delete config.data;
22860     }
22861
22862     Roo.apply(this, config);
22863     
22864     if(this.reader){ // reader passed
22865         this.reader = Roo.factory(this.reader, Roo.data);
22866         this.reader.xmodule = this.xmodule || false;
22867         if(!this.recordType){
22868             this.recordType = this.reader.recordType;
22869         }
22870         if(this.reader.onMetaChange){
22871             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22872         }
22873     }
22874
22875     if(this.recordType){
22876         this.fields = this.recordType.prototype.fields;
22877     }
22878     this.modified = [];
22879
22880     this.addEvents({
22881         /**
22882          * @event datachanged
22883          * Fires when the data cache has changed, and a widget which is using this Store
22884          * as a Record cache should refresh its view.
22885          * @param {Store} this
22886          */
22887         datachanged : true,
22888         /**
22889          * @event metachange
22890          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22891          * @param {Store} this
22892          * @param {Object} meta The JSON metadata
22893          */
22894         metachange : true,
22895         /**
22896          * @event add
22897          * Fires when Records have been added to the Store
22898          * @param {Store} this
22899          * @param {Roo.data.Record[]} records The array of Records added
22900          * @param {Number} index The index at which the record(s) were added
22901          */
22902         add : true,
22903         /**
22904          * @event remove
22905          * Fires when a Record has been removed from the Store
22906          * @param {Store} this
22907          * @param {Roo.data.Record} record The Record that was removed
22908          * @param {Number} index The index at which the record was removed
22909          */
22910         remove : true,
22911         /**
22912          * @event update
22913          * Fires when a Record has been updated
22914          * @param {Store} this
22915          * @param {Roo.data.Record} record The Record that was updated
22916          * @param {String} operation The update operation being performed.  Value may be one of:
22917          * <pre><code>
22918  Roo.data.Record.EDIT
22919  Roo.data.Record.REJECT
22920  Roo.data.Record.COMMIT
22921          * </code></pre>
22922          */
22923         update : true,
22924         /**
22925          * @event clear
22926          * Fires when the data cache has been cleared.
22927          * @param {Store} this
22928          */
22929         clear : true,
22930         /**
22931          * @event beforeload
22932          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22933          * the load action will be canceled.
22934          * @param {Store} this
22935          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22936          */
22937         beforeload : true,
22938         /**
22939          * @event beforeloadadd
22940          * Fires after a new set of Records has been loaded.
22941          * @param {Store} this
22942          * @param {Roo.data.Record[]} records The Records that were loaded
22943          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22944          */
22945         beforeloadadd : true,
22946         /**
22947          * @event load
22948          * Fires after a new set of Records has been loaded, before they are added to the store.
22949          * @param {Store} this
22950          * @param {Roo.data.Record[]} records The Records that were loaded
22951          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22952          * @params {Object} return from reader
22953          */
22954         load : true,
22955         /**
22956          * @event loadexception
22957          * Fires if an exception occurs in the Proxy during loading.
22958          * Called with the signature of the Proxy's "loadexception" event.
22959          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22960          * 
22961          * @param {Proxy} 
22962          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22963          * @param {Object} load options 
22964          * @param {Object} jsonData from your request (normally this contains the Exception)
22965          */
22966         loadexception : true
22967     });
22968     
22969     if(this.proxy){
22970         this.proxy = Roo.factory(this.proxy, Roo.data);
22971         this.proxy.xmodule = this.xmodule || false;
22972         this.relayEvents(this.proxy,  ["loadexception"]);
22973     }
22974     this.sortToggle = {};
22975     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22976
22977     Roo.data.Store.superclass.constructor.call(this);
22978
22979     if(this.inlineData){
22980         this.loadData(this.inlineData);
22981         delete this.inlineData;
22982     }
22983 };
22984
22985 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22986      /**
22987     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22988     * without a remote query - used by combo/forms at present.
22989     */
22990     
22991     /**
22992     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22993     */
22994     /**
22995     * @cfg {Array} data Inline data to be loaded when the store is initialized.
22996     */
22997     /**
22998     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
22999     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23000     */
23001     /**
23002     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23003     * on any HTTP request
23004     */
23005     /**
23006     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23007     */
23008     /**
23009     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23010     */
23011     multiSort: false,
23012     /**
23013     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23014     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23015     */
23016     remoteSort : false,
23017
23018     /**
23019     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23020      * loaded or when a record is removed. (defaults to false).
23021     */
23022     pruneModifiedRecords : false,
23023
23024     // private
23025     lastOptions : null,
23026
23027     /**
23028      * Add Records to the Store and fires the add event.
23029      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23030      */
23031     add : function(records){
23032         records = [].concat(records);
23033         for(var i = 0, len = records.length; i < len; i++){
23034             records[i].join(this);
23035         }
23036         var index = this.data.length;
23037         this.data.addAll(records);
23038         this.fireEvent("add", this, records, index);
23039     },
23040
23041     /**
23042      * Remove a Record from the Store and fires the remove event.
23043      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23044      */
23045     remove : function(record){
23046         var index = this.data.indexOf(record);
23047         this.data.removeAt(index);
23048  
23049         if(this.pruneModifiedRecords){
23050             this.modified.remove(record);
23051         }
23052         this.fireEvent("remove", this, record, index);
23053     },
23054
23055     /**
23056      * Remove all Records from the Store and fires the clear event.
23057      */
23058     removeAll : function(){
23059         this.data.clear();
23060         if(this.pruneModifiedRecords){
23061             this.modified = [];
23062         }
23063         this.fireEvent("clear", this);
23064     },
23065
23066     /**
23067      * Inserts Records to the Store at the given index and fires the add event.
23068      * @param {Number} index The start index at which to insert the passed Records.
23069      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23070      */
23071     insert : function(index, records){
23072         records = [].concat(records);
23073         for(var i = 0, len = records.length; i < len; i++){
23074             this.data.insert(index, records[i]);
23075             records[i].join(this);
23076         }
23077         this.fireEvent("add", this, records, index);
23078     },
23079
23080     /**
23081      * Get the index within the cache of the passed Record.
23082      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23083      * @return {Number} The index of the passed Record. Returns -1 if not found.
23084      */
23085     indexOf : function(record){
23086         return this.data.indexOf(record);
23087     },
23088
23089     /**
23090      * Get the index within the cache of the Record with the passed id.
23091      * @param {String} id The id of the Record to find.
23092      * @return {Number} The index of the Record. Returns -1 if not found.
23093      */
23094     indexOfId : function(id){
23095         return this.data.indexOfKey(id);
23096     },
23097
23098     /**
23099      * Get the Record with the specified id.
23100      * @param {String} id The id of the Record to find.
23101      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23102      */
23103     getById : function(id){
23104         return this.data.key(id);
23105     },
23106
23107     /**
23108      * Get the Record at the specified index.
23109      * @param {Number} index The index of the Record to find.
23110      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23111      */
23112     getAt : function(index){
23113         return this.data.itemAt(index);
23114     },
23115
23116     /**
23117      * Returns a range of Records between specified indices.
23118      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23119      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23120      * @return {Roo.data.Record[]} An array of Records
23121      */
23122     getRange : function(start, end){
23123         return this.data.getRange(start, end);
23124     },
23125
23126     // private
23127     storeOptions : function(o){
23128         o = Roo.apply({}, o);
23129         delete o.callback;
23130         delete o.scope;
23131         this.lastOptions = o;
23132     },
23133
23134     /**
23135      * Loads the Record cache from the configured Proxy using the configured Reader.
23136      * <p>
23137      * If using remote paging, then the first load call must specify the <em>start</em>
23138      * and <em>limit</em> properties in the options.params property to establish the initial
23139      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23140      * <p>
23141      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23142      * and this call will return before the new data has been loaded. Perform any post-processing
23143      * in a callback function, or in a "load" event handler.</strong>
23144      * <p>
23145      * @param {Object} options An object containing properties which control loading options:<ul>
23146      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23147      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23148      * passed the following arguments:<ul>
23149      * <li>r : Roo.data.Record[]</li>
23150      * <li>options: Options object from the load call</li>
23151      * <li>success: Boolean success indicator</li></ul></li>
23152      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23153      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23154      * </ul>
23155      */
23156     load : function(options){
23157         options = options || {};
23158         if(this.fireEvent("beforeload", this, options) !== false){
23159             this.storeOptions(options);
23160             var p = Roo.apply(options.params || {}, this.baseParams);
23161             // if meta was not loaded from remote source.. try requesting it.
23162             if (!this.reader.metaFromRemote) {
23163                 p._requestMeta = 1;
23164             }
23165             if(this.sortInfo && this.remoteSort){
23166                 var pn = this.paramNames;
23167                 p[pn["sort"]] = this.sortInfo.field;
23168                 p[pn["dir"]] = this.sortInfo.direction;
23169             }
23170             if (this.multiSort) {
23171                 var pn = this.paramNames;
23172                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23173             }
23174             
23175             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23176         }
23177     },
23178
23179     /**
23180      * Reloads the Record cache from the configured Proxy using the configured Reader and
23181      * the options from the last load operation performed.
23182      * @param {Object} options (optional) An object containing properties which may override the options
23183      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23184      * the most recently used options are reused).
23185      */
23186     reload : function(options){
23187         this.load(Roo.applyIf(options||{}, this.lastOptions));
23188     },
23189
23190     // private
23191     // Called as a callback by the Reader during a load operation.
23192     loadRecords : function(o, options, success){
23193         if(!o || success === false){
23194             if(success !== false){
23195                 this.fireEvent("load", this, [], options, o);
23196             }
23197             if(options.callback){
23198                 options.callback.call(options.scope || this, [], options, false);
23199             }
23200             return;
23201         }
23202         // if data returned failure - throw an exception.
23203         if (o.success === false) {
23204             // show a message if no listener is registered.
23205             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23206                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23207             }
23208             // loadmask wil be hooked into this..
23209             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23210             return;
23211         }
23212         var r = o.records, t = o.totalRecords || r.length;
23213         
23214         this.fireEvent("beforeloadadd", this, r, options, o);
23215         
23216         if(!options || options.add !== true){
23217             if(this.pruneModifiedRecords){
23218                 this.modified = [];
23219             }
23220             for(var i = 0, len = r.length; i < len; i++){
23221                 r[i].join(this);
23222             }
23223             if(this.snapshot){
23224                 this.data = this.snapshot;
23225                 delete this.snapshot;
23226             }
23227             this.data.clear();
23228             this.data.addAll(r);
23229             this.totalLength = t;
23230             this.applySort();
23231             this.fireEvent("datachanged", this);
23232         }else{
23233             this.totalLength = Math.max(t, this.data.length+r.length);
23234             this.add(r);
23235         }
23236         
23237         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23238                 
23239             var e = new Roo.data.Record({});
23240
23241             e.set(this.parent.displayField, this.parent.emptyTitle);
23242             e.set(this.parent.valueField, '');
23243
23244             this.insert(0, e);
23245         }
23246             
23247         this.fireEvent("load", this, r, options, o);
23248         if(options.callback){
23249             options.callback.call(options.scope || this, r, options, true);
23250         }
23251     },
23252
23253
23254     /**
23255      * Loads data from a passed data block. A Reader which understands the format of the data
23256      * must have been configured in the constructor.
23257      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23258      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23259      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23260      */
23261     loadData : function(o, append){
23262         var r = this.reader.readRecords(o);
23263         this.loadRecords(r, {add: append}, true);
23264     },
23265
23266     /**
23267      * Gets the number of cached records.
23268      * <p>
23269      * <em>If using paging, this may not be the total size of the dataset. If the data object
23270      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23271      * the data set size</em>
23272      */
23273     getCount : function(){
23274         return this.data.length || 0;
23275     },
23276
23277     /**
23278      * Gets the total number of records in the dataset as returned by the server.
23279      * <p>
23280      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23281      * the dataset size</em>
23282      */
23283     getTotalCount : function(){
23284         return this.totalLength || 0;
23285     },
23286
23287     /**
23288      * Returns the sort state of the Store as an object with two properties:
23289      * <pre><code>
23290  field {String} The name of the field by which the Records are sorted
23291  direction {String} The sort order, "ASC" or "DESC"
23292      * </code></pre>
23293      */
23294     getSortState : function(){
23295         return this.sortInfo;
23296     },
23297
23298     // private
23299     applySort : function(){
23300         if(this.sortInfo && !this.remoteSort){
23301             var s = this.sortInfo, f = s.field;
23302             var st = this.fields.get(f).sortType;
23303             var fn = function(r1, r2){
23304                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23305                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23306             };
23307             this.data.sort(s.direction, fn);
23308             if(this.snapshot && this.snapshot != this.data){
23309                 this.snapshot.sort(s.direction, fn);
23310             }
23311         }
23312     },
23313
23314     /**
23315      * Sets the default sort column and order to be used by the next load operation.
23316      * @param {String} fieldName The name of the field to sort by.
23317      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23318      */
23319     setDefaultSort : function(field, dir){
23320         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23321     },
23322
23323     /**
23324      * Sort the Records.
23325      * If remote sorting is used, the sort is performed on the server, and the cache is
23326      * reloaded. If local sorting is used, the cache is sorted internally.
23327      * @param {String} fieldName The name of the field to sort by.
23328      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23329      */
23330     sort : function(fieldName, dir){
23331         var f = this.fields.get(fieldName);
23332         if(!dir){
23333             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23334             
23335             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23336                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23337             }else{
23338                 dir = f.sortDir;
23339             }
23340         }
23341         this.sortToggle[f.name] = dir;
23342         this.sortInfo = {field: f.name, direction: dir};
23343         if(!this.remoteSort){
23344             this.applySort();
23345             this.fireEvent("datachanged", this);
23346         }else{
23347             this.load(this.lastOptions);
23348         }
23349     },
23350
23351     /**
23352      * Calls the specified function for each of the Records in the cache.
23353      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23354      * Returning <em>false</em> aborts and exits the iteration.
23355      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23356      */
23357     each : function(fn, scope){
23358         this.data.each(fn, scope);
23359     },
23360
23361     /**
23362      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23363      * (e.g., during paging).
23364      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23365      */
23366     getModifiedRecords : function(){
23367         return this.modified;
23368     },
23369
23370     // private
23371     createFilterFn : function(property, value, anyMatch){
23372         if(!value.exec){ // not a regex
23373             value = String(value);
23374             if(value.length == 0){
23375                 return false;
23376             }
23377             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23378         }
23379         return function(r){
23380             return value.test(r.data[property]);
23381         };
23382     },
23383
23384     /**
23385      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23386      * @param {String} property A field on your records
23387      * @param {Number} start The record index to start at (defaults to 0)
23388      * @param {Number} end The last record index to include (defaults to length - 1)
23389      * @return {Number} The sum
23390      */
23391     sum : function(property, start, end){
23392         var rs = this.data.items, v = 0;
23393         start = start || 0;
23394         end = (end || end === 0) ? end : rs.length-1;
23395
23396         for(var i = start; i <= end; i++){
23397             v += (rs[i].data[property] || 0);
23398         }
23399         return v;
23400     },
23401
23402     /**
23403      * Filter the records by a specified property.
23404      * @param {String} field A field on your records
23405      * @param {String/RegExp} value Either a string that the field
23406      * should start with or a RegExp to test against the field
23407      * @param {Boolean} anyMatch True to match any part not just the beginning
23408      */
23409     filter : function(property, value, anyMatch){
23410         var fn = this.createFilterFn(property, value, anyMatch);
23411         return fn ? this.filterBy(fn) : this.clearFilter();
23412     },
23413
23414     /**
23415      * Filter by a function. The specified function will be called with each
23416      * record in this data source. If the function returns true the record is included,
23417      * otherwise it is filtered.
23418      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23419      * @param {Object} scope (optional) The scope of the function (defaults to this)
23420      */
23421     filterBy : function(fn, scope){
23422         this.snapshot = this.snapshot || this.data;
23423         this.data = this.queryBy(fn, scope||this);
23424         this.fireEvent("datachanged", this);
23425     },
23426
23427     /**
23428      * Query the records by a specified property.
23429      * @param {String} field A field on your records
23430      * @param {String/RegExp} value Either a string that the field
23431      * should start with or a RegExp to test against the field
23432      * @param {Boolean} anyMatch True to match any part not just the beginning
23433      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23434      */
23435     query : function(property, value, anyMatch){
23436         var fn = this.createFilterFn(property, value, anyMatch);
23437         return fn ? this.queryBy(fn) : this.data.clone();
23438     },
23439
23440     /**
23441      * Query by a function. The specified function will be called with each
23442      * record in this data source. If the function returns true the record is included
23443      * in the results.
23444      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23445      * @param {Object} scope (optional) The scope of the function (defaults to this)
23446       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23447      **/
23448     queryBy : function(fn, scope){
23449         var data = this.snapshot || this.data;
23450         return data.filterBy(fn, scope||this);
23451     },
23452
23453     /**
23454      * Collects unique values for a particular dataIndex from this store.
23455      * @param {String} dataIndex The property to collect
23456      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23457      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23458      * @return {Array} An array of the unique values
23459      **/
23460     collect : function(dataIndex, allowNull, bypassFilter){
23461         var d = (bypassFilter === true && this.snapshot) ?
23462                 this.snapshot.items : this.data.items;
23463         var v, sv, r = [], l = {};
23464         for(var i = 0, len = d.length; i < len; i++){
23465             v = d[i].data[dataIndex];
23466             sv = String(v);
23467             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23468                 l[sv] = true;
23469                 r[r.length] = v;
23470             }
23471         }
23472         return r;
23473     },
23474
23475     /**
23476      * Revert to a view of the Record cache with no filtering applied.
23477      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23478      */
23479     clearFilter : function(suppressEvent){
23480         if(this.snapshot && this.snapshot != this.data){
23481             this.data = this.snapshot;
23482             delete this.snapshot;
23483             if(suppressEvent !== true){
23484                 this.fireEvent("datachanged", this);
23485             }
23486         }
23487     },
23488
23489     // private
23490     afterEdit : function(record){
23491         if(this.modified.indexOf(record) == -1){
23492             this.modified.push(record);
23493         }
23494         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23495     },
23496     
23497     // private
23498     afterReject : function(record){
23499         this.modified.remove(record);
23500         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23501     },
23502
23503     // private
23504     afterCommit : function(record){
23505         this.modified.remove(record);
23506         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23507     },
23508
23509     /**
23510      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23511      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23512      */
23513     commitChanges : function(){
23514         var m = this.modified.slice(0);
23515         this.modified = [];
23516         for(var i = 0, len = m.length; i < len; i++){
23517             m[i].commit();
23518         }
23519     },
23520
23521     /**
23522      * Cancel outstanding changes on all changed records.
23523      */
23524     rejectChanges : function(){
23525         var m = this.modified.slice(0);
23526         this.modified = [];
23527         for(var i = 0, len = m.length; i < len; i++){
23528             m[i].reject();
23529         }
23530     },
23531
23532     onMetaChange : function(meta, rtype, o){
23533         this.recordType = rtype;
23534         this.fields = rtype.prototype.fields;
23535         delete this.snapshot;
23536         this.sortInfo = meta.sortInfo || this.sortInfo;
23537         this.modified = [];
23538         this.fireEvent('metachange', this, this.reader.meta);
23539     },
23540     
23541     moveIndex : function(data, type)
23542     {
23543         var index = this.indexOf(data);
23544         
23545         var newIndex = index + type;
23546         
23547         this.remove(data);
23548         
23549         this.insert(newIndex, data);
23550         
23551     }
23552 });/*
23553  * Based on:
23554  * Ext JS Library 1.1.1
23555  * Copyright(c) 2006-2007, Ext JS, LLC.
23556  *
23557  * Originally Released Under LGPL - original licence link has changed is not relivant.
23558  *
23559  * Fork - LGPL
23560  * <script type="text/javascript">
23561  */
23562
23563 /**
23564  * @class Roo.data.SimpleStore
23565  * @extends Roo.data.Store
23566  * Small helper class to make creating Stores from Array data easier.
23567  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23568  * @cfg {Array} fields An array of field definition objects, or field name strings.
23569  * @cfg {Object} an existing reader (eg. copied from another store)
23570  * @cfg {Array} data The multi-dimensional array of data
23571  * @constructor
23572  * @param {Object} config
23573  */
23574 Roo.data.SimpleStore = function(config)
23575 {
23576     Roo.data.SimpleStore.superclass.constructor.call(this, {
23577         isLocal : true,
23578         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
23579                 id: config.id
23580             },
23581             Roo.data.Record.create(config.fields)
23582         ),
23583         proxy : new Roo.data.MemoryProxy(config.data)
23584     });
23585     this.load();
23586 };
23587 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23588  * Based on:
23589  * Ext JS Library 1.1.1
23590  * Copyright(c) 2006-2007, Ext JS, LLC.
23591  *
23592  * Originally Released Under LGPL - original licence link has changed is not relivant.
23593  *
23594  * Fork - LGPL
23595  * <script type="text/javascript">
23596  */
23597
23598 /**
23599 /**
23600  * @extends Roo.data.Store
23601  * @class Roo.data.JsonStore
23602  * Small helper class to make creating Stores for JSON data easier. <br/>
23603 <pre><code>
23604 var store = new Roo.data.JsonStore({
23605     url: 'get-images.php',
23606     root: 'images',
23607     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23608 });
23609 </code></pre>
23610  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23611  * JsonReader and HttpProxy (unless inline data is provided).</b>
23612  * @cfg {Array} fields An array of field definition objects, or field name strings.
23613  * @constructor
23614  * @param {Object} config
23615  */
23616 Roo.data.JsonStore = function(c){
23617     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23618         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23619         reader: new Roo.data.JsonReader(c, c.fields)
23620     }));
23621 };
23622 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23623  * Based on:
23624  * Ext JS Library 1.1.1
23625  * Copyright(c) 2006-2007, Ext JS, LLC.
23626  *
23627  * Originally Released Under LGPL - original licence link has changed is not relivant.
23628  *
23629  * Fork - LGPL
23630  * <script type="text/javascript">
23631  */
23632
23633  
23634 Roo.data.Field = function(config){
23635     if(typeof config == "string"){
23636         config = {name: config};
23637     }
23638     Roo.apply(this, config);
23639     
23640     if(!this.type){
23641         this.type = "auto";
23642     }
23643     
23644     var st = Roo.data.SortTypes;
23645     // named sortTypes are supported, here we look them up
23646     if(typeof this.sortType == "string"){
23647         this.sortType = st[this.sortType];
23648     }
23649     
23650     // set default sortType for strings and dates
23651     if(!this.sortType){
23652         switch(this.type){
23653             case "string":
23654                 this.sortType = st.asUCString;
23655                 break;
23656             case "date":
23657                 this.sortType = st.asDate;
23658                 break;
23659             default:
23660                 this.sortType = st.none;
23661         }
23662     }
23663
23664     // define once
23665     var stripRe = /[\$,%]/g;
23666
23667     // prebuilt conversion function for this field, instead of
23668     // switching every time we're reading a value
23669     if(!this.convert){
23670         var cv, dateFormat = this.dateFormat;
23671         switch(this.type){
23672             case "":
23673             case "auto":
23674             case undefined:
23675                 cv = function(v){ return v; };
23676                 break;
23677             case "string":
23678                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23679                 break;
23680             case "int":
23681                 cv = function(v){
23682                     return v !== undefined && v !== null && v !== '' ?
23683                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23684                     };
23685                 break;
23686             case "float":
23687                 cv = function(v){
23688                     return v !== undefined && v !== null && v !== '' ?
23689                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23690                     };
23691                 break;
23692             case "bool":
23693             case "boolean":
23694                 cv = function(v){ return v === true || v === "true" || v == 1; };
23695                 break;
23696             case "date":
23697                 cv = function(v){
23698                     if(!v){
23699                         return '';
23700                     }
23701                     if(v instanceof Date){
23702                         return v;
23703                     }
23704                     if(dateFormat){
23705                         if(dateFormat == "timestamp"){
23706                             return new Date(v*1000);
23707                         }
23708                         return Date.parseDate(v, dateFormat);
23709                     }
23710                     var parsed = Date.parse(v);
23711                     return parsed ? new Date(parsed) : null;
23712                 };
23713              break;
23714             
23715         }
23716         this.convert = cv;
23717     }
23718 };
23719
23720 Roo.data.Field.prototype = {
23721     dateFormat: null,
23722     defaultValue: "",
23723     mapping: null,
23724     sortType : null,
23725     sortDir : "ASC"
23726 };/*
23727  * Based on:
23728  * Ext JS Library 1.1.1
23729  * Copyright(c) 2006-2007, Ext JS, LLC.
23730  *
23731  * Originally Released Under LGPL - original licence link has changed is not relivant.
23732  *
23733  * Fork - LGPL
23734  * <script type="text/javascript">
23735  */
23736  
23737 // Base class for reading structured data from a data source.  This class is intended to be
23738 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23739
23740 /**
23741  * @class Roo.data.DataReader
23742  * Base class for reading structured data from a data source.  This class is intended to be
23743  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23744  */
23745
23746 Roo.data.DataReader = function(meta, recordType){
23747     
23748     this.meta = meta;
23749     
23750     this.recordType = recordType instanceof Array ? 
23751         Roo.data.Record.create(recordType) : recordType;
23752 };
23753
23754 Roo.data.DataReader.prototype = {
23755      /**
23756      * Create an empty record
23757      * @param {Object} data (optional) - overlay some values
23758      * @return {Roo.data.Record} record created.
23759      */
23760     newRow :  function(d) {
23761         var da =  {};
23762         this.recordType.prototype.fields.each(function(c) {
23763             switch( c.type) {
23764                 case 'int' : da[c.name] = 0; break;
23765                 case 'date' : da[c.name] = new Date(); break;
23766                 case 'float' : da[c.name] = 0.0; break;
23767                 case 'boolean' : da[c.name] = false; break;
23768                 default : da[c.name] = ""; break;
23769             }
23770             
23771         });
23772         return new this.recordType(Roo.apply(da, d));
23773     }
23774     
23775 };/*
23776  * Based on:
23777  * Ext JS Library 1.1.1
23778  * Copyright(c) 2006-2007, Ext JS, LLC.
23779  *
23780  * Originally Released Under LGPL - original licence link has changed is not relivant.
23781  *
23782  * Fork - LGPL
23783  * <script type="text/javascript">
23784  */
23785
23786 /**
23787  * @class Roo.data.DataProxy
23788  * @extends Roo.data.Observable
23789  * This class is an abstract base class for implementations which provide retrieval of
23790  * unformatted data objects.<br>
23791  * <p>
23792  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23793  * (of the appropriate type which knows how to parse the data object) to provide a block of
23794  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23795  * <p>
23796  * Custom implementations must implement the load method as described in
23797  * {@link Roo.data.HttpProxy#load}.
23798  */
23799 Roo.data.DataProxy = function(){
23800     this.addEvents({
23801         /**
23802          * @event beforeload
23803          * Fires before a network request is made to retrieve a data object.
23804          * @param {Object} This DataProxy object.
23805          * @param {Object} params The params parameter to the load function.
23806          */
23807         beforeload : true,
23808         /**
23809          * @event load
23810          * Fires before the load method's callback is called.
23811          * @param {Object} This DataProxy object.
23812          * @param {Object} o The data object.
23813          * @param {Object} arg The callback argument object passed to the load function.
23814          */
23815         load : true,
23816         /**
23817          * @event loadexception
23818          * Fires if an Exception occurs during data retrieval.
23819          * @param {Object} This DataProxy object.
23820          * @param {Object} o The data object.
23821          * @param {Object} arg The callback argument object passed to the load function.
23822          * @param {Object} e The Exception.
23823          */
23824         loadexception : true
23825     });
23826     Roo.data.DataProxy.superclass.constructor.call(this);
23827 };
23828
23829 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23830
23831     /**
23832      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23833      */
23834 /*
23835  * Based on:
23836  * Ext JS Library 1.1.1
23837  * Copyright(c) 2006-2007, Ext JS, LLC.
23838  *
23839  * Originally Released Under LGPL - original licence link has changed is not relivant.
23840  *
23841  * Fork - LGPL
23842  * <script type="text/javascript">
23843  */
23844 /**
23845  * @class Roo.data.MemoryProxy
23846  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23847  * to the Reader when its load method is called.
23848  * @constructor
23849  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23850  */
23851 Roo.data.MemoryProxy = function(data){
23852     if (data.data) {
23853         data = data.data;
23854     }
23855     Roo.data.MemoryProxy.superclass.constructor.call(this);
23856     this.data = data;
23857 };
23858
23859 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23860     
23861     /**
23862      * Load data from the requested source (in this case an in-memory
23863      * data object passed to the constructor), read the data object into
23864      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23865      * process that block using the passed callback.
23866      * @param {Object} params This parameter is not used by the MemoryProxy class.
23867      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23868      * object into a block of Roo.data.Records.
23869      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23870      * The function must be passed <ul>
23871      * <li>The Record block object</li>
23872      * <li>The "arg" argument from the load function</li>
23873      * <li>A boolean success indicator</li>
23874      * </ul>
23875      * @param {Object} scope The scope in which to call the callback
23876      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23877      */
23878     load : function(params, reader, callback, scope, arg){
23879         params = params || {};
23880         var result;
23881         try {
23882             result = reader.readRecords(params.data ? params.data :this.data);
23883         }catch(e){
23884             this.fireEvent("loadexception", this, arg, null, e);
23885             callback.call(scope, null, arg, false);
23886             return;
23887         }
23888         callback.call(scope, result, arg, true);
23889     },
23890     
23891     // private
23892     update : function(params, records){
23893         
23894     }
23895 });/*
23896  * Based on:
23897  * Ext JS Library 1.1.1
23898  * Copyright(c) 2006-2007, Ext JS, LLC.
23899  *
23900  * Originally Released Under LGPL - original licence link has changed is not relivant.
23901  *
23902  * Fork - LGPL
23903  * <script type="text/javascript">
23904  */
23905 /**
23906  * @class Roo.data.HttpProxy
23907  * @extends Roo.data.DataProxy
23908  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23909  * configured to reference a certain URL.<br><br>
23910  * <p>
23911  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23912  * from which the running page was served.<br><br>
23913  * <p>
23914  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23915  * <p>
23916  * Be aware that to enable the browser to parse an XML document, the server must set
23917  * the Content-Type header in the HTTP response to "text/xml".
23918  * @constructor
23919  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23920  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23921  * will be used to make the request.
23922  */
23923 Roo.data.HttpProxy = function(conn){
23924     Roo.data.HttpProxy.superclass.constructor.call(this);
23925     // is conn a conn config or a real conn?
23926     this.conn = conn;
23927     this.useAjax = !conn || !conn.events;
23928   
23929 };
23930
23931 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23932     // thse are take from connection...
23933     
23934     /**
23935      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23936      */
23937     /**
23938      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23939      * extra parameters to each request made by this object. (defaults to undefined)
23940      */
23941     /**
23942      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23943      *  to each request made by this object. (defaults to undefined)
23944      */
23945     /**
23946      * @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)
23947      */
23948     /**
23949      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23950      */
23951      /**
23952      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23953      * @type Boolean
23954      */
23955   
23956
23957     /**
23958      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23959      * @type Boolean
23960      */
23961     /**
23962      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23963      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23964      * a finer-grained basis than the DataProxy events.
23965      */
23966     getConnection : function(){
23967         return this.useAjax ? Roo.Ajax : this.conn;
23968     },
23969
23970     /**
23971      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23972      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23973      * process that block using the passed callback.
23974      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23975      * for the request to the remote server.
23976      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23977      * object into a block of Roo.data.Records.
23978      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23979      * The function must be passed <ul>
23980      * <li>The Record block object</li>
23981      * <li>The "arg" argument from the load function</li>
23982      * <li>A boolean success indicator</li>
23983      * </ul>
23984      * @param {Object} scope The scope in which to call the callback
23985      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23986      */
23987     load : function(params, reader, callback, scope, arg){
23988         if(this.fireEvent("beforeload", this, params) !== false){
23989             var  o = {
23990                 params : params || {},
23991                 request: {
23992                     callback : callback,
23993                     scope : scope,
23994                     arg : arg
23995                 },
23996                 reader: reader,
23997                 callback : this.loadResponse,
23998                 scope: this
23999             };
24000             if(this.useAjax){
24001                 Roo.applyIf(o, this.conn);
24002                 if(this.activeRequest){
24003                     Roo.Ajax.abort(this.activeRequest);
24004                 }
24005                 this.activeRequest = Roo.Ajax.request(o);
24006             }else{
24007                 this.conn.request(o);
24008             }
24009         }else{
24010             callback.call(scope||this, null, arg, false);
24011         }
24012     },
24013
24014     // private
24015     loadResponse : function(o, success, response){
24016         delete this.activeRequest;
24017         if(!success){
24018             this.fireEvent("loadexception", this, o, response);
24019             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24020             return;
24021         }
24022         var result;
24023         try {
24024             result = o.reader.read(response);
24025         }catch(e){
24026             this.fireEvent("loadexception", this, o, response, e);
24027             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24028             return;
24029         }
24030         
24031         this.fireEvent("load", this, o, o.request.arg);
24032         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24033     },
24034
24035     // private
24036     update : function(dataSet){
24037
24038     },
24039
24040     // private
24041     updateResponse : function(dataSet){
24042
24043     }
24044 });/*
24045  * Based on:
24046  * Ext JS Library 1.1.1
24047  * Copyright(c) 2006-2007, Ext JS, LLC.
24048  *
24049  * Originally Released Under LGPL - original licence link has changed is not relivant.
24050  *
24051  * Fork - LGPL
24052  * <script type="text/javascript">
24053  */
24054
24055 /**
24056  * @class Roo.data.ScriptTagProxy
24057  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24058  * other than the originating domain of the running page.<br><br>
24059  * <p>
24060  * <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
24061  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24062  * <p>
24063  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24064  * source code that is used as the source inside a &lt;script> tag.<br><br>
24065  * <p>
24066  * In order for the browser to process the returned data, the server must wrap the data object
24067  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24068  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24069  * depending on whether the callback name was passed:
24070  * <p>
24071  * <pre><code>
24072 boolean scriptTag = false;
24073 String cb = request.getParameter("callback");
24074 if (cb != null) {
24075     scriptTag = true;
24076     response.setContentType("text/javascript");
24077 } else {
24078     response.setContentType("application/x-json");
24079 }
24080 Writer out = response.getWriter();
24081 if (scriptTag) {
24082     out.write(cb + "(");
24083 }
24084 out.print(dataBlock.toJsonString());
24085 if (scriptTag) {
24086     out.write(");");
24087 }
24088 </pre></code>
24089  *
24090  * @constructor
24091  * @param {Object} config A configuration object.
24092  */
24093 Roo.data.ScriptTagProxy = function(config){
24094     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24095     Roo.apply(this, config);
24096     this.head = document.getElementsByTagName("head")[0];
24097 };
24098
24099 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24100
24101 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24102     /**
24103      * @cfg {String} url The URL from which to request the data object.
24104      */
24105     /**
24106      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24107      */
24108     timeout : 30000,
24109     /**
24110      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24111      * the server the name of the callback function set up by the load call to process the returned data object.
24112      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24113      * javascript output which calls this named function passing the data object as its only parameter.
24114      */
24115     callbackParam : "callback",
24116     /**
24117      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24118      * name to the request.
24119      */
24120     nocache : true,
24121
24122     /**
24123      * Load data from the configured URL, read the data object into
24124      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24125      * process that block using the passed callback.
24126      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24127      * for the request to the remote server.
24128      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24129      * object into a block of Roo.data.Records.
24130      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24131      * The function must be passed <ul>
24132      * <li>The Record block object</li>
24133      * <li>The "arg" argument from the load function</li>
24134      * <li>A boolean success indicator</li>
24135      * </ul>
24136      * @param {Object} scope The scope in which to call the callback
24137      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24138      */
24139     load : function(params, reader, callback, scope, arg){
24140         if(this.fireEvent("beforeload", this, params) !== false){
24141
24142             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24143
24144             var url = this.url;
24145             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24146             if(this.nocache){
24147                 url += "&_dc=" + (new Date().getTime());
24148             }
24149             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24150             var trans = {
24151                 id : transId,
24152                 cb : "stcCallback"+transId,
24153                 scriptId : "stcScript"+transId,
24154                 params : params,
24155                 arg : arg,
24156                 url : url,
24157                 callback : callback,
24158                 scope : scope,
24159                 reader : reader
24160             };
24161             var conn = this;
24162
24163             window[trans.cb] = function(o){
24164                 conn.handleResponse(o, trans);
24165             };
24166
24167             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24168
24169             if(this.autoAbort !== false){
24170                 this.abort();
24171             }
24172
24173             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24174
24175             var script = document.createElement("script");
24176             script.setAttribute("src", url);
24177             script.setAttribute("type", "text/javascript");
24178             script.setAttribute("id", trans.scriptId);
24179             this.head.appendChild(script);
24180
24181             this.trans = trans;
24182         }else{
24183             callback.call(scope||this, null, arg, false);
24184         }
24185     },
24186
24187     // private
24188     isLoading : function(){
24189         return this.trans ? true : false;
24190     },
24191
24192     /**
24193      * Abort the current server request.
24194      */
24195     abort : function(){
24196         if(this.isLoading()){
24197             this.destroyTrans(this.trans);
24198         }
24199     },
24200
24201     // private
24202     destroyTrans : function(trans, isLoaded){
24203         this.head.removeChild(document.getElementById(trans.scriptId));
24204         clearTimeout(trans.timeoutId);
24205         if(isLoaded){
24206             window[trans.cb] = undefined;
24207             try{
24208                 delete window[trans.cb];
24209             }catch(e){}
24210         }else{
24211             // if hasn't been loaded, wait for load to remove it to prevent script error
24212             window[trans.cb] = function(){
24213                 window[trans.cb] = undefined;
24214                 try{
24215                     delete window[trans.cb];
24216                 }catch(e){}
24217             };
24218         }
24219     },
24220
24221     // private
24222     handleResponse : function(o, trans){
24223         this.trans = false;
24224         this.destroyTrans(trans, true);
24225         var result;
24226         try {
24227             result = trans.reader.readRecords(o);
24228         }catch(e){
24229             this.fireEvent("loadexception", this, o, trans.arg, e);
24230             trans.callback.call(trans.scope||window, null, trans.arg, false);
24231             return;
24232         }
24233         this.fireEvent("load", this, o, trans.arg);
24234         trans.callback.call(trans.scope||window, result, trans.arg, true);
24235     },
24236
24237     // private
24238     handleFailure : function(trans){
24239         this.trans = false;
24240         this.destroyTrans(trans, false);
24241         this.fireEvent("loadexception", this, null, trans.arg);
24242         trans.callback.call(trans.scope||window, null, trans.arg, false);
24243     }
24244 });/*
24245  * Based on:
24246  * Ext JS Library 1.1.1
24247  * Copyright(c) 2006-2007, Ext JS, LLC.
24248  *
24249  * Originally Released Under LGPL - original licence link has changed is not relivant.
24250  *
24251  * Fork - LGPL
24252  * <script type="text/javascript">
24253  */
24254
24255 /**
24256  * @class Roo.data.JsonReader
24257  * @extends Roo.data.DataReader
24258  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24259  * based on mappings in a provided Roo.data.Record constructor.
24260  * 
24261  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24262  * in the reply previously. 
24263  * 
24264  * <p>
24265  * Example code:
24266  * <pre><code>
24267 var RecordDef = Roo.data.Record.create([
24268     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24269     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24270 ]);
24271 var myReader = new Roo.data.JsonReader({
24272     totalProperty: "results",    // The property which contains the total dataset size (optional)
24273     root: "rows",                // The property which contains an Array of row objects
24274     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24275 }, RecordDef);
24276 </code></pre>
24277  * <p>
24278  * This would consume a JSON file like this:
24279  * <pre><code>
24280 { 'results': 2, 'rows': [
24281     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24282     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24283 }
24284 </code></pre>
24285  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24286  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24287  * paged from the remote server.
24288  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24289  * @cfg {String} root name of the property which contains the Array of row objects.
24290  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24291  * @cfg {Array} fields Array of field definition objects
24292  * @constructor
24293  * Create a new JsonReader
24294  * @param {Object} meta Metadata configuration options
24295  * @param {Object} recordType Either an Array of field definition objects,
24296  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24297  */
24298 Roo.data.JsonReader = function(meta, recordType){
24299     
24300     meta = meta || {};
24301     // set some defaults:
24302     Roo.applyIf(meta, {
24303         totalProperty: 'total',
24304         successProperty : 'success',
24305         root : 'data',
24306         id : 'id'
24307     });
24308     
24309     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24310 };
24311 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24312     
24313     /**
24314      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24315      * Used by Store query builder to append _requestMeta to params.
24316      * 
24317      */
24318     metaFromRemote : false,
24319     /**
24320      * This method is only used by a DataProxy which has retrieved data from a remote server.
24321      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24322      * @return {Object} data A data block which is used by an Roo.data.Store object as
24323      * a cache of Roo.data.Records.
24324      */
24325     read : function(response){
24326         var json = response.responseText;
24327        
24328         var o = /* eval:var:o */ eval("("+json+")");
24329         if(!o) {
24330             throw {message: "JsonReader.read: Json object not found"};
24331         }
24332         
24333         if(o.metaData){
24334             
24335             delete this.ef;
24336             this.metaFromRemote = true;
24337             this.meta = o.metaData;
24338             this.recordType = Roo.data.Record.create(o.metaData.fields);
24339             this.onMetaChange(this.meta, this.recordType, o);
24340         }
24341         return this.readRecords(o);
24342     },
24343
24344     // private function a store will implement
24345     onMetaChange : function(meta, recordType, o){
24346
24347     },
24348
24349     /**
24350          * @ignore
24351          */
24352     simpleAccess: function(obj, subsc) {
24353         return obj[subsc];
24354     },
24355
24356         /**
24357          * @ignore
24358          */
24359     getJsonAccessor: function(){
24360         var re = /[\[\.]/;
24361         return function(expr) {
24362             try {
24363                 return(re.test(expr))
24364                     ? new Function("obj", "return obj." + expr)
24365                     : function(obj){
24366                         return obj[expr];
24367                     };
24368             } catch(e){}
24369             return Roo.emptyFn;
24370         };
24371     }(),
24372
24373     /**
24374      * Create a data block containing Roo.data.Records from an XML document.
24375      * @param {Object} o An object which contains an Array of row objects in the property specified
24376      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24377      * which contains the total size of the dataset.
24378      * @return {Object} data A data block which is used by an Roo.data.Store object as
24379      * a cache of Roo.data.Records.
24380      */
24381     readRecords : function(o){
24382         /**
24383          * After any data loads, the raw JSON data is available for further custom processing.
24384          * @type Object
24385          */
24386         this.o = o;
24387         var s = this.meta, Record = this.recordType,
24388             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24389
24390 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24391         if (!this.ef) {
24392             if(s.totalProperty) {
24393                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24394                 }
24395                 if(s.successProperty) {
24396                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24397                 }
24398                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24399                 if (s.id) {
24400                         var g = this.getJsonAccessor(s.id);
24401                         this.getId = function(rec) {
24402                                 var r = g(rec);  
24403                                 return (r === undefined || r === "") ? null : r;
24404                         };
24405                 } else {
24406                         this.getId = function(){return null;};
24407                 }
24408             this.ef = [];
24409             for(var jj = 0; jj < fl; jj++){
24410                 f = fi[jj];
24411                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24412                 this.ef[jj] = this.getJsonAccessor(map);
24413             }
24414         }
24415
24416         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24417         if(s.totalProperty){
24418             var vt = parseInt(this.getTotal(o), 10);
24419             if(!isNaN(vt)){
24420                 totalRecords = vt;
24421             }
24422         }
24423         if(s.successProperty){
24424             var vs = this.getSuccess(o);
24425             if(vs === false || vs === 'false'){
24426                 success = false;
24427             }
24428         }
24429         var records = [];
24430         for(var i = 0; i < c; i++){
24431                 var n = root[i];
24432             var values = {};
24433             var id = this.getId(n);
24434             for(var j = 0; j < fl; j++){
24435                 f = fi[j];
24436             var v = this.ef[j](n);
24437             if (!f.convert) {
24438                 Roo.log('missing convert for ' + f.name);
24439                 Roo.log(f);
24440                 continue;
24441             }
24442             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24443             }
24444             var record = new Record(values, id);
24445             record.json = n;
24446             records[i] = record;
24447         }
24448         return {
24449             raw : o,
24450             success : success,
24451             records : records,
24452             totalRecords : totalRecords
24453         };
24454     }
24455 });/*
24456  * Based on:
24457  * Ext JS Library 1.1.1
24458  * Copyright(c) 2006-2007, Ext JS, LLC.
24459  *
24460  * Originally Released Under LGPL - original licence link has changed is not relivant.
24461  *
24462  * Fork - LGPL
24463  * <script type="text/javascript">
24464  */
24465
24466 /**
24467  * @class Roo.data.XmlReader
24468  * @extends Roo.data.DataReader
24469  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24470  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24471  * <p>
24472  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24473  * header in the HTTP response must be set to "text/xml".</em>
24474  * <p>
24475  * Example code:
24476  * <pre><code>
24477 var RecordDef = Roo.data.Record.create([
24478    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24479    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24480 ]);
24481 var myReader = new Roo.data.XmlReader({
24482    totalRecords: "results", // The element which contains the total dataset size (optional)
24483    record: "row",           // The repeated element which contains row information
24484    id: "id"                 // The element within the row that provides an ID for the record (optional)
24485 }, RecordDef);
24486 </code></pre>
24487  * <p>
24488  * This would consume an XML file like this:
24489  * <pre><code>
24490 &lt;?xml?>
24491 &lt;dataset>
24492  &lt;results>2&lt;/results>
24493  &lt;row>
24494    &lt;id>1&lt;/id>
24495    &lt;name>Bill&lt;/name>
24496    &lt;occupation>Gardener&lt;/occupation>
24497  &lt;/row>
24498  &lt;row>
24499    &lt;id>2&lt;/id>
24500    &lt;name>Ben&lt;/name>
24501    &lt;occupation>Horticulturalist&lt;/occupation>
24502  &lt;/row>
24503 &lt;/dataset>
24504 </code></pre>
24505  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24506  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24507  * paged from the remote server.
24508  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24509  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24510  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24511  * a record identifier value.
24512  * @constructor
24513  * Create a new XmlReader
24514  * @param {Object} meta Metadata configuration options
24515  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24516  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24517  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24518  */
24519 Roo.data.XmlReader = function(meta, recordType){
24520     meta = meta || {};
24521     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24522 };
24523 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24524     /**
24525      * This method is only used by a DataProxy which has retrieved data from a remote server.
24526          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24527          * to contain a method called 'responseXML' that returns an XML document object.
24528      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24529      * a cache of Roo.data.Records.
24530      */
24531     read : function(response){
24532         var doc = response.responseXML;
24533         if(!doc) {
24534             throw {message: "XmlReader.read: XML Document not available"};
24535         }
24536         return this.readRecords(doc);
24537     },
24538
24539     /**
24540      * Create a data block containing Roo.data.Records from an XML document.
24541          * @param {Object} doc A parsed XML document.
24542      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24543      * a cache of Roo.data.Records.
24544      */
24545     readRecords : function(doc){
24546         /**
24547          * After any data loads/reads, the raw XML Document is available for further custom processing.
24548          * @type XMLDocument
24549          */
24550         this.xmlData = doc;
24551         var root = doc.documentElement || doc;
24552         var q = Roo.DomQuery;
24553         var recordType = this.recordType, fields = recordType.prototype.fields;
24554         var sid = this.meta.id;
24555         var totalRecords = 0, success = true;
24556         if(this.meta.totalRecords){
24557             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24558         }
24559         
24560         if(this.meta.success){
24561             var sv = q.selectValue(this.meta.success, root, true);
24562             success = sv !== false && sv !== 'false';
24563         }
24564         var records = [];
24565         var ns = q.select(this.meta.record, root);
24566         for(var i = 0, len = ns.length; i < len; i++) {
24567                 var n = ns[i];
24568                 var values = {};
24569                 var id = sid ? q.selectValue(sid, n) : undefined;
24570                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24571                     var f = fields.items[j];
24572                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24573                     v = f.convert(v);
24574                     values[f.name] = v;
24575                 }
24576                 var record = new recordType(values, id);
24577                 record.node = n;
24578                 records[records.length] = record;
24579             }
24580
24581             return {
24582                 success : success,
24583                 records : records,
24584                 totalRecords : totalRecords || records.length
24585             };
24586     }
24587 });/*
24588  * Based on:
24589  * Ext JS Library 1.1.1
24590  * Copyright(c) 2006-2007, Ext JS, LLC.
24591  *
24592  * Originally Released Under LGPL - original licence link has changed is not relivant.
24593  *
24594  * Fork - LGPL
24595  * <script type="text/javascript">
24596  */
24597
24598 /**
24599  * @class Roo.data.ArrayReader
24600  * @extends Roo.data.DataReader
24601  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24602  * Each element of that Array represents a row of data fields. The
24603  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24604  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24605  * <p>
24606  * Example code:.
24607  * <pre><code>
24608 var RecordDef = Roo.data.Record.create([
24609     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24610     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24611 ]);
24612 var myReader = new Roo.data.ArrayReader({
24613     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24614 }, RecordDef);
24615 </code></pre>
24616  * <p>
24617  * This would consume an Array like this:
24618  * <pre><code>
24619 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24620   </code></pre>
24621  
24622  * @constructor
24623  * Create a new JsonReader
24624  * @param {Object} meta Metadata configuration options.
24625  * @param {Object|Array} recordType Either an Array of field definition objects
24626  * 
24627  * @cfg {Array} fields Array of field definition objects
24628  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24629  * as specified to {@link Roo.data.Record#create},
24630  * or an {@link Roo.data.Record} object
24631  *
24632  * 
24633  * created using {@link Roo.data.Record#create}.
24634  */
24635 Roo.data.ArrayReader = function(meta, recordType){
24636     
24637      
24638     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24639 };
24640
24641 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24642     /**
24643      * Create a data block containing Roo.data.Records from an XML document.
24644      * @param {Object} o An Array of row objects which represents the dataset.
24645      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24646      * a cache of Roo.data.Records.
24647      */
24648     readRecords : function(o)
24649     {
24650         var sid = this.meta ? this.meta.id : null;
24651         var recordType = this.recordType, fields = recordType.prototype.fields;
24652         var records = [];
24653         var root = o;
24654         for(var i = 0; i < root.length; i++){
24655                 var n = root[i];
24656             var values = {};
24657             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24658             for(var j = 0, jlen = fields.length; j < jlen; j++){
24659                 var f = fields.items[j];
24660                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24661                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24662                 v = f.convert(v);
24663                 values[f.name] = v;
24664             }
24665             var record = new recordType(values, id);
24666             record.json = n;
24667             records[records.length] = record;
24668         }
24669         return {
24670             records : records,
24671             totalRecords : records.length
24672         };
24673     }
24674 });/*
24675  * Based on:
24676  * Ext JS Library 1.1.1
24677  * Copyright(c) 2006-2007, Ext JS, LLC.
24678  *
24679  * Originally Released Under LGPL - original licence link has changed is not relivant.
24680  *
24681  * Fork - LGPL
24682  * <script type="text/javascript">
24683  */
24684
24685
24686 /**
24687  * @class Roo.data.Tree
24688  * @extends Roo.util.Observable
24689  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24690  * in the tree have most standard DOM functionality.
24691  * @constructor
24692  * @param {Node} root (optional) The root node
24693  */
24694 Roo.data.Tree = function(root){
24695    this.nodeHash = {};
24696    /**
24697     * The root node for this tree
24698     * @type Node
24699     */
24700    this.root = null;
24701    if(root){
24702        this.setRootNode(root);
24703    }
24704    this.addEvents({
24705        /**
24706         * @event append
24707         * Fires when a new child node is appended to a node in this tree.
24708         * @param {Tree} tree The owner tree
24709         * @param {Node} parent The parent node
24710         * @param {Node} node The newly appended node
24711         * @param {Number} index The index of the newly appended node
24712         */
24713        "append" : true,
24714        /**
24715         * @event remove
24716         * Fires when a child node is removed from a node in this tree.
24717         * @param {Tree} tree The owner tree
24718         * @param {Node} parent The parent node
24719         * @param {Node} node The child node removed
24720         */
24721        "remove" : true,
24722        /**
24723         * @event move
24724         * Fires when a node is moved to a new location in the tree
24725         * @param {Tree} tree The owner tree
24726         * @param {Node} node The node moved
24727         * @param {Node} oldParent The old parent of this node
24728         * @param {Node} newParent The new parent of this node
24729         * @param {Number} index The index it was moved to
24730         */
24731        "move" : true,
24732        /**
24733         * @event insert
24734         * Fires when a new child node is inserted in a node in this tree.
24735         * @param {Tree} tree The owner tree
24736         * @param {Node} parent The parent node
24737         * @param {Node} node The child node inserted
24738         * @param {Node} refNode The child node the node was inserted before
24739         */
24740        "insert" : true,
24741        /**
24742         * @event beforeappend
24743         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24744         * @param {Tree} tree The owner tree
24745         * @param {Node} parent The parent node
24746         * @param {Node} node The child node to be appended
24747         */
24748        "beforeappend" : true,
24749        /**
24750         * @event beforeremove
24751         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24752         * @param {Tree} tree The owner tree
24753         * @param {Node} parent The parent node
24754         * @param {Node} node The child node to be removed
24755         */
24756        "beforeremove" : true,
24757        /**
24758         * @event beforemove
24759         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24760         * @param {Tree} tree The owner tree
24761         * @param {Node} node The node being moved
24762         * @param {Node} oldParent The parent of the node
24763         * @param {Node} newParent The new parent the node is moving to
24764         * @param {Number} index The index it is being moved to
24765         */
24766        "beforemove" : true,
24767        /**
24768         * @event beforeinsert
24769         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24770         * @param {Tree} tree The owner tree
24771         * @param {Node} parent The parent node
24772         * @param {Node} node The child node to be inserted
24773         * @param {Node} refNode The child node the node is being inserted before
24774         */
24775        "beforeinsert" : true
24776    });
24777
24778     Roo.data.Tree.superclass.constructor.call(this);
24779 };
24780
24781 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24782     pathSeparator: "/",
24783
24784     proxyNodeEvent : function(){
24785         return this.fireEvent.apply(this, arguments);
24786     },
24787
24788     /**
24789      * Returns the root node for this tree.
24790      * @return {Node}
24791      */
24792     getRootNode : function(){
24793         return this.root;
24794     },
24795
24796     /**
24797      * Sets the root node for this tree.
24798      * @param {Node} node
24799      * @return {Node}
24800      */
24801     setRootNode : function(node){
24802         this.root = node;
24803         node.ownerTree = this;
24804         node.isRoot = true;
24805         this.registerNode(node);
24806         return node;
24807     },
24808
24809     /**
24810      * Gets a node in this tree by its id.
24811      * @param {String} id
24812      * @return {Node}
24813      */
24814     getNodeById : function(id){
24815         return this.nodeHash[id];
24816     },
24817
24818     registerNode : function(node){
24819         this.nodeHash[node.id] = node;
24820     },
24821
24822     unregisterNode : function(node){
24823         delete this.nodeHash[node.id];
24824     },
24825
24826     toString : function(){
24827         return "[Tree"+(this.id?" "+this.id:"")+"]";
24828     }
24829 });
24830
24831 /**
24832  * @class Roo.data.Node
24833  * @extends Roo.util.Observable
24834  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24835  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24836  * @constructor
24837  * @param {Object} attributes The attributes/config for the node
24838  */
24839 Roo.data.Node = function(attributes){
24840     /**
24841      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24842      * @type {Object}
24843      */
24844     this.attributes = attributes || {};
24845     this.leaf = this.attributes.leaf;
24846     /**
24847      * The node id. @type String
24848      */
24849     this.id = this.attributes.id;
24850     if(!this.id){
24851         this.id = Roo.id(null, "ynode-");
24852         this.attributes.id = this.id;
24853     }
24854      
24855     
24856     /**
24857      * All child nodes of this node. @type Array
24858      */
24859     this.childNodes = [];
24860     if(!this.childNodes.indexOf){ // indexOf is a must
24861         this.childNodes.indexOf = function(o){
24862             for(var i = 0, len = this.length; i < len; i++){
24863                 if(this[i] == o) {
24864                     return i;
24865                 }
24866             }
24867             return -1;
24868         };
24869     }
24870     /**
24871      * The parent node for this node. @type Node
24872      */
24873     this.parentNode = null;
24874     /**
24875      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24876      */
24877     this.firstChild = null;
24878     /**
24879      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24880      */
24881     this.lastChild = null;
24882     /**
24883      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24884      */
24885     this.previousSibling = null;
24886     /**
24887      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24888      */
24889     this.nextSibling = null;
24890
24891     this.addEvents({
24892        /**
24893         * @event append
24894         * Fires when a new child node is appended
24895         * @param {Tree} tree The owner tree
24896         * @param {Node} this This node
24897         * @param {Node} node The newly appended node
24898         * @param {Number} index The index of the newly appended node
24899         */
24900        "append" : true,
24901        /**
24902         * @event remove
24903         * Fires when a child node is removed
24904         * @param {Tree} tree The owner tree
24905         * @param {Node} this This node
24906         * @param {Node} node The removed node
24907         */
24908        "remove" : true,
24909        /**
24910         * @event move
24911         * Fires when this node is moved to a new location in the tree
24912         * @param {Tree} tree The owner tree
24913         * @param {Node} this This node
24914         * @param {Node} oldParent The old parent of this node
24915         * @param {Node} newParent The new parent of this node
24916         * @param {Number} index The index it was moved to
24917         */
24918        "move" : true,
24919        /**
24920         * @event insert
24921         * Fires when a new child node is inserted.
24922         * @param {Tree} tree The owner tree
24923         * @param {Node} this This node
24924         * @param {Node} node The child node inserted
24925         * @param {Node} refNode The child node the node was inserted before
24926         */
24927        "insert" : true,
24928        /**
24929         * @event beforeappend
24930         * Fires before a new child is appended, return false to cancel the append.
24931         * @param {Tree} tree The owner tree
24932         * @param {Node} this This node
24933         * @param {Node} node The child node to be appended
24934         */
24935        "beforeappend" : true,
24936        /**
24937         * @event beforeremove
24938         * Fires before a child is removed, return false to cancel the remove.
24939         * @param {Tree} tree The owner tree
24940         * @param {Node} this This node
24941         * @param {Node} node The child node to be removed
24942         */
24943        "beforeremove" : true,
24944        /**
24945         * @event beforemove
24946         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24947         * @param {Tree} tree The owner tree
24948         * @param {Node} this This node
24949         * @param {Node} oldParent The parent of this node
24950         * @param {Node} newParent The new parent this node is moving to
24951         * @param {Number} index The index it is being moved to
24952         */
24953        "beforemove" : true,
24954        /**
24955         * @event beforeinsert
24956         * Fires before a new child is inserted, return false to cancel the insert.
24957         * @param {Tree} tree The owner tree
24958         * @param {Node} this This node
24959         * @param {Node} node The child node to be inserted
24960         * @param {Node} refNode The child node the node is being inserted before
24961         */
24962        "beforeinsert" : true
24963    });
24964     this.listeners = this.attributes.listeners;
24965     Roo.data.Node.superclass.constructor.call(this);
24966 };
24967
24968 Roo.extend(Roo.data.Node, Roo.util.Observable, {
24969     fireEvent : function(evtName){
24970         // first do standard event for this node
24971         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
24972             return false;
24973         }
24974         // then bubble it up to the tree if the event wasn't cancelled
24975         var ot = this.getOwnerTree();
24976         if(ot){
24977             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
24978                 return false;
24979             }
24980         }
24981         return true;
24982     },
24983
24984     /**
24985      * Returns true if this node is a leaf
24986      * @return {Boolean}
24987      */
24988     isLeaf : function(){
24989         return this.leaf === true;
24990     },
24991
24992     // private
24993     setFirstChild : function(node){
24994         this.firstChild = node;
24995     },
24996
24997     //private
24998     setLastChild : function(node){
24999         this.lastChild = node;
25000     },
25001
25002
25003     /**
25004      * Returns true if this node is the last child of its parent
25005      * @return {Boolean}
25006      */
25007     isLast : function(){
25008        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25009     },
25010
25011     /**
25012      * Returns true if this node is the first child of its parent
25013      * @return {Boolean}
25014      */
25015     isFirst : function(){
25016        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25017     },
25018
25019     hasChildNodes : function(){
25020         return !this.isLeaf() && this.childNodes.length > 0;
25021     },
25022
25023     /**
25024      * Insert node(s) as the last child node of this node.
25025      * @param {Node/Array} node The node or Array of nodes to append
25026      * @return {Node} The appended node if single append, or null if an array was passed
25027      */
25028     appendChild : function(node){
25029         var multi = false;
25030         if(node instanceof Array){
25031             multi = node;
25032         }else if(arguments.length > 1){
25033             multi = arguments;
25034         }
25035         
25036         // if passed an array or multiple args do them one by one
25037         if(multi){
25038             for(var i = 0, len = multi.length; i < len; i++) {
25039                 this.appendChild(multi[i]);
25040             }
25041         }else{
25042             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25043                 return false;
25044             }
25045             var index = this.childNodes.length;
25046             var oldParent = node.parentNode;
25047             // it's a move, make sure we move it cleanly
25048             if(oldParent){
25049                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25050                     return false;
25051                 }
25052                 oldParent.removeChild(node);
25053             }
25054             
25055             index = this.childNodes.length;
25056             if(index == 0){
25057                 this.setFirstChild(node);
25058             }
25059             this.childNodes.push(node);
25060             node.parentNode = this;
25061             var ps = this.childNodes[index-1];
25062             if(ps){
25063                 node.previousSibling = ps;
25064                 ps.nextSibling = node;
25065             }else{
25066                 node.previousSibling = null;
25067             }
25068             node.nextSibling = null;
25069             this.setLastChild(node);
25070             node.setOwnerTree(this.getOwnerTree());
25071             this.fireEvent("append", this.ownerTree, this, node, index);
25072             if(this.ownerTree) {
25073                 this.ownerTree.fireEvent("appendnode", this, node, index);
25074             }
25075             if(oldParent){
25076                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25077             }
25078             return node;
25079         }
25080     },
25081
25082     /**
25083      * Removes a child node from this node.
25084      * @param {Node} node The node to remove
25085      * @return {Node} The removed node
25086      */
25087     removeChild : function(node){
25088         var index = this.childNodes.indexOf(node);
25089         if(index == -1){
25090             return false;
25091         }
25092         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25093             return false;
25094         }
25095
25096         // remove it from childNodes collection
25097         this.childNodes.splice(index, 1);
25098
25099         // update siblings
25100         if(node.previousSibling){
25101             node.previousSibling.nextSibling = node.nextSibling;
25102         }
25103         if(node.nextSibling){
25104             node.nextSibling.previousSibling = node.previousSibling;
25105         }
25106
25107         // update child refs
25108         if(this.firstChild == node){
25109             this.setFirstChild(node.nextSibling);
25110         }
25111         if(this.lastChild == node){
25112             this.setLastChild(node.previousSibling);
25113         }
25114
25115         node.setOwnerTree(null);
25116         // clear any references from the node
25117         node.parentNode = null;
25118         node.previousSibling = null;
25119         node.nextSibling = null;
25120         this.fireEvent("remove", this.ownerTree, this, node);
25121         return node;
25122     },
25123
25124     /**
25125      * Inserts the first node before the second node in this nodes childNodes collection.
25126      * @param {Node} node The node to insert
25127      * @param {Node} refNode The node to insert before (if null the node is appended)
25128      * @return {Node} The inserted node
25129      */
25130     insertBefore : function(node, refNode){
25131         if(!refNode){ // like standard Dom, refNode can be null for append
25132             return this.appendChild(node);
25133         }
25134         // nothing to do
25135         if(node == refNode){
25136             return false;
25137         }
25138
25139         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25140             return false;
25141         }
25142         var index = this.childNodes.indexOf(refNode);
25143         var oldParent = node.parentNode;
25144         var refIndex = index;
25145
25146         // when moving internally, indexes will change after remove
25147         if(oldParent == this && this.childNodes.indexOf(node) < index){
25148             refIndex--;
25149         }
25150
25151         // it's a move, make sure we move it cleanly
25152         if(oldParent){
25153             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25154                 return false;
25155             }
25156             oldParent.removeChild(node);
25157         }
25158         if(refIndex == 0){
25159             this.setFirstChild(node);
25160         }
25161         this.childNodes.splice(refIndex, 0, node);
25162         node.parentNode = this;
25163         var ps = this.childNodes[refIndex-1];
25164         if(ps){
25165             node.previousSibling = ps;
25166             ps.nextSibling = node;
25167         }else{
25168             node.previousSibling = null;
25169         }
25170         node.nextSibling = refNode;
25171         refNode.previousSibling = node;
25172         node.setOwnerTree(this.getOwnerTree());
25173         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25174         if(oldParent){
25175             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25176         }
25177         return node;
25178     },
25179
25180     /**
25181      * Returns the child node at the specified index.
25182      * @param {Number} index
25183      * @return {Node}
25184      */
25185     item : function(index){
25186         return this.childNodes[index];
25187     },
25188
25189     /**
25190      * Replaces one child node in this node with another.
25191      * @param {Node} newChild The replacement node
25192      * @param {Node} oldChild The node to replace
25193      * @return {Node} The replaced node
25194      */
25195     replaceChild : function(newChild, oldChild){
25196         this.insertBefore(newChild, oldChild);
25197         this.removeChild(oldChild);
25198         return oldChild;
25199     },
25200
25201     /**
25202      * Returns the index of a child node
25203      * @param {Node} node
25204      * @return {Number} The index of the node or -1 if it was not found
25205      */
25206     indexOf : function(child){
25207         return this.childNodes.indexOf(child);
25208     },
25209
25210     /**
25211      * Returns the tree this node is in.
25212      * @return {Tree}
25213      */
25214     getOwnerTree : function(){
25215         // if it doesn't have one, look for one
25216         if(!this.ownerTree){
25217             var p = this;
25218             while(p){
25219                 if(p.ownerTree){
25220                     this.ownerTree = p.ownerTree;
25221                     break;
25222                 }
25223                 p = p.parentNode;
25224             }
25225         }
25226         return this.ownerTree;
25227     },
25228
25229     /**
25230      * Returns depth of this node (the root node has a depth of 0)
25231      * @return {Number}
25232      */
25233     getDepth : function(){
25234         var depth = 0;
25235         var p = this;
25236         while(p.parentNode){
25237             ++depth;
25238             p = p.parentNode;
25239         }
25240         return depth;
25241     },
25242
25243     // private
25244     setOwnerTree : function(tree){
25245         // if it's move, we need to update everyone
25246         if(tree != this.ownerTree){
25247             if(this.ownerTree){
25248                 this.ownerTree.unregisterNode(this);
25249             }
25250             this.ownerTree = tree;
25251             var cs = this.childNodes;
25252             for(var i = 0, len = cs.length; i < len; i++) {
25253                 cs[i].setOwnerTree(tree);
25254             }
25255             if(tree){
25256                 tree.registerNode(this);
25257             }
25258         }
25259     },
25260
25261     /**
25262      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25263      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25264      * @return {String} The path
25265      */
25266     getPath : function(attr){
25267         attr = attr || "id";
25268         var p = this.parentNode;
25269         var b = [this.attributes[attr]];
25270         while(p){
25271             b.unshift(p.attributes[attr]);
25272             p = p.parentNode;
25273         }
25274         var sep = this.getOwnerTree().pathSeparator;
25275         return sep + b.join(sep);
25276     },
25277
25278     /**
25279      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25280      * function call will be the scope provided or the current node. The arguments to the function
25281      * will be the args provided or the current node. If the function returns false at any point,
25282      * the bubble is stopped.
25283      * @param {Function} fn The function to call
25284      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25285      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25286      */
25287     bubble : function(fn, scope, args){
25288         var p = this;
25289         while(p){
25290             if(fn.call(scope || p, args || p) === false){
25291                 break;
25292             }
25293             p = p.parentNode;
25294         }
25295     },
25296
25297     /**
25298      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25299      * function call will be the scope provided or the current node. The arguments to the function
25300      * will be the args provided or the current node. If the function returns false at any point,
25301      * the cascade is stopped on that branch.
25302      * @param {Function} fn The function to call
25303      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25304      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25305      */
25306     cascade : function(fn, scope, args){
25307         if(fn.call(scope || this, args || this) !== false){
25308             var cs = this.childNodes;
25309             for(var i = 0, len = cs.length; i < len; i++) {
25310                 cs[i].cascade(fn, scope, args);
25311             }
25312         }
25313     },
25314
25315     /**
25316      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25317      * function call will be the scope provided or the current node. The arguments to the function
25318      * will be the args provided or the current node. If the function returns false at any point,
25319      * the iteration stops.
25320      * @param {Function} fn The function to call
25321      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25322      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25323      */
25324     eachChild : function(fn, scope, args){
25325         var cs = this.childNodes;
25326         for(var i = 0, len = cs.length; i < len; i++) {
25327                 if(fn.call(scope || this, args || cs[i]) === false){
25328                     break;
25329                 }
25330         }
25331     },
25332
25333     /**
25334      * Finds the first child that has the attribute with the specified value.
25335      * @param {String} attribute The attribute name
25336      * @param {Mixed} value The value to search for
25337      * @return {Node} The found child or null if none was found
25338      */
25339     findChild : function(attribute, value){
25340         var cs = this.childNodes;
25341         for(var i = 0, len = cs.length; i < len; i++) {
25342                 if(cs[i].attributes[attribute] == value){
25343                     return cs[i];
25344                 }
25345         }
25346         return null;
25347     },
25348
25349     /**
25350      * Finds the first child by a custom function. The child matches if the function passed
25351      * returns true.
25352      * @param {Function} fn
25353      * @param {Object} scope (optional)
25354      * @return {Node} The found child or null if none was found
25355      */
25356     findChildBy : function(fn, scope){
25357         var cs = this.childNodes;
25358         for(var i = 0, len = cs.length; i < len; i++) {
25359                 if(fn.call(scope||cs[i], cs[i]) === true){
25360                     return cs[i];
25361                 }
25362         }
25363         return null;
25364     },
25365
25366     /**
25367      * Sorts this nodes children using the supplied sort function
25368      * @param {Function} fn
25369      * @param {Object} scope (optional)
25370      */
25371     sort : function(fn, scope){
25372         var cs = this.childNodes;
25373         var len = cs.length;
25374         if(len > 0){
25375             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25376             cs.sort(sortFn);
25377             for(var i = 0; i < len; i++){
25378                 var n = cs[i];
25379                 n.previousSibling = cs[i-1];
25380                 n.nextSibling = cs[i+1];
25381                 if(i == 0){
25382                     this.setFirstChild(n);
25383                 }
25384                 if(i == len-1){
25385                     this.setLastChild(n);
25386                 }
25387             }
25388         }
25389     },
25390
25391     /**
25392      * Returns true if this node is an ancestor (at any point) of the passed node.
25393      * @param {Node} node
25394      * @return {Boolean}
25395      */
25396     contains : function(node){
25397         return node.isAncestor(this);
25398     },
25399
25400     /**
25401      * Returns true if the passed node is an ancestor (at any point) of this node.
25402      * @param {Node} node
25403      * @return {Boolean}
25404      */
25405     isAncestor : function(node){
25406         var p = this.parentNode;
25407         while(p){
25408             if(p == node){
25409                 return true;
25410             }
25411             p = p.parentNode;
25412         }
25413         return false;
25414     },
25415
25416     toString : function(){
25417         return "[Node"+(this.id?" "+this.id:"")+"]";
25418     }
25419 });/*
25420  * Based on:
25421  * Ext JS Library 1.1.1
25422  * Copyright(c) 2006-2007, Ext JS, LLC.
25423  *
25424  * Originally Released Under LGPL - original licence link has changed is not relivant.
25425  *
25426  * Fork - LGPL
25427  * <script type="text/javascript">
25428  */
25429  (function(){ 
25430 /**
25431  * @class Roo.Layer
25432  * @extends Roo.Element
25433  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25434  * automatic maintaining of shadow/shim positions.
25435  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25436  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25437  * you can pass a string with a CSS class name. False turns off the shadow.
25438  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25439  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25440  * @cfg {String} cls CSS class to add to the element
25441  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25442  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25443  * @constructor
25444  * @param {Object} config An object with config options.
25445  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25446  */
25447
25448 Roo.Layer = function(config, existingEl){
25449     config = config || {};
25450     var dh = Roo.DomHelper;
25451     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25452     if(existingEl){
25453         this.dom = Roo.getDom(existingEl);
25454     }
25455     if(!this.dom){
25456         var o = config.dh || {tag: "div", cls: "x-layer"};
25457         this.dom = dh.append(pel, o);
25458     }
25459     if(config.cls){
25460         this.addClass(config.cls);
25461     }
25462     this.constrain = config.constrain !== false;
25463     this.visibilityMode = Roo.Element.VISIBILITY;
25464     if(config.id){
25465         this.id = this.dom.id = config.id;
25466     }else{
25467         this.id = Roo.id(this.dom);
25468     }
25469     this.zindex = config.zindex || this.getZIndex();
25470     this.position("absolute", this.zindex);
25471     if(config.shadow){
25472         this.shadowOffset = config.shadowOffset || 4;
25473         this.shadow = new Roo.Shadow({
25474             offset : this.shadowOffset,
25475             mode : config.shadow
25476         });
25477     }else{
25478         this.shadowOffset = 0;
25479     }
25480     this.useShim = config.shim !== false && Roo.useShims;
25481     this.useDisplay = config.useDisplay;
25482     this.hide();
25483 };
25484
25485 var supr = Roo.Element.prototype;
25486
25487 // shims are shared among layer to keep from having 100 iframes
25488 var shims = [];
25489
25490 Roo.extend(Roo.Layer, Roo.Element, {
25491
25492     getZIndex : function(){
25493         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25494     },
25495
25496     getShim : function(){
25497         if(!this.useShim){
25498             return null;
25499         }
25500         if(this.shim){
25501             return this.shim;
25502         }
25503         var shim = shims.shift();
25504         if(!shim){
25505             shim = this.createShim();
25506             shim.enableDisplayMode('block');
25507             shim.dom.style.display = 'none';
25508             shim.dom.style.visibility = 'visible';
25509         }
25510         var pn = this.dom.parentNode;
25511         if(shim.dom.parentNode != pn){
25512             pn.insertBefore(shim.dom, this.dom);
25513         }
25514         shim.setStyle('z-index', this.getZIndex()-2);
25515         this.shim = shim;
25516         return shim;
25517     },
25518
25519     hideShim : function(){
25520         if(this.shim){
25521             this.shim.setDisplayed(false);
25522             shims.push(this.shim);
25523             delete this.shim;
25524         }
25525     },
25526
25527     disableShadow : function(){
25528         if(this.shadow){
25529             this.shadowDisabled = true;
25530             this.shadow.hide();
25531             this.lastShadowOffset = this.shadowOffset;
25532             this.shadowOffset = 0;
25533         }
25534     },
25535
25536     enableShadow : function(show){
25537         if(this.shadow){
25538             this.shadowDisabled = false;
25539             this.shadowOffset = this.lastShadowOffset;
25540             delete this.lastShadowOffset;
25541             if(show){
25542                 this.sync(true);
25543             }
25544         }
25545     },
25546
25547     // private
25548     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25549     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25550     sync : function(doShow){
25551         var sw = this.shadow;
25552         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25553             var sh = this.getShim();
25554
25555             var w = this.getWidth(),
25556                 h = this.getHeight();
25557
25558             var l = this.getLeft(true),
25559                 t = this.getTop(true);
25560
25561             if(sw && !this.shadowDisabled){
25562                 if(doShow && !sw.isVisible()){
25563                     sw.show(this);
25564                 }else{
25565                     sw.realign(l, t, w, h);
25566                 }
25567                 if(sh){
25568                     if(doShow){
25569                        sh.show();
25570                     }
25571                     // fit the shim behind the shadow, so it is shimmed too
25572                     var a = sw.adjusts, s = sh.dom.style;
25573                     s.left = (Math.min(l, l+a.l))+"px";
25574                     s.top = (Math.min(t, t+a.t))+"px";
25575                     s.width = (w+a.w)+"px";
25576                     s.height = (h+a.h)+"px";
25577                 }
25578             }else if(sh){
25579                 if(doShow){
25580                    sh.show();
25581                 }
25582                 sh.setSize(w, h);
25583                 sh.setLeftTop(l, t);
25584             }
25585             
25586         }
25587     },
25588
25589     // private
25590     destroy : function(){
25591         this.hideShim();
25592         if(this.shadow){
25593             this.shadow.hide();
25594         }
25595         this.removeAllListeners();
25596         var pn = this.dom.parentNode;
25597         if(pn){
25598             pn.removeChild(this.dom);
25599         }
25600         Roo.Element.uncache(this.id);
25601     },
25602
25603     remove : function(){
25604         this.destroy();
25605     },
25606
25607     // private
25608     beginUpdate : function(){
25609         this.updating = true;
25610     },
25611
25612     // private
25613     endUpdate : function(){
25614         this.updating = false;
25615         this.sync(true);
25616     },
25617
25618     // private
25619     hideUnders : function(negOffset){
25620         if(this.shadow){
25621             this.shadow.hide();
25622         }
25623         this.hideShim();
25624     },
25625
25626     // private
25627     constrainXY : function(){
25628         if(this.constrain){
25629             var vw = Roo.lib.Dom.getViewWidth(),
25630                 vh = Roo.lib.Dom.getViewHeight();
25631             var s = Roo.get(document).getScroll();
25632
25633             var xy = this.getXY();
25634             var x = xy[0], y = xy[1];   
25635             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25636             // only move it if it needs it
25637             var moved = false;
25638             // first validate right/bottom
25639             if((x + w) > vw+s.left){
25640                 x = vw - w - this.shadowOffset;
25641                 moved = true;
25642             }
25643             if((y + h) > vh+s.top){
25644                 y = vh - h - this.shadowOffset;
25645                 moved = true;
25646             }
25647             // then make sure top/left isn't negative
25648             if(x < s.left){
25649                 x = s.left;
25650                 moved = true;
25651             }
25652             if(y < s.top){
25653                 y = s.top;
25654                 moved = true;
25655             }
25656             if(moved){
25657                 if(this.avoidY){
25658                     var ay = this.avoidY;
25659                     if(y <= ay && (y+h) >= ay){
25660                         y = ay-h-5;   
25661                     }
25662                 }
25663                 xy = [x, y];
25664                 this.storeXY(xy);
25665                 supr.setXY.call(this, xy);
25666                 this.sync();
25667             }
25668         }
25669     },
25670
25671     isVisible : function(){
25672         return this.visible;    
25673     },
25674
25675     // private
25676     showAction : function(){
25677         this.visible = true; // track visibility to prevent getStyle calls
25678         if(this.useDisplay === true){
25679             this.setDisplayed("");
25680         }else if(this.lastXY){
25681             supr.setXY.call(this, this.lastXY);
25682         }else if(this.lastLT){
25683             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25684         }
25685     },
25686
25687     // private
25688     hideAction : function(){
25689         this.visible = false;
25690         if(this.useDisplay === true){
25691             this.setDisplayed(false);
25692         }else{
25693             this.setLeftTop(-10000,-10000);
25694         }
25695     },
25696
25697     // overridden Element method
25698     setVisible : function(v, a, d, c, e){
25699         if(v){
25700             this.showAction();
25701         }
25702         if(a && v){
25703             var cb = function(){
25704                 this.sync(true);
25705                 if(c){
25706                     c();
25707                 }
25708             }.createDelegate(this);
25709             supr.setVisible.call(this, true, true, d, cb, e);
25710         }else{
25711             if(!v){
25712                 this.hideUnders(true);
25713             }
25714             var cb = c;
25715             if(a){
25716                 cb = function(){
25717                     this.hideAction();
25718                     if(c){
25719                         c();
25720                     }
25721                 }.createDelegate(this);
25722             }
25723             supr.setVisible.call(this, v, a, d, cb, e);
25724             if(v){
25725                 this.sync(true);
25726             }else if(!a){
25727                 this.hideAction();
25728             }
25729         }
25730     },
25731
25732     storeXY : function(xy){
25733         delete this.lastLT;
25734         this.lastXY = xy;
25735     },
25736
25737     storeLeftTop : function(left, top){
25738         delete this.lastXY;
25739         this.lastLT = [left, top];
25740     },
25741
25742     // private
25743     beforeFx : function(){
25744         this.beforeAction();
25745         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25746     },
25747
25748     // private
25749     afterFx : function(){
25750         Roo.Layer.superclass.afterFx.apply(this, arguments);
25751         this.sync(this.isVisible());
25752     },
25753
25754     // private
25755     beforeAction : function(){
25756         if(!this.updating && this.shadow){
25757             this.shadow.hide();
25758         }
25759     },
25760
25761     // overridden Element method
25762     setLeft : function(left){
25763         this.storeLeftTop(left, this.getTop(true));
25764         supr.setLeft.apply(this, arguments);
25765         this.sync();
25766     },
25767
25768     setTop : function(top){
25769         this.storeLeftTop(this.getLeft(true), top);
25770         supr.setTop.apply(this, arguments);
25771         this.sync();
25772     },
25773
25774     setLeftTop : function(left, top){
25775         this.storeLeftTop(left, top);
25776         supr.setLeftTop.apply(this, arguments);
25777         this.sync();
25778     },
25779
25780     setXY : function(xy, a, d, c, e){
25781         this.fixDisplay();
25782         this.beforeAction();
25783         this.storeXY(xy);
25784         var cb = this.createCB(c);
25785         supr.setXY.call(this, xy, a, d, cb, e);
25786         if(!a){
25787             cb();
25788         }
25789     },
25790
25791     // private
25792     createCB : function(c){
25793         var el = this;
25794         return function(){
25795             el.constrainXY();
25796             el.sync(true);
25797             if(c){
25798                 c();
25799             }
25800         };
25801     },
25802
25803     // overridden Element method
25804     setX : function(x, a, d, c, e){
25805         this.setXY([x, this.getY()], a, d, c, e);
25806     },
25807
25808     // overridden Element method
25809     setY : function(y, a, d, c, e){
25810         this.setXY([this.getX(), y], a, d, c, e);
25811     },
25812
25813     // overridden Element method
25814     setSize : function(w, h, a, d, c, e){
25815         this.beforeAction();
25816         var cb = this.createCB(c);
25817         supr.setSize.call(this, w, h, a, d, cb, e);
25818         if(!a){
25819             cb();
25820         }
25821     },
25822
25823     // overridden Element method
25824     setWidth : function(w, a, d, c, e){
25825         this.beforeAction();
25826         var cb = this.createCB(c);
25827         supr.setWidth.call(this, w, a, d, cb, e);
25828         if(!a){
25829             cb();
25830         }
25831     },
25832
25833     // overridden Element method
25834     setHeight : function(h, a, d, c, e){
25835         this.beforeAction();
25836         var cb = this.createCB(c);
25837         supr.setHeight.call(this, h, a, d, cb, e);
25838         if(!a){
25839             cb();
25840         }
25841     },
25842
25843     // overridden Element method
25844     setBounds : function(x, y, w, h, a, d, c, e){
25845         this.beforeAction();
25846         var cb = this.createCB(c);
25847         if(!a){
25848             this.storeXY([x, y]);
25849             supr.setXY.call(this, [x, y]);
25850             supr.setSize.call(this, w, h, a, d, cb, e);
25851             cb();
25852         }else{
25853             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25854         }
25855         return this;
25856     },
25857     
25858     /**
25859      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25860      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25861      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25862      * @param {Number} zindex The new z-index to set
25863      * @return {this} The Layer
25864      */
25865     setZIndex : function(zindex){
25866         this.zindex = zindex;
25867         this.setStyle("z-index", zindex + 2);
25868         if(this.shadow){
25869             this.shadow.setZIndex(zindex + 1);
25870         }
25871         if(this.shim){
25872             this.shim.setStyle("z-index", zindex);
25873         }
25874     }
25875 });
25876 })();/*
25877  * Based on:
25878  * Ext JS Library 1.1.1
25879  * Copyright(c) 2006-2007, Ext JS, LLC.
25880  *
25881  * Originally Released Under LGPL - original licence link has changed is not relivant.
25882  *
25883  * Fork - LGPL
25884  * <script type="text/javascript">
25885  */
25886
25887
25888 /**
25889  * @class Roo.Shadow
25890  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25891  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25892  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25893  * @constructor
25894  * Create a new Shadow
25895  * @param {Object} config The config object
25896  */
25897 Roo.Shadow = function(config){
25898     Roo.apply(this, config);
25899     if(typeof this.mode != "string"){
25900         this.mode = this.defaultMode;
25901     }
25902     var o = this.offset, a = {h: 0};
25903     var rad = Math.floor(this.offset/2);
25904     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25905         case "drop":
25906             a.w = 0;
25907             a.l = a.t = o;
25908             a.t -= 1;
25909             if(Roo.isIE){
25910                 a.l -= this.offset + rad;
25911                 a.t -= this.offset + rad;
25912                 a.w -= rad;
25913                 a.h -= rad;
25914                 a.t += 1;
25915             }
25916         break;
25917         case "sides":
25918             a.w = (o*2);
25919             a.l = -o;
25920             a.t = o-1;
25921             if(Roo.isIE){
25922                 a.l -= (this.offset - rad);
25923                 a.t -= this.offset + rad;
25924                 a.l += 1;
25925                 a.w -= (this.offset - rad)*2;
25926                 a.w -= rad + 1;
25927                 a.h -= 1;
25928             }
25929         break;
25930         case "frame":
25931             a.w = a.h = (o*2);
25932             a.l = a.t = -o;
25933             a.t += 1;
25934             a.h -= 2;
25935             if(Roo.isIE){
25936                 a.l -= (this.offset - rad);
25937                 a.t -= (this.offset - rad);
25938                 a.l += 1;
25939                 a.w -= (this.offset + rad + 1);
25940                 a.h -= (this.offset + rad);
25941                 a.h += 1;
25942             }
25943         break;
25944     };
25945
25946     this.adjusts = a;
25947 };
25948
25949 Roo.Shadow.prototype = {
25950     /**
25951      * @cfg {String} mode
25952      * The shadow display mode.  Supports the following options:<br />
25953      * sides: Shadow displays on both sides and bottom only<br />
25954      * frame: Shadow displays equally on all four sides<br />
25955      * drop: Traditional bottom-right drop shadow (default)
25956      */
25957     /**
25958      * @cfg {String} offset
25959      * The number of pixels to offset the shadow from the element (defaults to 4)
25960      */
25961     offset: 4,
25962
25963     // private
25964     defaultMode: "drop",
25965
25966     /**
25967      * Displays the shadow under the target element
25968      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
25969      */
25970     show : function(target){
25971         target = Roo.get(target);
25972         if(!this.el){
25973             this.el = Roo.Shadow.Pool.pull();
25974             if(this.el.dom.nextSibling != target.dom){
25975                 this.el.insertBefore(target);
25976             }
25977         }
25978         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
25979         if(Roo.isIE){
25980             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
25981         }
25982         this.realign(
25983             target.getLeft(true),
25984             target.getTop(true),
25985             target.getWidth(),
25986             target.getHeight()
25987         );
25988         this.el.dom.style.display = "block";
25989     },
25990
25991     /**
25992      * Returns true if the shadow is visible, else false
25993      */
25994     isVisible : function(){
25995         return this.el ? true : false;  
25996     },
25997
25998     /**
25999      * Direct alignment when values are already available. Show must be called at least once before
26000      * calling this method to ensure it is initialized.
26001      * @param {Number} left The target element left position
26002      * @param {Number} top The target element top position
26003      * @param {Number} width The target element width
26004      * @param {Number} height The target element height
26005      */
26006     realign : function(l, t, w, h){
26007         if(!this.el){
26008             return;
26009         }
26010         var a = this.adjusts, d = this.el.dom, s = d.style;
26011         var iea = 0;
26012         s.left = (l+a.l)+"px";
26013         s.top = (t+a.t)+"px";
26014         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26015  
26016         if(s.width != sws || s.height != shs){
26017             s.width = sws;
26018             s.height = shs;
26019             if(!Roo.isIE){
26020                 var cn = d.childNodes;
26021                 var sww = Math.max(0, (sw-12))+"px";
26022                 cn[0].childNodes[1].style.width = sww;
26023                 cn[1].childNodes[1].style.width = sww;
26024                 cn[2].childNodes[1].style.width = sww;
26025                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26026             }
26027         }
26028     },
26029
26030     /**
26031      * Hides this shadow
26032      */
26033     hide : function(){
26034         if(this.el){
26035             this.el.dom.style.display = "none";
26036             Roo.Shadow.Pool.push(this.el);
26037             delete this.el;
26038         }
26039     },
26040
26041     /**
26042      * Adjust the z-index of this shadow
26043      * @param {Number} zindex The new z-index
26044      */
26045     setZIndex : function(z){
26046         this.zIndex = z;
26047         if(this.el){
26048             this.el.setStyle("z-index", z);
26049         }
26050     }
26051 };
26052
26053 // Private utility class that manages the internal Shadow cache
26054 Roo.Shadow.Pool = function(){
26055     var p = [];
26056     var markup = Roo.isIE ?
26057                  '<div class="x-ie-shadow"></div>' :
26058                  '<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>';
26059     return {
26060         pull : function(){
26061             var sh = p.shift();
26062             if(!sh){
26063                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26064                 sh.autoBoxAdjust = false;
26065             }
26066             return sh;
26067         },
26068
26069         push : function(sh){
26070             p.push(sh);
26071         }
26072     };
26073 }();/*
26074  * Based on:
26075  * Ext JS Library 1.1.1
26076  * Copyright(c) 2006-2007, Ext JS, LLC.
26077  *
26078  * Originally Released Under LGPL - original licence link has changed is not relivant.
26079  *
26080  * Fork - LGPL
26081  * <script type="text/javascript">
26082  */
26083
26084
26085 /**
26086  * @class Roo.SplitBar
26087  * @extends Roo.util.Observable
26088  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26089  * <br><br>
26090  * Usage:
26091  * <pre><code>
26092 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26093                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26094 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26095 split.minSize = 100;
26096 split.maxSize = 600;
26097 split.animate = true;
26098 split.on('moved', splitterMoved);
26099 </code></pre>
26100  * @constructor
26101  * Create a new SplitBar
26102  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26103  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26104  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26105  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26106                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26107                         position of the SplitBar).
26108  */
26109 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26110     
26111     /** @private */
26112     this.el = Roo.get(dragElement, true);
26113     this.el.dom.unselectable = "on";
26114     /** @private */
26115     this.resizingEl = Roo.get(resizingElement, true);
26116
26117     /**
26118      * @private
26119      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26120      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26121      * @type Number
26122      */
26123     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26124     
26125     /**
26126      * The minimum size of the resizing element. (Defaults to 0)
26127      * @type Number
26128      */
26129     this.minSize = 0;
26130     
26131     /**
26132      * The maximum size of the resizing element. (Defaults to 2000)
26133      * @type Number
26134      */
26135     this.maxSize = 2000;
26136     
26137     /**
26138      * Whether to animate the transition to the new size
26139      * @type Boolean
26140      */
26141     this.animate = false;
26142     
26143     /**
26144      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26145      * @type Boolean
26146      */
26147     this.useShim = false;
26148     
26149     /** @private */
26150     this.shim = null;
26151     
26152     if(!existingProxy){
26153         /** @private */
26154         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26155     }else{
26156         this.proxy = Roo.get(existingProxy).dom;
26157     }
26158     /** @private */
26159     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26160     
26161     /** @private */
26162     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26163     
26164     /** @private */
26165     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26166     
26167     /** @private */
26168     this.dragSpecs = {};
26169     
26170     /**
26171      * @private The adapter to use to positon and resize elements
26172      */
26173     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26174     this.adapter.init(this);
26175     
26176     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26177         /** @private */
26178         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26179         this.el.addClass("x-splitbar-h");
26180     }else{
26181         /** @private */
26182         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26183         this.el.addClass("x-splitbar-v");
26184     }
26185     
26186     this.addEvents({
26187         /**
26188          * @event resize
26189          * Fires when the splitter is moved (alias for {@link #event-moved})
26190          * @param {Roo.SplitBar} this
26191          * @param {Number} newSize the new width or height
26192          */
26193         "resize" : true,
26194         /**
26195          * @event moved
26196          * Fires when the splitter is moved
26197          * @param {Roo.SplitBar} this
26198          * @param {Number} newSize the new width or height
26199          */
26200         "moved" : true,
26201         /**
26202          * @event beforeresize
26203          * Fires before the splitter is dragged
26204          * @param {Roo.SplitBar} this
26205          */
26206         "beforeresize" : true,
26207
26208         "beforeapply" : true
26209     });
26210
26211     Roo.util.Observable.call(this);
26212 };
26213
26214 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26215     onStartProxyDrag : function(x, y){
26216         this.fireEvent("beforeresize", this);
26217         if(!this.overlay){
26218             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26219             o.unselectable();
26220             o.enableDisplayMode("block");
26221             // all splitbars share the same overlay
26222             Roo.SplitBar.prototype.overlay = o;
26223         }
26224         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26225         this.overlay.show();
26226         Roo.get(this.proxy).setDisplayed("block");
26227         var size = this.adapter.getElementSize(this);
26228         this.activeMinSize = this.getMinimumSize();;
26229         this.activeMaxSize = this.getMaximumSize();;
26230         var c1 = size - this.activeMinSize;
26231         var c2 = Math.max(this.activeMaxSize - size, 0);
26232         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26233             this.dd.resetConstraints();
26234             this.dd.setXConstraint(
26235                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26236                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26237             );
26238             this.dd.setYConstraint(0, 0);
26239         }else{
26240             this.dd.resetConstraints();
26241             this.dd.setXConstraint(0, 0);
26242             this.dd.setYConstraint(
26243                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26244                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26245             );
26246          }
26247         this.dragSpecs.startSize = size;
26248         this.dragSpecs.startPoint = [x, y];
26249         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26250     },
26251     
26252     /** 
26253      * @private Called after the drag operation by the DDProxy
26254      */
26255     onEndProxyDrag : function(e){
26256         Roo.get(this.proxy).setDisplayed(false);
26257         var endPoint = Roo.lib.Event.getXY(e);
26258         if(this.overlay){
26259             this.overlay.hide();
26260         }
26261         var newSize;
26262         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26263             newSize = this.dragSpecs.startSize + 
26264                 (this.placement == Roo.SplitBar.LEFT ?
26265                     endPoint[0] - this.dragSpecs.startPoint[0] :
26266                     this.dragSpecs.startPoint[0] - endPoint[0]
26267                 );
26268         }else{
26269             newSize = this.dragSpecs.startSize + 
26270                 (this.placement == Roo.SplitBar.TOP ?
26271                     endPoint[1] - this.dragSpecs.startPoint[1] :
26272                     this.dragSpecs.startPoint[1] - endPoint[1]
26273                 );
26274         }
26275         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26276         if(newSize != this.dragSpecs.startSize){
26277             if(this.fireEvent('beforeapply', this, newSize) !== false){
26278                 this.adapter.setElementSize(this, newSize);
26279                 this.fireEvent("moved", this, newSize);
26280                 this.fireEvent("resize", this, newSize);
26281             }
26282         }
26283     },
26284     
26285     /**
26286      * Get the adapter this SplitBar uses
26287      * @return The adapter object
26288      */
26289     getAdapter : function(){
26290         return this.adapter;
26291     },
26292     
26293     /**
26294      * Set the adapter this SplitBar uses
26295      * @param {Object} adapter A SplitBar adapter object
26296      */
26297     setAdapter : function(adapter){
26298         this.adapter = adapter;
26299         this.adapter.init(this);
26300     },
26301     
26302     /**
26303      * Gets the minimum size for the resizing element
26304      * @return {Number} The minimum size
26305      */
26306     getMinimumSize : function(){
26307         return this.minSize;
26308     },
26309     
26310     /**
26311      * Sets the minimum size for the resizing element
26312      * @param {Number} minSize The minimum size
26313      */
26314     setMinimumSize : function(minSize){
26315         this.minSize = minSize;
26316     },
26317     
26318     /**
26319      * Gets the maximum size for the resizing element
26320      * @return {Number} The maximum size
26321      */
26322     getMaximumSize : function(){
26323         return this.maxSize;
26324     },
26325     
26326     /**
26327      * Sets the maximum size for the resizing element
26328      * @param {Number} maxSize The maximum size
26329      */
26330     setMaximumSize : function(maxSize){
26331         this.maxSize = maxSize;
26332     },
26333     
26334     /**
26335      * Sets the initialize size for the resizing element
26336      * @param {Number} size The initial size
26337      */
26338     setCurrentSize : function(size){
26339         var oldAnimate = this.animate;
26340         this.animate = false;
26341         this.adapter.setElementSize(this, size);
26342         this.animate = oldAnimate;
26343     },
26344     
26345     /**
26346      * Destroy this splitbar. 
26347      * @param {Boolean} removeEl True to remove the element
26348      */
26349     destroy : function(removeEl){
26350         if(this.shim){
26351             this.shim.remove();
26352         }
26353         this.dd.unreg();
26354         this.proxy.parentNode.removeChild(this.proxy);
26355         if(removeEl){
26356             this.el.remove();
26357         }
26358     }
26359 });
26360
26361 /**
26362  * @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.
26363  */
26364 Roo.SplitBar.createProxy = function(dir){
26365     var proxy = new Roo.Element(document.createElement("div"));
26366     proxy.unselectable();
26367     var cls = 'x-splitbar-proxy';
26368     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26369     document.body.appendChild(proxy.dom);
26370     return proxy.dom;
26371 };
26372
26373 /** 
26374  * @class Roo.SplitBar.BasicLayoutAdapter
26375  * Default Adapter. It assumes the splitter and resizing element are not positioned
26376  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26377  */
26378 Roo.SplitBar.BasicLayoutAdapter = function(){
26379 };
26380
26381 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26382     // do nothing for now
26383     init : function(s){
26384     
26385     },
26386     /**
26387      * Called before drag operations to get the current size of the resizing element. 
26388      * @param {Roo.SplitBar} s The SplitBar using this adapter
26389      */
26390      getElementSize : function(s){
26391         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26392             return s.resizingEl.getWidth();
26393         }else{
26394             return s.resizingEl.getHeight();
26395         }
26396     },
26397     
26398     /**
26399      * Called after drag operations to set the size of the resizing element.
26400      * @param {Roo.SplitBar} s The SplitBar using this adapter
26401      * @param {Number} newSize The new size to set
26402      * @param {Function} onComplete A function to be invoked when resizing is complete
26403      */
26404     setElementSize : function(s, newSize, onComplete){
26405         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26406             if(!s.animate){
26407                 s.resizingEl.setWidth(newSize);
26408                 if(onComplete){
26409                     onComplete(s, newSize);
26410                 }
26411             }else{
26412                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26413             }
26414         }else{
26415             
26416             if(!s.animate){
26417                 s.resizingEl.setHeight(newSize);
26418                 if(onComplete){
26419                     onComplete(s, newSize);
26420                 }
26421             }else{
26422                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26423             }
26424         }
26425     }
26426 };
26427
26428 /** 
26429  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26430  * @extends Roo.SplitBar.BasicLayoutAdapter
26431  * Adapter that  moves the splitter element to align with the resized sizing element. 
26432  * Used with an absolute positioned SplitBar.
26433  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26434  * document.body, make sure you assign an id to the body element.
26435  */
26436 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26437     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26438     this.container = Roo.get(container);
26439 };
26440
26441 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26442     init : function(s){
26443         this.basic.init(s);
26444     },
26445     
26446     getElementSize : function(s){
26447         return this.basic.getElementSize(s);
26448     },
26449     
26450     setElementSize : function(s, newSize, onComplete){
26451         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26452     },
26453     
26454     moveSplitter : function(s){
26455         var yes = Roo.SplitBar;
26456         switch(s.placement){
26457             case yes.LEFT:
26458                 s.el.setX(s.resizingEl.getRight());
26459                 break;
26460             case yes.RIGHT:
26461                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26462                 break;
26463             case yes.TOP:
26464                 s.el.setY(s.resizingEl.getBottom());
26465                 break;
26466             case yes.BOTTOM:
26467                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26468                 break;
26469         }
26470     }
26471 };
26472
26473 /**
26474  * Orientation constant - Create a vertical SplitBar
26475  * @static
26476  * @type Number
26477  */
26478 Roo.SplitBar.VERTICAL = 1;
26479
26480 /**
26481  * Orientation constant - Create a horizontal SplitBar
26482  * @static
26483  * @type Number
26484  */
26485 Roo.SplitBar.HORIZONTAL = 2;
26486
26487 /**
26488  * Placement constant - The resizing element is to the left of the splitter element
26489  * @static
26490  * @type Number
26491  */
26492 Roo.SplitBar.LEFT = 1;
26493
26494 /**
26495  * Placement constant - The resizing element is to the right of the splitter element
26496  * @static
26497  * @type Number
26498  */
26499 Roo.SplitBar.RIGHT = 2;
26500
26501 /**
26502  * Placement constant - The resizing element is positioned above the splitter element
26503  * @static
26504  * @type Number
26505  */
26506 Roo.SplitBar.TOP = 3;
26507
26508 /**
26509  * Placement constant - The resizing element is positioned under splitter element
26510  * @static
26511  * @type Number
26512  */
26513 Roo.SplitBar.BOTTOM = 4;
26514 /*
26515  * Based on:
26516  * Ext JS Library 1.1.1
26517  * Copyright(c) 2006-2007, Ext JS, LLC.
26518  *
26519  * Originally Released Under LGPL - original licence link has changed is not relivant.
26520  *
26521  * Fork - LGPL
26522  * <script type="text/javascript">
26523  */
26524
26525 /**
26526  * @class Roo.View
26527  * @extends Roo.util.Observable
26528  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26529  * This class also supports single and multi selection modes. <br>
26530  * Create a data model bound view:
26531  <pre><code>
26532  var store = new Roo.data.Store(...);
26533
26534  var view = new Roo.View({
26535     el : "my-element",
26536     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26537  
26538     singleSelect: true,
26539     selectedClass: "ydataview-selected",
26540     store: store
26541  });
26542
26543  // listen for node click?
26544  view.on("click", function(vw, index, node, e){
26545  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26546  });
26547
26548  // load XML data
26549  dataModel.load("foobar.xml");
26550  </code></pre>
26551  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26552  * <br><br>
26553  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26554  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26555  * 
26556  * Note: old style constructor is still suported (container, template, config)
26557  * 
26558  * @constructor
26559  * Create a new View
26560  * @param {Object} config The config object
26561  * 
26562  */
26563 Roo.View = function(config, depreciated_tpl, depreciated_config){
26564     
26565     this.parent = false;
26566     
26567     if (typeof(depreciated_tpl) == 'undefined') {
26568         // new way.. - universal constructor.
26569         Roo.apply(this, config);
26570         this.el  = Roo.get(this.el);
26571     } else {
26572         // old format..
26573         this.el  = Roo.get(config);
26574         this.tpl = depreciated_tpl;
26575         Roo.apply(this, depreciated_config);
26576     }
26577     this.wrapEl  = this.el.wrap().wrap();
26578     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26579     
26580     
26581     if(typeof(this.tpl) == "string"){
26582         this.tpl = new Roo.Template(this.tpl);
26583     } else {
26584         // support xtype ctors..
26585         this.tpl = new Roo.factory(this.tpl, Roo);
26586     }
26587     
26588     
26589     this.tpl.compile();
26590     
26591     /** @private */
26592     this.addEvents({
26593         /**
26594          * @event beforeclick
26595          * Fires before a click is processed. Returns false to cancel the default action.
26596          * @param {Roo.View} this
26597          * @param {Number} index The index of the target node
26598          * @param {HTMLElement} node The target node
26599          * @param {Roo.EventObject} e The raw event object
26600          */
26601             "beforeclick" : true,
26602         /**
26603          * @event click
26604          * Fires when a template node is clicked.
26605          * @param {Roo.View} this
26606          * @param {Number} index The index of the target node
26607          * @param {HTMLElement} node The target node
26608          * @param {Roo.EventObject} e The raw event object
26609          */
26610             "click" : true,
26611         /**
26612          * @event dblclick
26613          * Fires when a template node is double clicked.
26614          * @param {Roo.View} this
26615          * @param {Number} index The index of the target node
26616          * @param {HTMLElement} node The target node
26617          * @param {Roo.EventObject} e The raw event object
26618          */
26619             "dblclick" : true,
26620         /**
26621          * @event contextmenu
26622          * Fires when a template node is right clicked.
26623          * @param {Roo.View} this
26624          * @param {Number} index The index of the target node
26625          * @param {HTMLElement} node The target node
26626          * @param {Roo.EventObject} e The raw event object
26627          */
26628             "contextmenu" : true,
26629         /**
26630          * @event selectionchange
26631          * Fires when the selected nodes change.
26632          * @param {Roo.View} this
26633          * @param {Array} selections Array of the selected nodes
26634          */
26635             "selectionchange" : true,
26636     
26637         /**
26638          * @event beforeselect
26639          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26640          * @param {Roo.View} this
26641          * @param {HTMLElement} node The node to be selected
26642          * @param {Array} selections Array of currently selected nodes
26643          */
26644             "beforeselect" : true,
26645         /**
26646          * @event preparedata
26647          * Fires on every row to render, to allow you to change the data.
26648          * @param {Roo.View} this
26649          * @param {Object} data to be rendered (change this)
26650          */
26651           "preparedata" : true
26652           
26653           
26654         });
26655
26656
26657
26658     this.el.on({
26659         "click": this.onClick,
26660         "dblclick": this.onDblClick,
26661         "contextmenu": this.onContextMenu,
26662         scope:this
26663     });
26664
26665     this.selections = [];
26666     this.nodes = [];
26667     this.cmp = new Roo.CompositeElementLite([]);
26668     if(this.store){
26669         this.store = Roo.factory(this.store, Roo.data);
26670         this.setStore(this.store, true);
26671     }
26672     
26673     if ( this.footer && this.footer.xtype) {
26674            
26675          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26676         
26677         this.footer.dataSource = this.store;
26678         this.footer.container = fctr;
26679         this.footer = Roo.factory(this.footer, Roo);
26680         fctr.insertFirst(this.el);
26681         
26682         // this is a bit insane - as the paging toolbar seems to detach the el..
26683 //        dom.parentNode.parentNode.parentNode
26684          // they get detached?
26685     }
26686     
26687     
26688     Roo.View.superclass.constructor.call(this);
26689     
26690     
26691 };
26692
26693 Roo.extend(Roo.View, Roo.util.Observable, {
26694     
26695      /**
26696      * @cfg {Roo.data.Store} store Data store to load data from.
26697      */
26698     store : false,
26699     
26700     /**
26701      * @cfg {String|Roo.Element} el The container element.
26702      */
26703     el : '',
26704     
26705     /**
26706      * @cfg {String|Roo.Template} tpl The template used by this View 
26707      */
26708     tpl : false,
26709     /**
26710      * @cfg {String} dataName the named area of the template to use as the data area
26711      *                          Works with domtemplates roo-name="name"
26712      */
26713     dataName: false,
26714     /**
26715      * @cfg {String} selectedClass The css class to add to selected nodes
26716      */
26717     selectedClass : "x-view-selected",
26718      /**
26719      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26720      */
26721     emptyText : "",
26722     
26723     /**
26724      * @cfg {String} text to display on mask (default Loading)
26725      */
26726     mask : false,
26727     /**
26728      * @cfg {Boolean} multiSelect Allow multiple selection
26729      */
26730     multiSelect : false,
26731     /**
26732      * @cfg {Boolean} singleSelect Allow single selection
26733      */
26734     singleSelect:  false,
26735     
26736     /**
26737      * @cfg {Boolean} toggleSelect - selecting 
26738      */
26739     toggleSelect : false,
26740     
26741     /**
26742      * @cfg {Boolean} tickable - selecting 
26743      */
26744     tickable : false,
26745     
26746     /**
26747      * Returns the element this view is bound to.
26748      * @return {Roo.Element}
26749      */
26750     getEl : function(){
26751         return this.wrapEl;
26752     },
26753     
26754     
26755
26756     /**
26757      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26758      */
26759     refresh : function(){
26760         //Roo.log('refresh');
26761         var t = this.tpl;
26762         
26763         // if we are using something like 'domtemplate', then
26764         // the what gets used is:
26765         // t.applySubtemplate(NAME, data, wrapping data..)
26766         // the outer template then get' applied with
26767         //     the store 'extra data'
26768         // and the body get's added to the
26769         //      roo-name="data" node?
26770         //      <span class='roo-tpl-{name}'></span> ?????
26771         
26772         
26773         
26774         this.clearSelections();
26775         this.el.update("");
26776         var html = [];
26777         var records = this.store.getRange();
26778         if(records.length < 1) {
26779             
26780             // is this valid??  = should it render a template??
26781             
26782             this.el.update(this.emptyText);
26783             return;
26784         }
26785         var el = this.el;
26786         if (this.dataName) {
26787             this.el.update(t.apply(this.store.meta)); //????
26788             el = this.el.child('.roo-tpl-' + this.dataName);
26789         }
26790         
26791         for(var i = 0, len = records.length; i < len; i++){
26792             var data = this.prepareData(records[i].data, i, records[i]);
26793             this.fireEvent("preparedata", this, data, i, records[i]);
26794             
26795             var d = Roo.apply({}, data);
26796             
26797             if(this.tickable){
26798                 Roo.apply(d, {'roo-id' : Roo.id()});
26799                 
26800                 var _this = this;
26801             
26802                 Roo.each(this.parent.item, function(item){
26803                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26804                         return;
26805                     }
26806                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26807                 });
26808             }
26809             
26810             html[html.length] = Roo.util.Format.trim(
26811                 this.dataName ?
26812                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26813                     t.apply(d)
26814             );
26815         }
26816         
26817         
26818         
26819         el.update(html.join(""));
26820         this.nodes = el.dom.childNodes;
26821         this.updateIndexes(0);
26822     },
26823     
26824
26825     /**
26826      * Function to override to reformat the data that is sent to
26827      * the template for each node.
26828      * DEPRICATED - use the preparedata event handler.
26829      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26830      * a JSON object for an UpdateManager bound view).
26831      */
26832     prepareData : function(data, index, record)
26833     {
26834         this.fireEvent("preparedata", this, data, index, record);
26835         return data;
26836     },
26837
26838     onUpdate : function(ds, record){
26839         // Roo.log('on update');   
26840         this.clearSelections();
26841         var index = this.store.indexOf(record);
26842         var n = this.nodes[index];
26843         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26844         n.parentNode.removeChild(n);
26845         this.updateIndexes(index, index);
26846     },
26847
26848     
26849     
26850 // --------- FIXME     
26851     onAdd : function(ds, records, index)
26852     {
26853         //Roo.log(['on Add', ds, records, index] );        
26854         this.clearSelections();
26855         if(this.nodes.length == 0){
26856             this.refresh();
26857             return;
26858         }
26859         var n = this.nodes[index];
26860         for(var i = 0, len = records.length; i < len; i++){
26861             var d = this.prepareData(records[i].data, i, records[i]);
26862             if(n){
26863                 this.tpl.insertBefore(n, d);
26864             }else{
26865                 
26866                 this.tpl.append(this.el, d);
26867             }
26868         }
26869         this.updateIndexes(index);
26870     },
26871
26872     onRemove : function(ds, record, index){
26873        // Roo.log('onRemove');
26874         this.clearSelections();
26875         var el = this.dataName  ?
26876             this.el.child('.roo-tpl-' + this.dataName) :
26877             this.el; 
26878         
26879         el.dom.removeChild(this.nodes[index]);
26880         this.updateIndexes(index);
26881     },
26882
26883     /**
26884      * Refresh an individual node.
26885      * @param {Number} index
26886      */
26887     refreshNode : function(index){
26888         this.onUpdate(this.store, this.store.getAt(index));
26889     },
26890
26891     updateIndexes : function(startIndex, endIndex){
26892         var ns = this.nodes;
26893         startIndex = startIndex || 0;
26894         endIndex = endIndex || ns.length - 1;
26895         for(var i = startIndex; i <= endIndex; i++){
26896             ns[i].nodeIndex = i;
26897         }
26898     },
26899
26900     /**
26901      * Changes the data store this view uses and refresh the view.
26902      * @param {Store} store
26903      */
26904     setStore : function(store, initial){
26905         if(!initial && this.store){
26906             this.store.un("datachanged", this.refresh);
26907             this.store.un("add", this.onAdd);
26908             this.store.un("remove", this.onRemove);
26909             this.store.un("update", this.onUpdate);
26910             this.store.un("clear", this.refresh);
26911             this.store.un("beforeload", this.onBeforeLoad);
26912             this.store.un("load", this.onLoad);
26913             this.store.un("loadexception", this.onLoad);
26914         }
26915         if(store){
26916           
26917             store.on("datachanged", this.refresh, this);
26918             store.on("add", this.onAdd, this);
26919             store.on("remove", this.onRemove, this);
26920             store.on("update", this.onUpdate, this);
26921             store.on("clear", this.refresh, this);
26922             store.on("beforeload", this.onBeforeLoad, this);
26923             store.on("load", this.onLoad, this);
26924             store.on("loadexception", this.onLoad, this);
26925         }
26926         
26927         if(store){
26928             this.refresh();
26929         }
26930     },
26931     /**
26932      * onbeforeLoad - masks the loading area.
26933      *
26934      */
26935     onBeforeLoad : function(store,opts)
26936     {
26937          //Roo.log('onBeforeLoad');   
26938         if (!opts.add) {
26939             this.el.update("");
26940         }
26941         this.el.mask(this.mask ? this.mask : "Loading" ); 
26942     },
26943     onLoad : function ()
26944     {
26945         this.el.unmask();
26946     },
26947     
26948
26949     /**
26950      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26951      * @param {HTMLElement} node
26952      * @return {HTMLElement} The template node
26953      */
26954     findItemFromChild : function(node){
26955         var el = this.dataName  ?
26956             this.el.child('.roo-tpl-' + this.dataName,true) :
26957             this.el.dom; 
26958         
26959         if(!node || node.parentNode == el){
26960                     return node;
26961             }
26962             var p = node.parentNode;
26963             while(p && p != el){
26964             if(p.parentNode == el){
26965                 return p;
26966             }
26967             p = p.parentNode;
26968         }
26969             return null;
26970     },
26971
26972     /** @ignore */
26973     onClick : function(e){
26974         var item = this.findItemFromChild(e.getTarget());
26975         if(item){
26976             var index = this.indexOf(item);
26977             if(this.onItemClick(item, index, e) !== false){
26978                 this.fireEvent("click", this, index, item, e);
26979             }
26980         }else{
26981             this.clearSelections();
26982         }
26983     },
26984
26985     /** @ignore */
26986     onContextMenu : function(e){
26987         var item = this.findItemFromChild(e.getTarget());
26988         if(item){
26989             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
26990         }
26991     },
26992
26993     /** @ignore */
26994     onDblClick : function(e){
26995         var item = this.findItemFromChild(e.getTarget());
26996         if(item){
26997             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
26998         }
26999     },
27000
27001     onItemClick : function(item, index, e)
27002     {
27003         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27004             return false;
27005         }
27006         if (this.toggleSelect) {
27007             var m = this.isSelected(item) ? 'unselect' : 'select';
27008             //Roo.log(m);
27009             var _t = this;
27010             _t[m](item, true, false);
27011             return true;
27012         }
27013         if(this.multiSelect || this.singleSelect){
27014             if(this.multiSelect && e.shiftKey && this.lastSelection){
27015                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27016             }else{
27017                 this.select(item, this.multiSelect && e.ctrlKey);
27018                 this.lastSelection = item;
27019             }
27020             
27021             if(!this.tickable){
27022                 e.preventDefault();
27023             }
27024             
27025         }
27026         return true;
27027     },
27028
27029     /**
27030      * Get the number of selected nodes.
27031      * @return {Number}
27032      */
27033     getSelectionCount : function(){
27034         return this.selections.length;
27035     },
27036
27037     /**
27038      * Get the currently selected nodes.
27039      * @return {Array} An array of HTMLElements
27040      */
27041     getSelectedNodes : function(){
27042         return this.selections;
27043     },
27044
27045     /**
27046      * Get the indexes of the selected nodes.
27047      * @return {Array}
27048      */
27049     getSelectedIndexes : function(){
27050         var indexes = [], s = this.selections;
27051         for(var i = 0, len = s.length; i < len; i++){
27052             indexes.push(s[i].nodeIndex);
27053         }
27054         return indexes;
27055     },
27056
27057     /**
27058      * Clear all selections
27059      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27060      */
27061     clearSelections : function(suppressEvent){
27062         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27063             this.cmp.elements = this.selections;
27064             this.cmp.removeClass(this.selectedClass);
27065             this.selections = [];
27066             if(!suppressEvent){
27067                 this.fireEvent("selectionchange", this, this.selections);
27068             }
27069         }
27070     },
27071
27072     /**
27073      * Returns true if the passed node is selected
27074      * @param {HTMLElement/Number} node The node or node index
27075      * @return {Boolean}
27076      */
27077     isSelected : function(node){
27078         var s = this.selections;
27079         if(s.length < 1){
27080             return false;
27081         }
27082         node = this.getNode(node);
27083         return s.indexOf(node) !== -1;
27084     },
27085
27086     /**
27087      * Selects nodes.
27088      * @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
27089      * @param {Boolean} keepExisting (optional) true to keep existing selections
27090      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27091      */
27092     select : function(nodeInfo, keepExisting, suppressEvent){
27093         if(nodeInfo instanceof Array){
27094             if(!keepExisting){
27095                 this.clearSelections(true);
27096             }
27097             for(var i = 0, len = nodeInfo.length; i < len; i++){
27098                 this.select(nodeInfo[i], true, true);
27099             }
27100             return;
27101         } 
27102         var node = this.getNode(nodeInfo);
27103         if(!node || this.isSelected(node)){
27104             return; // already selected.
27105         }
27106         if(!keepExisting){
27107             this.clearSelections(true);
27108         }
27109         
27110         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27111             Roo.fly(node).addClass(this.selectedClass);
27112             this.selections.push(node);
27113             if(!suppressEvent){
27114                 this.fireEvent("selectionchange", this, this.selections);
27115             }
27116         }
27117         
27118         
27119     },
27120       /**
27121      * Unselects nodes.
27122      * @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
27123      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27124      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27125      */
27126     unselect : function(nodeInfo, keepExisting, suppressEvent)
27127     {
27128         if(nodeInfo instanceof Array){
27129             Roo.each(this.selections, function(s) {
27130                 this.unselect(s, nodeInfo);
27131             }, this);
27132             return;
27133         }
27134         var node = this.getNode(nodeInfo);
27135         if(!node || !this.isSelected(node)){
27136             //Roo.log("not selected");
27137             return; // not selected.
27138         }
27139         // fireevent???
27140         var ns = [];
27141         Roo.each(this.selections, function(s) {
27142             if (s == node ) {
27143                 Roo.fly(node).removeClass(this.selectedClass);
27144
27145                 return;
27146             }
27147             ns.push(s);
27148         },this);
27149         
27150         this.selections= ns;
27151         this.fireEvent("selectionchange", this, this.selections);
27152     },
27153
27154     /**
27155      * Gets a template node.
27156      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27157      * @return {HTMLElement} The node or null if it wasn't found
27158      */
27159     getNode : function(nodeInfo){
27160         if(typeof nodeInfo == "string"){
27161             return document.getElementById(nodeInfo);
27162         }else if(typeof nodeInfo == "number"){
27163             return this.nodes[nodeInfo];
27164         }
27165         return nodeInfo;
27166     },
27167
27168     /**
27169      * Gets a range template nodes.
27170      * @param {Number} startIndex
27171      * @param {Number} endIndex
27172      * @return {Array} An array of nodes
27173      */
27174     getNodes : function(start, end){
27175         var ns = this.nodes;
27176         start = start || 0;
27177         end = typeof end == "undefined" ? ns.length - 1 : end;
27178         var nodes = [];
27179         if(start <= end){
27180             for(var i = start; i <= end; i++){
27181                 nodes.push(ns[i]);
27182             }
27183         } else{
27184             for(var i = start; i >= end; i--){
27185                 nodes.push(ns[i]);
27186             }
27187         }
27188         return nodes;
27189     },
27190
27191     /**
27192      * Finds the index of the passed node
27193      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27194      * @return {Number} The index of the node or -1
27195      */
27196     indexOf : function(node){
27197         node = this.getNode(node);
27198         if(typeof node.nodeIndex == "number"){
27199             return node.nodeIndex;
27200         }
27201         var ns = this.nodes;
27202         for(var i = 0, len = ns.length; i < len; i++){
27203             if(ns[i] == node){
27204                 return i;
27205             }
27206         }
27207         return -1;
27208     }
27209 });
27210 /*
27211  * Based on:
27212  * Ext JS Library 1.1.1
27213  * Copyright(c) 2006-2007, Ext JS, LLC.
27214  *
27215  * Originally Released Under LGPL - original licence link has changed is not relivant.
27216  *
27217  * Fork - LGPL
27218  * <script type="text/javascript">
27219  */
27220
27221 /**
27222  * @class Roo.JsonView
27223  * @extends Roo.View
27224  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27225 <pre><code>
27226 var view = new Roo.JsonView({
27227     container: "my-element",
27228     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27229     multiSelect: true, 
27230     jsonRoot: "data" 
27231 });
27232
27233 // listen for node click?
27234 view.on("click", function(vw, index, node, e){
27235     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27236 });
27237
27238 // direct load of JSON data
27239 view.load("foobar.php");
27240
27241 // Example from my blog list
27242 var tpl = new Roo.Template(
27243     '&lt;div class="entry"&gt;' +
27244     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27245     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27246     "&lt;/div&gt;&lt;hr /&gt;"
27247 );
27248
27249 var moreView = new Roo.JsonView({
27250     container :  "entry-list", 
27251     template : tpl,
27252     jsonRoot: "posts"
27253 });
27254 moreView.on("beforerender", this.sortEntries, this);
27255 moreView.load({
27256     url: "/blog/get-posts.php",
27257     params: "allposts=true",
27258     text: "Loading Blog Entries..."
27259 });
27260 </code></pre>
27261
27262 * Note: old code is supported with arguments : (container, template, config)
27263
27264
27265  * @constructor
27266  * Create a new JsonView
27267  * 
27268  * @param {Object} config The config object
27269  * 
27270  */
27271 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27272     
27273     
27274     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27275
27276     var um = this.el.getUpdateManager();
27277     um.setRenderer(this);
27278     um.on("update", this.onLoad, this);
27279     um.on("failure", this.onLoadException, this);
27280
27281     /**
27282      * @event beforerender
27283      * Fires before rendering of the downloaded JSON data.
27284      * @param {Roo.JsonView} this
27285      * @param {Object} data The JSON data loaded
27286      */
27287     /**
27288      * @event load
27289      * Fires when data is loaded.
27290      * @param {Roo.JsonView} this
27291      * @param {Object} data The JSON data loaded
27292      * @param {Object} response The raw Connect response object
27293      */
27294     /**
27295      * @event loadexception
27296      * Fires when loading fails.
27297      * @param {Roo.JsonView} this
27298      * @param {Object} response The raw Connect response object
27299      */
27300     this.addEvents({
27301         'beforerender' : true,
27302         'load' : true,
27303         'loadexception' : true
27304     });
27305 };
27306 Roo.extend(Roo.JsonView, Roo.View, {
27307     /**
27308      * @type {String} The root property in the loaded JSON object that contains the data
27309      */
27310     jsonRoot : "",
27311
27312     /**
27313      * Refreshes the view.
27314      */
27315     refresh : function(){
27316         this.clearSelections();
27317         this.el.update("");
27318         var html = [];
27319         var o = this.jsonData;
27320         if(o && o.length > 0){
27321             for(var i = 0, len = o.length; i < len; i++){
27322                 var data = this.prepareData(o[i], i, o);
27323                 html[html.length] = this.tpl.apply(data);
27324             }
27325         }else{
27326             html.push(this.emptyText);
27327         }
27328         this.el.update(html.join(""));
27329         this.nodes = this.el.dom.childNodes;
27330         this.updateIndexes(0);
27331     },
27332
27333     /**
27334      * 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.
27335      * @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:
27336      <pre><code>
27337      view.load({
27338          url: "your-url.php",
27339          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27340          callback: yourFunction,
27341          scope: yourObject, //(optional scope)
27342          discardUrl: false,
27343          nocache: false,
27344          text: "Loading...",
27345          timeout: 30,
27346          scripts: false
27347      });
27348      </code></pre>
27349      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27350      * 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.
27351      * @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}
27352      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27353      * @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.
27354      */
27355     load : function(){
27356         var um = this.el.getUpdateManager();
27357         um.update.apply(um, arguments);
27358     },
27359
27360     // note - render is a standard framework call...
27361     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27362     render : function(el, response){
27363         
27364         this.clearSelections();
27365         this.el.update("");
27366         var o;
27367         try{
27368             if (response != '') {
27369                 o = Roo.util.JSON.decode(response.responseText);
27370                 if(this.jsonRoot){
27371                     
27372                     o = o[this.jsonRoot];
27373                 }
27374             }
27375         } catch(e){
27376         }
27377         /**
27378          * The current JSON data or null
27379          */
27380         this.jsonData = o;
27381         this.beforeRender();
27382         this.refresh();
27383     },
27384
27385 /**
27386  * Get the number of records in the current JSON dataset
27387  * @return {Number}
27388  */
27389     getCount : function(){
27390         return this.jsonData ? this.jsonData.length : 0;
27391     },
27392
27393 /**
27394  * Returns the JSON object for the specified node(s)
27395  * @param {HTMLElement/Array} node The node or an array of nodes
27396  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27397  * you get the JSON object for the node
27398  */
27399     getNodeData : function(node){
27400         if(node instanceof Array){
27401             var data = [];
27402             for(var i = 0, len = node.length; i < len; i++){
27403                 data.push(this.getNodeData(node[i]));
27404             }
27405             return data;
27406         }
27407         return this.jsonData[this.indexOf(node)] || null;
27408     },
27409
27410     beforeRender : function(){
27411         this.snapshot = this.jsonData;
27412         if(this.sortInfo){
27413             this.sort.apply(this, this.sortInfo);
27414         }
27415         this.fireEvent("beforerender", this, this.jsonData);
27416     },
27417
27418     onLoad : function(el, o){
27419         this.fireEvent("load", this, this.jsonData, o);
27420     },
27421
27422     onLoadException : function(el, o){
27423         this.fireEvent("loadexception", this, o);
27424     },
27425
27426 /**
27427  * Filter the data by a specific property.
27428  * @param {String} property A property on your JSON objects
27429  * @param {String/RegExp} value Either string that the property values
27430  * should start with, or a RegExp to test against the property
27431  */
27432     filter : function(property, value){
27433         if(this.jsonData){
27434             var data = [];
27435             var ss = this.snapshot;
27436             if(typeof value == "string"){
27437                 var vlen = value.length;
27438                 if(vlen == 0){
27439                     this.clearFilter();
27440                     return;
27441                 }
27442                 value = value.toLowerCase();
27443                 for(var i = 0, len = ss.length; i < len; i++){
27444                     var o = ss[i];
27445                     if(o[property].substr(0, vlen).toLowerCase() == value){
27446                         data.push(o);
27447                     }
27448                 }
27449             } else if(value.exec){ // regex?
27450                 for(var i = 0, len = ss.length; i < len; i++){
27451                     var o = ss[i];
27452                     if(value.test(o[property])){
27453                         data.push(o);
27454                     }
27455                 }
27456             } else{
27457                 return;
27458             }
27459             this.jsonData = data;
27460             this.refresh();
27461         }
27462     },
27463
27464 /**
27465  * Filter by a function. The passed function will be called with each
27466  * object in the current dataset. If the function returns true the value is kept,
27467  * otherwise it is filtered.
27468  * @param {Function} fn
27469  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27470  */
27471     filterBy : function(fn, scope){
27472         if(this.jsonData){
27473             var data = [];
27474             var ss = this.snapshot;
27475             for(var i = 0, len = ss.length; i < len; i++){
27476                 var o = ss[i];
27477                 if(fn.call(scope || this, o)){
27478                     data.push(o);
27479                 }
27480             }
27481             this.jsonData = data;
27482             this.refresh();
27483         }
27484     },
27485
27486 /**
27487  * Clears the current filter.
27488  */
27489     clearFilter : function(){
27490         if(this.snapshot && this.jsonData != this.snapshot){
27491             this.jsonData = this.snapshot;
27492             this.refresh();
27493         }
27494     },
27495
27496
27497 /**
27498  * Sorts the data for this view and refreshes it.
27499  * @param {String} property A property on your JSON objects to sort on
27500  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27501  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27502  */
27503     sort : function(property, dir, sortType){
27504         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27505         if(this.jsonData){
27506             var p = property;
27507             var dsc = dir && dir.toLowerCase() == "desc";
27508             var f = function(o1, o2){
27509                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27510                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27511                 ;
27512                 if(v1 < v2){
27513                     return dsc ? +1 : -1;
27514                 } else if(v1 > v2){
27515                     return dsc ? -1 : +1;
27516                 } else{
27517                     return 0;
27518                 }
27519             };
27520             this.jsonData.sort(f);
27521             this.refresh();
27522             if(this.jsonData != this.snapshot){
27523                 this.snapshot.sort(f);
27524             }
27525         }
27526     }
27527 });/*
27528  * Based on:
27529  * Ext JS Library 1.1.1
27530  * Copyright(c) 2006-2007, Ext JS, LLC.
27531  *
27532  * Originally Released Under LGPL - original licence link has changed is not relivant.
27533  *
27534  * Fork - LGPL
27535  * <script type="text/javascript">
27536  */
27537  
27538
27539 /**
27540  * @class Roo.ColorPalette
27541  * @extends Roo.Component
27542  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27543  * Here's an example of typical usage:
27544  * <pre><code>
27545 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27546 cp.render('my-div');
27547
27548 cp.on('select', function(palette, selColor){
27549     // do something with selColor
27550 });
27551 </code></pre>
27552  * @constructor
27553  * Create a new ColorPalette
27554  * @param {Object} config The config object
27555  */
27556 Roo.ColorPalette = function(config){
27557     Roo.ColorPalette.superclass.constructor.call(this, config);
27558     this.addEvents({
27559         /**
27560              * @event select
27561              * Fires when a color is selected
27562              * @param {ColorPalette} this
27563              * @param {String} color The 6-digit color hex code (without the # symbol)
27564              */
27565         select: true
27566     });
27567
27568     if(this.handler){
27569         this.on("select", this.handler, this.scope, true);
27570     }
27571 };
27572 Roo.extend(Roo.ColorPalette, Roo.Component, {
27573     /**
27574      * @cfg {String} itemCls
27575      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27576      */
27577     itemCls : "x-color-palette",
27578     /**
27579      * @cfg {String} value
27580      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27581      * the hex codes are case-sensitive.
27582      */
27583     value : null,
27584     clickEvent:'click',
27585     // private
27586     ctype: "Roo.ColorPalette",
27587
27588     /**
27589      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27590      */
27591     allowReselect : false,
27592
27593     /**
27594      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27595      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27596      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27597      * of colors with the width setting until the box is symmetrical.</p>
27598      * <p>You can override individual colors if needed:</p>
27599      * <pre><code>
27600 var cp = new Roo.ColorPalette();
27601 cp.colors[0] = "FF0000";  // change the first box to red
27602 </code></pre>
27603
27604 Or you can provide a custom array of your own for complete control:
27605 <pre><code>
27606 var cp = new Roo.ColorPalette();
27607 cp.colors = ["000000", "993300", "333300"];
27608 </code></pre>
27609      * @type Array
27610      */
27611     colors : [
27612         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27613         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27614         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27615         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27616         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27617     ],
27618
27619     // private
27620     onRender : function(container, position){
27621         var t = new Roo.MasterTemplate(
27622             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27623         );
27624         var c = this.colors;
27625         for(var i = 0, len = c.length; i < len; i++){
27626             t.add([c[i]]);
27627         }
27628         var el = document.createElement("div");
27629         el.className = this.itemCls;
27630         t.overwrite(el);
27631         container.dom.insertBefore(el, position);
27632         this.el = Roo.get(el);
27633         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27634         if(this.clickEvent != 'click'){
27635             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27636         }
27637     },
27638
27639     // private
27640     afterRender : function(){
27641         Roo.ColorPalette.superclass.afterRender.call(this);
27642         if(this.value){
27643             var s = this.value;
27644             this.value = null;
27645             this.select(s);
27646         }
27647     },
27648
27649     // private
27650     handleClick : function(e, t){
27651         e.preventDefault();
27652         if(!this.disabled){
27653             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27654             this.select(c.toUpperCase());
27655         }
27656     },
27657
27658     /**
27659      * Selects the specified color in the palette (fires the select event)
27660      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27661      */
27662     select : function(color){
27663         color = color.replace("#", "");
27664         if(color != this.value || this.allowReselect){
27665             var el = this.el;
27666             if(this.value){
27667                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27668             }
27669             el.child("a.color-"+color).addClass("x-color-palette-sel");
27670             this.value = color;
27671             this.fireEvent("select", this, color);
27672         }
27673     }
27674 });/*
27675  * Based on:
27676  * Ext JS Library 1.1.1
27677  * Copyright(c) 2006-2007, Ext JS, LLC.
27678  *
27679  * Originally Released Under LGPL - original licence link has changed is not relivant.
27680  *
27681  * Fork - LGPL
27682  * <script type="text/javascript">
27683  */
27684  
27685 /**
27686  * @class Roo.DatePicker
27687  * @extends Roo.Component
27688  * Simple date picker class.
27689  * @constructor
27690  * Create a new DatePicker
27691  * @param {Object} config The config object
27692  */
27693 Roo.DatePicker = function(config){
27694     Roo.DatePicker.superclass.constructor.call(this, config);
27695
27696     this.value = config && config.value ?
27697                  config.value.clearTime() : new Date().clearTime();
27698
27699     this.addEvents({
27700         /**
27701              * @event select
27702              * Fires when a date is selected
27703              * @param {DatePicker} this
27704              * @param {Date} date The selected date
27705              */
27706         'select': true,
27707         /**
27708              * @event monthchange
27709              * Fires when the displayed month changes 
27710              * @param {DatePicker} this
27711              * @param {Date} date The selected month
27712              */
27713         'monthchange': true
27714     });
27715
27716     if(this.handler){
27717         this.on("select", this.handler,  this.scope || this);
27718     }
27719     // build the disabledDatesRE
27720     if(!this.disabledDatesRE && this.disabledDates){
27721         var dd = this.disabledDates;
27722         var re = "(?:";
27723         for(var i = 0; i < dd.length; i++){
27724             re += dd[i];
27725             if(i != dd.length-1) {
27726                 re += "|";
27727             }
27728         }
27729         this.disabledDatesRE = new RegExp(re + ")");
27730     }
27731 };
27732
27733 Roo.extend(Roo.DatePicker, Roo.Component, {
27734     /**
27735      * @cfg {String} todayText
27736      * The text to display on the button that selects the current date (defaults to "Today")
27737      */
27738     todayText : "Today",
27739     /**
27740      * @cfg {String} okText
27741      * The text to display on the ok button
27742      */
27743     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27744     /**
27745      * @cfg {String} cancelText
27746      * The text to display on the cancel button
27747      */
27748     cancelText : "Cancel",
27749     /**
27750      * @cfg {String} todayTip
27751      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27752      */
27753     todayTip : "{0} (Spacebar)",
27754     /**
27755      * @cfg {Date} minDate
27756      * Minimum allowable date (JavaScript date object, defaults to null)
27757      */
27758     minDate : null,
27759     /**
27760      * @cfg {Date} maxDate
27761      * Maximum allowable date (JavaScript date object, defaults to null)
27762      */
27763     maxDate : null,
27764     /**
27765      * @cfg {String} minText
27766      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27767      */
27768     minText : "This date is before the minimum date",
27769     /**
27770      * @cfg {String} maxText
27771      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27772      */
27773     maxText : "This date is after the maximum date",
27774     /**
27775      * @cfg {String} format
27776      * The default date format string which can be overriden for localization support.  The format must be
27777      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27778      */
27779     format : "m/d/y",
27780     /**
27781      * @cfg {Array} disabledDays
27782      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27783      */
27784     disabledDays : null,
27785     /**
27786      * @cfg {String} disabledDaysText
27787      * The tooltip to display when the date falls on a disabled day (defaults to "")
27788      */
27789     disabledDaysText : "",
27790     /**
27791      * @cfg {RegExp} disabledDatesRE
27792      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27793      */
27794     disabledDatesRE : null,
27795     /**
27796      * @cfg {String} disabledDatesText
27797      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27798      */
27799     disabledDatesText : "",
27800     /**
27801      * @cfg {Boolean} constrainToViewport
27802      * True to constrain the date picker to the viewport (defaults to true)
27803      */
27804     constrainToViewport : true,
27805     /**
27806      * @cfg {Array} monthNames
27807      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27808      */
27809     monthNames : Date.monthNames,
27810     /**
27811      * @cfg {Array} dayNames
27812      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27813      */
27814     dayNames : Date.dayNames,
27815     /**
27816      * @cfg {String} nextText
27817      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27818      */
27819     nextText: 'Next Month (Control+Right)',
27820     /**
27821      * @cfg {String} prevText
27822      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27823      */
27824     prevText: 'Previous Month (Control+Left)',
27825     /**
27826      * @cfg {String} monthYearText
27827      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27828      */
27829     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27830     /**
27831      * @cfg {Number} startDay
27832      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27833      */
27834     startDay : 0,
27835     /**
27836      * @cfg {Bool} showClear
27837      * Show a clear button (usefull for date form elements that can be blank.)
27838      */
27839     
27840     showClear: false,
27841     
27842     /**
27843      * Sets the value of the date field
27844      * @param {Date} value The date to set
27845      */
27846     setValue : function(value){
27847         var old = this.value;
27848         
27849         if (typeof(value) == 'string') {
27850          
27851             value = Date.parseDate(value, this.format);
27852         }
27853         if (!value) {
27854             value = new Date();
27855         }
27856         
27857         this.value = value.clearTime(true);
27858         if(this.el){
27859             this.update(this.value);
27860         }
27861     },
27862
27863     /**
27864      * Gets the current selected value of the date field
27865      * @return {Date} The selected date
27866      */
27867     getValue : function(){
27868         return this.value;
27869     },
27870
27871     // private
27872     focus : function(){
27873         if(this.el){
27874             this.update(this.activeDate);
27875         }
27876     },
27877
27878     // privateval
27879     onRender : function(container, position){
27880         
27881         var m = [
27882              '<table cellspacing="0">',
27883                 '<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>',
27884                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27885         var dn = this.dayNames;
27886         for(var i = 0; i < 7; i++){
27887             var d = this.startDay+i;
27888             if(d > 6){
27889                 d = d-7;
27890             }
27891             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27892         }
27893         m[m.length] = "</tr></thead><tbody><tr>";
27894         for(var i = 0; i < 42; i++) {
27895             if(i % 7 == 0 && i != 0){
27896                 m[m.length] = "</tr><tr>";
27897             }
27898             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27899         }
27900         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27901             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27902
27903         var el = document.createElement("div");
27904         el.className = "x-date-picker";
27905         el.innerHTML = m.join("");
27906
27907         container.dom.insertBefore(el, position);
27908
27909         this.el = Roo.get(el);
27910         this.eventEl = Roo.get(el.firstChild);
27911
27912         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27913             handler: this.showPrevMonth,
27914             scope: this,
27915             preventDefault:true,
27916             stopDefault:true
27917         });
27918
27919         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27920             handler: this.showNextMonth,
27921             scope: this,
27922             preventDefault:true,
27923             stopDefault:true
27924         });
27925
27926         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27927
27928         this.monthPicker = this.el.down('div.x-date-mp');
27929         this.monthPicker.enableDisplayMode('block');
27930         
27931         var kn = new Roo.KeyNav(this.eventEl, {
27932             "left" : function(e){
27933                 e.ctrlKey ?
27934                     this.showPrevMonth() :
27935                     this.update(this.activeDate.add("d", -1));
27936             },
27937
27938             "right" : function(e){
27939                 e.ctrlKey ?
27940                     this.showNextMonth() :
27941                     this.update(this.activeDate.add("d", 1));
27942             },
27943
27944             "up" : function(e){
27945                 e.ctrlKey ?
27946                     this.showNextYear() :
27947                     this.update(this.activeDate.add("d", -7));
27948             },
27949
27950             "down" : function(e){
27951                 e.ctrlKey ?
27952                     this.showPrevYear() :
27953                     this.update(this.activeDate.add("d", 7));
27954             },
27955
27956             "pageUp" : function(e){
27957                 this.showNextMonth();
27958             },
27959
27960             "pageDown" : function(e){
27961                 this.showPrevMonth();
27962             },
27963
27964             "enter" : function(e){
27965                 e.stopPropagation();
27966                 return true;
27967             },
27968
27969             scope : this
27970         });
27971
27972         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
27973
27974         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
27975
27976         this.el.unselectable();
27977         
27978         this.cells = this.el.select("table.x-date-inner tbody td");
27979         this.textNodes = this.el.query("table.x-date-inner tbody span");
27980
27981         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
27982             text: "&#160;",
27983             tooltip: this.monthYearText
27984         });
27985
27986         this.mbtn.on('click', this.showMonthPicker, this);
27987         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
27988
27989
27990         var today = (new Date()).dateFormat(this.format);
27991         
27992         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
27993         if (this.showClear) {
27994             baseTb.add( new Roo.Toolbar.Fill());
27995         }
27996         baseTb.add({
27997             text: String.format(this.todayText, today),
27998             tooltip: String.format(this.todayTip, today),
27999             handler: this.selectToday,
28000             scope: this
28001         });
28002         
28003         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28004             
28005         //});
28006         if (this.showClear) {
28007             
28008             baseTb.add( new Roo.Toolbar.Fill());
28009             baseTb.add({
28010                 text: '&#160;',
28011                 cls: 'x-btn-icon x-btn-clear',
28012                 handler: function() {
28013                     //this.value = '';
28014                     this.fireEvent("select", this, '');
28015                 },
28016                 scope: this
28017             });
28018         }
28019         
28020         
28021         if(Roo.isIE){
28022             this.el.repaint();
28023         }
28024         this.update(this.value);
28025     },
28026
28027     createMonthPicker : function(){
28028         if(!this.monthPicker.dom.firstChild){
28029             var buf = ['<table border="0" cellspacing="0">'];
28030             for(var i = 0; i < 6; i++){
28031                 buf.push(
28032                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28033                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28034                     i == 0 ?
28035                     '<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>' :
28036                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28037                 );
28038             }
28039             buf.push(
28040                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28041                     this.okText,
28042                     '</button><button type="button" class="x-date-mp-cancel">',
28043                     this.cancelText,
28044                     '</button></td></tr>',
28045                 '</table>'
28046             );
28047             this.monthPicker.update(buf.join(''));
28048             this.monthPicker.on('click', this.onMonthClick, this);
28049             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28050
28051             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28052             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28053
28054             this.mpMonths.each(function(m, a, i){
28055                 i += 1;
28056                 if((i%2) == 0){
28057                     m.dom.xmonth = 5 + Math.round(i * .5);
28058                 }else{
28059                     m.dom.xmonth = Math.round((i-1) * .5);
28060                 }
28061             });
28062         }
28063     },
28064
28065     showMonthPicker : function(){
28066         this.createMonthPicker();
28067         var size = this.el.getSize();
28068         this.monthPicker.setSize(size);
28069         this.monthPicker.child('table').setSize(size);
28070
28071         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28072         this.updateMPMonth(this.mpSelMonth);
28073         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28074         this.updateMPYear(this.mpSelYear);
28075
28076         this.monthPicker.slideIn('t', {duration:.2});
28077     },
28078
28079     updateMPYear : function(y){
28080         this.mpyear = y;
28081         var ys = this.mpYears.elements;
28082         for(var i = 1; i <= 10; i++){
28083             var td = ys[i-1], y2;
28084             if((i%2) == 0){
28085                 y2 = y + Math.round(i * .5);
28086                 td.firstChild.innerHTML = y2;
28087                 td.xyear = y2;
28088             }else{
28089                 y2 = y - (5-Math.round(i * .5));
28090                 td.firstChild.innerHTML = y2;
28091                 td.xyear = y2;
28092             }
28093             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28094         }
28095     },
28096
28097     updateMPMonth : function(sm){
28098         this.mpMonths.each(function(m, a, i){
28099             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28100         });
28101     },
28102
28103     selectMPMonth: function(m){
28104         
28105     },
28106
28107     onMonthClick : function(e, t){
28108         e.stopEvent();
28109         var el = new Roo.Element(t), pn;
28110         if(el.is('button.x-date-mp-cancel')){
28111             this.hideMonthPicker();
28112         }
28113         else if(el.is('button.x-date-mp-ok')){
28114             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28115             this.hideMonthPicker();
28116         }
28117         else if(pn = el.up('td.x-date-mp-month', 2)){
28118             this.mpMonths.removeClass('x-date-mp-sel');
28119             pn.addClass('x-date-mp-sel');
28120             this.mpSelMonth = pn.dom.xmonth;
28121         }
28122         else if(pn = el.up('td.x-date-mp-year', 2)){
28123             this.mpYears.removeClass('x-date-mp-sel');
28124             pn.addClass('x-date-mp-sel');
28125             this.mpSelYear = pn.dom.xyear;
28126         }
28127         else if(el.is('a.x-date-mp-prev')){
28128             this.updateMPYear(this.mpyear-10);
28129         }
28130         else if(el.is('a.x-date-mp-next')){
28131             this.updateMPYear(this.mpyear+10);
28132         }
28133     },
28134
28135     onMonthDblClick : function(e, t){
28136         e.stopEvent();
28137         var el = new Roo.Element(t), pn;
28138         if(pn = el.up('td.x-date-mp-month', 2)){
28139             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28140             this.hideMonthPicker();
28141         }
28142         else if(pn = el.up('td.x-date-mp-year', 2)){
28143             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28144             this.hideMonthPicker();
28145         }
28146     },
28147
28148     hideMonthPicker : function(disableAnim){
28149         if(this.monthPicker){
28150             if(disableAnim === true){
28151                 this.monthPicker.hide();
28152             }else{
28153                 this.monthPicker.slideOut('t', {duration:.2});
28154             }
28155         }
28156     },
28157
28158     // private
28159     showPrevMonth : function(e){
28160         this.update(this.activeDate.add("mo", -1));
28161     },
28162
28163     // private
28164     showNextMonth : function(e){
28165         this.update(this.activeDate.add("mo", 1));
28166     },
28167
28168     // private
28169     showPrevYear : function(){
28170         this.update(this.activeDate.add("y", -1));
28171     },
28172
28173     // private
28174     showNextYear : function(){
28175         this.update(this.activeDate.add("y", 1));
28176     },
28177
28178     // private
28179     handleMouseWheel : function(e){
28180         var delta = e.getWheelDelta();
28181         if(delta > 0){
28182             this.showPrevMonth();
28183             e.stopEvent();
28184         } else if(delta < 0){
28185             this.showNextMonth();
28186             e.stopEvent();
28187         }
28188     },
28189
28190     // private
28191     handleDateClick : function(e, t){
28192         e.stopEvent();
28193         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28194             this.setValue(new Date(t.dateValue));
28195             this.fireEvent("select", this, this.value);
28196         }
28197     },
28198
28199     // private
28200     selectToday : function(){
28201         this.setValue(new Date().clearTime());
28202         this.fireEvent("select", this, this.value);
28203     },
28204
28205     // private
28206     update : function(date)
28207     {
28208         var vd = this.activeDate;
28209         this.activeDate = date;
28210         if(vd && this.el){
28211             var t = date.getTime();
28212             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28213                 this.cells.removeClass("x-date-selected");
28214                 this.cells.each(function(c){
28215                    if(c.dom.firstChild.dateValue == t){
28216                        c.addClass("x-date-selected");
28217                        setTimeout(function(){
28218                             try{c.dom.firstChild.focus();}catch(e){}
28219                        }, 50);
28220                        return false;
28221                    }
28222                 });
28223                 return;
28224             }
28225         }
28226         
28227         var days = date.getDaysInMonth();
28228         var firstOfMonth = date.getFirstDateOfMonth();
28229         var startingPos = firstOfMonth.getDay()-this.startDay;
28230
28231         if(startingPos <= this.startDay){
28232             startingPos += 7;
28233         }
28234
28235         var pm = date.add("mo", -1);
28236         var prevStart = pm.getDaysInMonth()-startingPos;
28237
28238         var cells = this.cells.elements;
28239         var textEls = this.textNodes;
28240         days += startingPos;
28241
28242         // convert everything to numbers so it's fast
28243         var day = 86400000;
28244         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28245         var today = new Date().clearTime().getTime();
28246         var sel = date.clearTime().getTime();
28247         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28248         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28249         var ddMatch = this.disabledDatesRE;
28250         var ddText = this.disabledDatesText;
28251         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28252         var ddaysText = this.disabledDaysText;
28253         var format = this.format;
28254
28255         var setCellClass = function(cal, cell){
28256             cell.title = "";
28257             var t = d.getTime();
28258             cell.firstChild.dateValue = t;
28259             if(t == today){
28260                 cell.className += " x-date-today";
28261                 cell.title = cal.todayText;
28262             }
28263             if(t == sel){
28264                 cell.className += " x-date-selected";
28265                 setTimeout(function(){
28266                     try{cell.firstChild.focus();}catch(e){}
28267                 }, 50);
28268             }
28269             // disabling
28270             if(t < min) {
28271                 cell.className = " x-date-disabled";
28272                 cell.title = cal.minText;
28273                 return;
28274             }
28275             if(t > max) {
28276                 cell.className = " x-date-disabled";
28277                 cell.title = cal.maxText;
28278                 return;
28279             }
28280             if(ddays){
28281                 if(ddays.indexOf(d.getDay()) != -1){
28282                     cell.title = ddaysText;
28283                     cell.className = " x-date-disabled";
28284                 }
28285             }
28286             if(ddMatch && format){
28287                 var fvalue = d.dateFormat(format);
28288                 if(ddMatch.test(fvalue)){
28289                     cell.title = ddText.replace("%0", fvalue);
28290                     cell.className = " x-date-disabled";
28291                 }
28292             }
28293         };
28294
28295         var i = 0;
28296         for(; i < startingPos; i++) {
28297             textEls[i].innerHTML = (++prevStart);
28298             d.setDate(d.getDate()+1);
28299             cells[i].className = "x-date-prevday";
28300             setCellClass(this, cells[i]);
28301         }
28302         for(; i < days; i++){
28303             intDay = i - startingPos + 1;
28304             textEls[i].innerHTML = (intDay);
28305             d.setDate(d.getDate()+1);
28306             cells[i].className = "x-date-active";
28307             setCellClass(this, cells[i]);
28308         }
28309         var extraDays = 0;
28310         for(; i < 42; i++) {
28311              textEls[i].innerHTML = (++extraDays);
28312              d.setDate(d.getDate()+1);
28313              cells[i].className = "x-date-nextday";
28314              setCellClass(this, cells[i]);
28315         }
28316
28317         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28318         this.fireEvent('monthchange', this, date);
28319         
28320         if(!this.internalRender){
28321             var main = this.el.dom.firstChild;
28322             var w = main.offsetWidth;
28323             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28324             Roo.fly(main).setWidth(w);
28325             this.internalRender = true;
28326             // opera does not respect the auto grow header center column
28327             // then, after it gets a width opera refuses to recalculate
28328             // without a second pass
28329             if(Roo.isOpera && !this.secondPass){
28330                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28331                 this.secondPass = true;
28332                 this.update.defer(10, this, [date]);
28333             }
28334         }
28335         
28336         
28337     }
28338 });        /*
28339  * Based on:
28340  * Ext JS Library 1.1.1
28341  * Copyright(c) 2006-2007, Ext JS, LLC.
28342  *
28343  * Originally Released Under LGPL - original licence link has changed is not relivant.
28344  *
28345  * Fork - LGPL
28346  * <script type="text/javascript">
28347  */
28348 /**
28349  * @class Roo.TabPanel
28350  * @extends Roo.util.Observable
28351  * A lightweight tab container.
28352  * <br><br>
28353  * Usage:
28354  * <pre><code>
28355 // basic tabs 1, built from existing content
28356 var tabs = new Roo.TabPanel("tabs1");
28357 tabs.addTab("script", "View Script");
28358 tabs.addTab("markup", "View Markup");
28359 tabs.activate("script");
28360
28361 // more advanced tabs, built from javascript
28362 var jtabs = new Roo.TabPanel("jtabs");
28363 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28364
28365 // set up the UpdateManager
28366 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28367 var updater = tab2.getUpdateManager();
28368 updater.setDefaultUrl("ajax1.htm");
28369 tab2.on('activate', updater.refresh, updater, true);
28370
28371 // Use setUrl for Ajax loading
28372 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28373 tab3.setUrl("ajax2.htm", null, true);
28374
28375 // Disabled tab
28376 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28377 tab4.disable();
28378
28379 jtabs.activate("jtabs-1");
28380  * </code></pre>
28381  * @constructor
28382  * Create a new TabPanel.
28383  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28384  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28385  */
28386 Roo.TabPanel = function(container, config){
28387     /**
28388     * The container element for this TabPanel.
28389     * @type Roo.Element
28390     */
28391     this.el = Roo.get(container, true);
28392     if(config){
28393         if(typeof config == "boolean"){
28394             this.tabPosition = config ? "bottom" : "top";
28395         }else{
28396             Roo.apply(this, config);
28397         }
28398     }
28399     if(this.tabPosition == "bottom"){
28400         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28401         this.el.addClass("x-tabs-bottom");
28402     }
28403     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28404     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28405     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28406     if(Roo.isIE){
28407         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28408     }
28409     if(this.tabPosition != "bottom"){
28410         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28411          * @type Roo.Element
28412          */
28413         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28414         this.el.addClass("x-tabs-top");
28415     }
28416     this.items = [];
28417
28418     this.bodyEl.setStyle("position", "relative");
28419
28420     this.active = null;
28421     this.activateDelegate = this.activate.createDelegate(this);
28422
28423     this.addEvents({
28424         /**
28425          * @event tabchange
28426          * Fires when the active tab changes
28427          * @param {Roo.TabPanel} this
28428          * @param {Roo.TabPanelItem} activePanel The new active tab
28429          */
28430         "tabchange": true,
28431         /**
28432          * @event beforetabchange
28433          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28434          * @param {Roo.TabPanel} this
28435          * @param {Object} e Set cancel to true on this object to cancel the tab change
28436          * @param {Roo.TabPanelItem} tab The tab being changed to
28437          */
28438         "beforetabchange" : true
28439     });
28440
28441     Roo.EventManager.onWindowResize(this.onResize, this);
28442     this.cpad = this.el.getPadding("lr");
28443     this.hiddenCount = 0;
28444
28445
28446     // toolbar on the tabbar support...
28447     if (this.toolbar) {
28448         var tcfg = this.toolbar;
28449         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28450         this.toolbar = new Roo.Toolbar(tcfg);
28451         if (Roo.isSafari) {
28452             var tbl = tcfg.container.child('table', true);
28453             tbl.setAttribute('width', '100%');
28454         }
28455         
28456     }
28457    
28458
28459
28460     Roo.TabPanel.superclass.constructor.call(this);
28461 };
28462
28463 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28464     /*
28465      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28466      */
28467     tabPosition : "top",
28468     /*
28469      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28470      */
28471     currentTabWidth : 0,
28472     /*
28473      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28474      */
28475     minTabWidth : 40,
28476     /*
28477      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28478      */
28479     maxTabWidth : 250,
28480     /*
28481      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28482      */
28483     preferredTabWidth : 175,
28484     /*
28485      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28486      */
28487     resizeTabs : false,
28488     /*
28489      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28490      */
28491     monitorResize : true,
28492     /*
28493      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28494      */
28495     toolbar : false,
28496
28497     /**
28498      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28499      * @param {String} id The id of the div to use <b>or create</b>
28500      * @param {String} text The text for the tab
28501      * @param {String} content (optional) Content to put in the TabPanelItem body
28502      * @param {Boolean} closable (optional) True to create a close icon on the tab
28503      * @return {Roo.TabPanelItem} The created TabPanelItem
28504      */
28505     addTab : function(id, text, content, closable){
28506         var item = new Roo.TabPanelItem(this, id, text, closable);
28507         this.addTabItem(item);
28508         if(content){
28509             item.setContent(content);
28510         }
28511         return item;
28512     },
28513
28514     /**
28515      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28516      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28517      * @return {Roo.TabPanelItem}
28518      */
28519     getTab : function(id){
28520         return this.items[id];
28521     },
28522
28523     /**
28524      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28525      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28526      */
28527     hideTab : function(id){
28528         var t = this.items[id];
28529         if(!t.isHidden()){
28530            t.setHidden(true);
28531            this.hiddenCount++;
28532            this.autoSizeTabs();
28533         }
28534     },
28535
28536     /**
28537      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28538      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28539      */
28540     unhideTab : function(id){
28541         var t = this.items[id];
28542         if(t.isHidden()){
28543            t.setHidden(false);
28544            this.hiddenCount--;
28545            this.autoSizeTabs();
28546         }
28547     },
28548
28549     /**
28550      * Adds an existing {@link Roo.TabPanelItem}.
28551      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28552      */
28553     addTabItem : function(item){
28554         this.items[item.id] = item;
28555         this.items.push(item);
28556         if(this.resizeTabs){
28557            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28558            this.autoSizeTabs();
28559         }else{
28560             item.autoSize();
28561         }
28562     },
28563
28564     /**
28565      * Removes a {@link Roo.TabPanelItem}.
28566      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28567      */
28568     removeTab : function(id){
28569         var items = this.items;
28570         var tab = items[id];
28571         if(!tab) { return; }
28572         var index = items.indexOf(tab);
28573         if(this.active == tab && items.length > 1){
28574             var newTab = this.getNextAvailable(index);
28575             if(newTab) {
28576                 newTab.activate();
28577             }
28578         }
28579         this.stripEl.dom.removeChild(tab.pnode.dom);
28580         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28581             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28582         }
28583         items.splice(index, 1);
28584         delete this.items[tab.id];
28585         tab.fireEvent("close", tab);
28586         tab.purgeListeners();
28587         this.autoSizeTabs();
28588     },
28589
28590     getNextAvailable : function(start){
28591         var items = this.items;
28592         var index = start;
28593         // look for a next tab that will slide over to
28594         // replace the one being removed
28595         while(index < items.length){
28596             var item = items[++index];
28597             if(item && !item.isHidden()){
28598                 return item;
28599             }
28600         }
28601         // if one isn't found select the previous tab (on the left)
28602         index = start;
28603         while(index >= 0){
28604             var item = items[--index];
28605             if(item && !item.isHidden()){
28606                 return item;
28607             }
28608         }
28609         return null;
28610     },
28611
28612     /**
28613      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28614      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28615      */
28616     disableTab : function(id){
28617         var tab = this.items[id];
28618         if(tab && this.active != tab){
28619             tab.disable();
28620         }
28621     },
28622
28623     /**
28624      * Enables a {@link Roo.TabPanelItem} that is disabled.
28625      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28626      */
28627     enableTab : function(id){
28628         var tab = this.items[id];
28629         tab.enable();
28630     },
28631
28632     /**
28633      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28634      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28635      * @return {Roo.TabPanelItem} The TabPanelItem.
28636      */
28637     activate : function(id){
28638         var tab = this.items[id];
28639         if(!tab){
28640             return null;
28641         }
28642         if(tab == this.active || tab.disabled){
28643             return tab;
28644         }
28645         var e = {};
28646         this.fireEvent("beforetabchange", this, e, tab);
28647         if(e.cancel !== true && !tab.disabled){
28648             if(this.active){
28649                 this.active.hide();
28650             }
28651             this.active = this.items[id];
28652             this.active.show();
28653             this.fireEvent("tabchange", this, this.active);
28654         }
28655         return tab;
28656     },
28657
28658     /**
28659      * Gets the active {@link Roo.TabPanelItem}.
28660      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28661      */
28662     getActiveTab : function(){
28663         return this.active;
28664     },
28665
28666     /**
28667      * Updates the tab body element to fit the height of the container element
28668      * for overflow scrolling
28669      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28670      */
28671     syncHeight : function(targetHeight){
28672         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28673         var bm = this.bodyEl.getMargins();
28674         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28675         this.bodyEl.setHeight(newHeight);
28676         return newHeight;
28677     },
28678
28679     onResize : function(){
28680         if(this.monitorResize){
28681             this.autoSizeTabs();
28682         }
28683     },
28684
28685     /**
28686      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28687      */
28688     beginUpdate : function(){
28689         this.updating = true;
28690     },
28691
28692     /**
28693      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28694      */
28695     endUpdate : function(){
28696         this.updating = false;
28697         this.autoSizeTabs();
28698     },
28699
28700     /**
28701      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28702      */
28703     autoSizeTabs : function(){
28704         var count = this.items.length;
28705         var vcount = count - this.hiddenCount;
28706         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28707             return;
28708         }
28709         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28710         var availWidth = Math.floor(w / vcount);
28711         var b = this.stripBody;
28712         if(b.getWidth() > w){
28713             var tabs = this.items;
28714             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28715             if(availWidth < this.minTabWidth){
28716                 /*if(!this.sleft){    // incomplete scrolling code
28717                     this.createScrollButtons();
28718                 }
28719                 this.showScroll();
28720                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28721             }
28722         }else{
28723             if(this.currentTabWidth < this.preferredTabWidth){
28724                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28725             }
28726         }
28727     },
28728
28729     /**
28730      * Returns the number of tabs in this TabPanel.
28731      * @return {Number}
28732      */
28733      getCount : function(){
28734          return this.items.length;
28735      },
28736
28737     /**
28738      * Resizes all the tabs to the passed width
28739      * @param {Number} The new width
28740      */
28741     setTabWidth : function(width){
28742         this.currentTabWidth = width;
28743         for(var i = 0, len = this.items.length; i < len; i++) {
28744                 if(!this.items[i].isHidden()) {
28745                 this.items[i].setWidth(width);
28746             }
28747         }
28748     },
28749
28750     /**
28751      * Destroys this TabPanel
28752      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28753      */
28754     destroy : function(removeEl){
28755         Roo.EventManager.removeResizeListener(this.onResize, this);
28756         for(var i = 0, len = this.items.length; i < len; i++){
28757             this.items[i].purgeListeners();
28758         }
28759         if(removeEl === true){
28760             this.el.update("");
28761             this.el.remove();
28762         }
28763     }
28764 });
28765
28766 /**
28767  * @class Roo.TabPanelItem
28768  * @extends Roo.util.Observable
28769  * Represents an individual item (tab plus body) in a TabPanel.
28770  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28771  * @param {String} id The id of this TabPanelItem
28772  * @param {String} text The text for the tab of this TabPanelItem
28773  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28774  */
28775 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28776     /**
28777      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28778      * @type Roo.TabPanel
28779      */
28780     this.tabPanel = tabPanel;
28781     /**
28782      * The id for this TabPanelItem
28783      * @type String
28784      */
28785     this.id = id;
28786     /** @private */
28787     this.disabled = false;
28788     /** @private */
28789     this.text = text;
28790     /** @private */
28791     this.loaded = false;
28792     this.closable = closable;
28793
28794     /**
28795      * The body element for this TabPanelItem.
28796      * @type Roo.Element
28797      */
28798     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28799     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28800     this.bodyEl.setStyle("display", "block");
28801     this.bodyEl.setStyle("zoom", "1");
28802     this.hideAction();
28803
28804     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28805     /** @private */
28806     this.el = Roo.get(els.el, true);
28807     this.inner = Roo.get(els.inner, true);
28808     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28809     this.pnode = Roo.get(els.el.parentNode, true);
28810     this.el.on("mousedown", this.onTabMouseDown, this);
28811     this.el.on("click", this.onTabClick, this);
28812     /** @private */
28813     if(closable){
28814         var c = Roo.get(els.close, true);
28815         c.dom.title = this.closeText;
28816         c.addClassOnOver("close-over");
28817         c.on("click", this.closeClick, this);
28818      }
28819
28820     this.addEvents({
28821          /**
28822          * @event activate
28823          * Fires when this tab becomes the active tab.
28824          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28825          * @param {Roo.TabPanelItem} this
28826          */
28827         "activate": true,
28828         /**
28829          * @event beforeclose
28830          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28831          * @param {Roo.TabPanelItem} this
28832          * @param {Object} e Set cancel to true on this object to cancel the close.
28833          */
28834         "beforeclose": true,
28835         /**
28836          * @event close
28837          * Fires when this tab is closed.
28838          * @param {Roo.TabPanelItem} this
28839          */
28840          "close": true,
28841         /**
28842          * @event deactivate
28843          * Fires when this tab is no longer the active tab.
28844          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28845          * @param {Roo.TabPanelItem} this
28846          */
28847          "deactivate" : true
28848     });
28849     this.hidden = false;
28850
28851     Roo.TabPanelItem.superclass.constructor.call(this);
28852 };
28853
28854 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28855     purgeListeners : function(){
28856        Roo.util.Observable.prototype.purgeListeners.call(this);
28857        this.el.removeAllListeners();
28858     },
28859     /**
28860      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28861      */
28862     show : function(){
28863         this.pnode.addClass("on");
28864         this.showAction();
28865         if(Roo.isOpera){
28866             this.tabPanel.stripWrap.repaint();
28867         }
28868         this.fireEvent("activate", this.tabPanel, this);
28869     },
28870
28871     /**
28872      * Returns true if this tab is the active tab.
28873      * @return {Boolean}
28874      */
28875     isActive : function(){
28876         return this.tabPanel.getActiveTab() == this;
28877     },
28878
28879     /**
28880      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28881      */
28882     hide : function(){
28883         this.pnode.removeClass("on");
28884         this.hideAction();
28885         this.fireEvent("deactivate", this.tabPanel, this);
28886     },
28887
28888     hideAction : function(){
28889         this.bodyEl.hide();
28890         this.bodyEl.setStyle("position", "absolute");
28891         this.bodyEl.setLeft("-20000px");
28892         this.bodyEl.setTop("-20000px");
28893     },
28894
28895     showAction : function(){
28896         this.bodyEl.setStyle("position", "relative");
28897         this.bodyEl.setTop("");
28898         this.bodyEl.setLeft("");
28899         this.bodyEl.show();
28900     },
28901
28902     /**
28903      * Set the tooltip for the tab.
28904      * @param {String} tooltip The tab's tooltip
28905      */
28906     setTooltip : function(text){
28907         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28908             this.textEl.dom.qtip = text;
28909             this.textEl.dom.removeAttribute('title');
28910         }else{
28911             this.textEl.dom.title = text;
28912         }
28913     },
28914
28915     onTabClick : function(e){
28916         e.preventDefault();
28917         this.tabPanel.activate(this.id);
28918     },
28919
28920     onTabMouseDown : function(e){
28921         e.preventDefault();
28922         this.tabPanel.activate(this.id);
28923     },
28924
28925     getWidth : function(){
28926         return this.inner.getWidth();
28927     },
28928
28929     setWidth : function(width){
28930         var iwidth = width - this.pnode.getPadding("lr");
28931         this.inner.setWidth(iwidth);
28932         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28933         this.pnode.setWidth(width);
28934     },
28935
28936     /**
28937      * Show or hide the tab
28938      * @param {Boolean} hidden True to hide or false to show.
28939      */
28940     setHidden : function(hidden){
28941         this.hidden = hidden;
28942         this.pnode.setStyle("display", hidden ? "none" : "");
28943     },
28944
28945     /**
28946      * Returns true if this tab is "hidden"
28947      * @return {Boolean}
28948      */
28949     isHidden : function(){
28950         return this.hidden;
28951     },
28952
28953     /**
28954      * Returns the text for this tab
28955      * @return {String}
28956      */
28957     getText : function(){
28958         return this.text;
28959     },
28960
28961     autoSize : function(){
28962         //this.el.beginMeasure();
28963         this.textEl.setWidth(1);
28964         /*
28965          *  #2804 [new] Tabs in Roojs
28966          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
28967          */
28968         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
28969         //this.el.endMeasure();
28970     },
28971
28972     /**
28973      * Sets the text for the tab (Note: this also sets the tooltip text)
28974      * @param {String} text The tab's text and tooltip
28975      */
28976     setText : function(text){
28977         this.text = text;
28978         this.textEl.update(text);
28979         this.setTooltip(text);
28980         if(!this.tabPanel.resizeTabs){
28981             this.autoSize();
28982         }
28983     },
28984     /**
28985      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
28986      */
28987     activate : function(){
28988         this.tabPanel.activate(this.id);
28989     },
28990
28991     /**
28992      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
28993      */
28994     disable : function(){
28995         if(this.tabPanel.active != this){
28996             this.disabled = true;
28997             this.pnode.addClass("disabled");
28998         }
28999     },
29000
29001     /**
29002      * Enables this TabPanelItem if it was previously disabled.
29003      */
29004     enable : function(){
29005         this.disabled = false;
29006         this.pnode.removeClass("disabled");
29007     },
29008
29009     /**
29010      * Sets the content for this TabPanelItem.
29011      * @param {String} content The content
29012      * @param {Boolean} loadScripts true to look for and load scripts
29013      */
29014     setContent : function(content, loadScripts){
29015         this.bodyEl.update(content, loadScripts);
29016     },
29017
29018     /**
29019      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29020      * @return {Roo.UpdateManager} The UpdateManager
29021      */
29022     getUpdateManager : function(){
29023         return this.bodyEl.getUpdateManager();
29024     },
29025
29026     /**
29027      * Set a URL to be used to load the content for this TabPanelItem.
29028      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29029      * @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)
29030      * @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)
29031      * @return {Roo.UpdateManager} The UpdateManager
29032      */
29033     setUrl : function(url, params, loadOnce){
29034         if(this.refreshDelegate){
29035             this.un('activate', this.refreshDelegate);
29036         }
29037         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29038         this.on("activate", this.refreshDelegate);
29039         return this.bodyEl.getUpdateManager();
29040     },
29041
29042     /** @private */
29043     _handleRefresh : function(url, params, loadOnce){
29044         if(!loadOnce || !this.loaded){
29045             var updater = this.bodyEl.getUpdateManager();
29046             updater.update(url, params, this._setLoaded.createDelegate(this));
29047         }
29048     },
29049
29050     /**
29051      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29052      *   Will fail silently if the setUrl method has not been called.
29053      *   This does not activate the panel, just updates its content.
29054      */
29055     refresh : function(){
29056         if(this.refreshDelegate){
29057            this.loaded = false;
29058            this.refreshDelegate();
29059         }
29060     },
29061
29062     /** @private */
29063     _setLoaded : function(){
29064         this.loaded = true;
29065     },
29066
29067     /** @private */
29068     closeClick : function(e){
29069         var o = {};
29070         e.stopEvent();
29071         this.fireEvent("beforeclose", this, o);
29072         if(o.cancel !== true){
29073             this.tabPanel.removeTab(this.id);
29074         }
29075     },
29076     /**
29077      * The text displayed in the tooltip for the close icon.
29078      * @type String
29079      */
29080     closeText : "Close this tab"
29081 });
29082
29083 /** @private */
29084 Roo.TabPanel.prototype.createStrip = function(container){
29085     var strip = document.createElement("div");
29086     strip.className = "x-tabs-wrap";
29087     container.appendChild(strip);
29088     return strip;
29089 };
29090 /** @private */
29091 Roo.TabPanel.prototype.createStripList = function(strip){
29092     // div wrapper for retard IE
29093     // returns the "tr" element.
29094     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29095         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29096         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29097     return strip.firstChild.firstChild.firstChild.firstChild;
29098 };
29099 /** @private */
29100 Roo.TabPanel.prototype.createBody = function(container){
29101     var body = document.createElement("div");
29102     Roo.id(body, "tab-body");
29103     Roo.fly(body).addClass("x-tabs-body");
29104     container.appendChild(body);
29105     return body;
29106 };
29107 /** @private */
29108 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29109     var body = Roo.getDom(id);
29110     if(!body){
29111         body = document.createElement("div");
29112         body.id = id;
29113     }
29114     Roo.fly(body).addClass("x-tabs-item-body");
29115     bodyEl.insertBefore(body, bodyEl.firstChild);
29116     return body;
29117 };
29118 /** @private */
29119 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29120     var td = document.createElement("td");
29121     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29122     //stripEl.appendChild(td);
29123     if(closable){
29124         td.className = "x-tabs-closable";
29125         if(!this.closeTpl){
29126             this.closeTpl = new Roo.Template(
29127                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29128                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29129                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29130             );
29131         }
29132         var el = this.closeTpl.overwrite(td, {"text": text});
29133         var close = el.getElementsByTagName("div")[0];
29134         var inner = el.getElementsByTagName("em")[0];
29135         return {"el": el, "close": close, "inner": inner};
29136     } else {
29137         if(!this.tabTpl){
29138             this.tabTpl = new Roo.Template(
29139                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29140                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29141             );
29142         }
29143         var el = this.tabTpl.overwrite(td, {"text": text});
29144         var inner = el.getElementsByTagName("em")[0];
29145         return {"el": el, "inner": inner};
29146     }
29147 };/*
29148  * Based on:
29149  * Ext JS Library 1.1.1
29150  * Copyright(c) 2006-2007, Ext JS, LLC.
29151  *
29152  * Originally Released Under LGPL - original licence link has changed is not relivant.
29153  *
29154  * Fork - LGPL
29155  * <script type="text/javascript">
29156  */
29157
29158 /**
29159  * @class Roo.Button
29160  * @extends Roo.util.Observable
29161  * Simple Button class
29162  * @cfg {String} text The button text
29163  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29164  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29165  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29166  * @cfg {Object} scope The scope of the handler
29167  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29168  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29169  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29170  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29171  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29172  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29173    applies if enableToggle = true)
29174  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29175  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29176   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29177  * @constructor
29178  * Create a new button
29179  * @param {Object} config The config object
29180  */
29181 Roo.Button = function(renderTo, config)
29182 {
29183     if (!config) {
29184         config = renderTo;
29185         renderTo = config.renderTo || false;
29186     }
29187     
29188     Roo.apply(this, config);
29189     this.addEvents({
29190         /**
29191              * @event click
29192              * Fires when this button is clicked
29193              * @param {Button} this
29194              * @param {EventObject} e The click event
29195              */
29196             "click" : true,
29197         /**
29198              * @event toggle
29199              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29200              * @param {Button} this
29201              * @param {Boolean} pressed
29202              */
29203             "toggle" : true,
29204         /**
29205              * @event mouseover
29206              * Fires when the mouse hovers over the button
29207              * @param {Button} this
29208              * @param {Event} e The event object
29209              */
29210         'mouseover' : true,
29211         /**
29212              * @event mouseout
29213              * Fires when the mouse exits the button
29214              * @param {Button} this
29215              * @param {Event} e The event object
29216              */
29217         'mouseout': true,
29218          /**
29219              * @event render
29220              * Fires when the button is rendered
29221              * @param {Button} this
29222              */
29223         'render': true
29224     });
29225     if(this.menu){
29226         this.menu = Roo.menu.MenuMgr.get(this.menu);
29227     }
29228     // register listeners first!!  - so render can be captured..
29229     Roo.util.Observable.call(this);
29230     if(renderTo){
29231         this.render(renderTo);
29232     }
29233     
29234   
29235 };
29236
29237 Roo.extend(Roo.Button, Roo.util.Observable, {
29238     /**
29239      * 
29240      */
29241     
29242     /**
29243      * Read-only. True if this button is hidden
29244      * @type Boolean
29245      */
29246     hidden : false,
29247     /**
29248      * Read-only. True if this button is disabled
29249      * @type Boolean
29250      */
29251     disabled : false,
29252     /**
29253      * Read-only. True if this button is pressed (only if enableToggle = true)
29254      * @type Boolean
29255      */
29256     pressed : false,
29257
29258     /**
29259      * @cfg {Number} tabIndex 
29260      * The DOM tabIndex for this button (defaults to undefined)
29261      */
29262     tabIndex : undefined,
29263
29264     /**
29265      * @cfg {Boolean} enableToggle
29266      * True to enable pressed/not pressed toggling (defaults to false)
29267      */
29268     enableToggle: false,
29269     /**
29270      * @cfg {Mixed} menu
29271      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29272      */
29273     menu : undefined,
29274     /**
29275      * @cfg {String} menuAlign
29276      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29277      */
29278     menuAlign : "tl-bl?",
29279
29280     /**
29281      * @cfg {String} iconCls
29282      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29283      */
29284     iconCls : undefined,
29285     /**
29286      * @cfg {String} type
29287      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29288      */
29289     type : 'button',
29290
29291     // private
29292     menuClassTarget: 'tr',
29293
29294     /**
29295      * @cfg {String} clickEvent
29296      * The type of event to map to the button's event handler (defaults to 'click')
29297      */
29298     clickEvent : 'click',
29299
29300     /**
29301      * @cfg {Boolean} handleMouseEvents
29302      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29303      */
29304     handleMouseEvents : true,
29305
29306     /**
29307      * @cfg {String} tooltipType
29308      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29309      */
29310     tooltipType : 'qtip',
29311
29312     /**
29313      * @cfg {String} cls
29314      * A CSS class to apply to the button's main element.
29315      */
29316     
29317     /**
29318      * @cfg {Roo.Template} template (Optional)
29319      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29320      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29321      * require code modifications if required elements (e.g. a button) aren't present.
29322      */
29323
29324     // private
29325     render : function(renderTo){
29326         var btn;
29327         if(this.hideParent){
29328             this.parentEl = Roo.get(renderTo);
29329         }
29330         if(!this.dhconfig){
29331             if(!this.template){
29332                 if(!Roo.Button.buttonTemplate){
29333                     // hideous table template
29334                     Roo.Button.buttonTemplate = new Roo.Template(
29335                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29336                         '<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>',
29337                         "</tr></tbody></table>");
29338                 }
29339                 this.template = Roo.Button.buttonTemplate;
29340             }
29341             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29342             var btnEl = btn.child("button:first");
29343             btnEl.on('focus', this.onFocus, this);
29344             btnEl.on('blur', this.onBlur, this);
29345             if(this.cls){
29346                 btn.addClass(this.cls);
29347             }
29348             if(this.icon){
29349                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29350             }
29351             if(this.iconCls){
29352                 btnEl.addClass(this.iconCls);
29353                 if(!this.cls){
29354                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29355                 }
29356             }
29357             if(this.tabIndex !== undefined){
29358                 btnEl.dom.tabIndex = this.tabIndex;
29359             }
29360             if(this.tooltip){
29361                 if(typeof this.tooltip == 'object'){
29362                     Roo.QuickTips.tips(Roo.apply({
29363                           target: btnEl.id
29364                     }, this.tooltip));
29365                 } else {
29366                     btnEl.dom[this.tooltipType] = this.tooltip;
29367                 }
29368             }
29369         }else{
29370             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29371         }
29372         this.el = btn;
29373         if(this.id){
29374             this.el.dom.id = this.el.id = this.id;
29375         }
29376         if(this.menu){
29377             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29378             this.menu.on("show", this.onMenuShow, this);
29379             this.menu.on("hide", this.onMenuHide, this);
29380         }
29381         btn.addClass("x-btn");
29382         if(Roo.isIE && !Roo.isIE7){
29383             this.autoWidth.defer(1, this);
29384         }else{
29385             this.autoWidth();
29386         }
29387         if(this.handleMouseEvents){
29388             btn.on("mouseover", this.onMouseOver, this);
29389             btn.on("mouseout", this.onMouseOut, this);
29390             btn.on("mousedown", this.onMouseDown, this);
29391         }
29392         btn.on(this.clickEvent, this.onClick, this);
29393         //btn.on("mouseup", this.onMouseUp, this);
29394         if(this.hidden){
29395             this.hide();
29396         }
29397         if(this.disabled){
29398             this.disable();
29399         }
29400         Roo.ButtonToggleMgr.register(this);
29401         if(this.pressed){
29402             this.el.addClass("x-btn-pressed");
29403         }
29404         if(this.repeat){
29405             var repeater = new Roo.util.ClickRepeater(btn,
29406                 typeof this.repeat == "object" ? this.repeat : {}
29407             );
29408             repeater.on("click", this.onClick,  this);
29409         }
29410         
29411         this.fireEvent('render', this);
29412         
29413     },
29414     /**
29415      * Returns the button's underlying element
29416      * @return {Roo.Element} The element
29417      */
29418     getEl : function(){
29419         return this.el;  
29420     },
29421     
29422     /**
29423      * Destroys this Button and removes any listeners.
29424      */
29425     destroy : function(){
29426         Roo.ButtonToggleMgr.unregister(this);
29427         this.el.removeAllListeners();
29428         this.purgeListeners();
29429         this.el.remove();
29430     },
29431
29432     // private
29433     autoWidth : function(){
29434         if(this.el){
29435             this.el.setWidth("auto");
29436             if(Roo.isIE7 && Roo.isStrict){
29437                 var ib = this.el.child('button');
29438                 if(ib && ib.getWidth() > 20){
29439                     ib.clip();
29440                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29441                 }
29442             }
29443             if(this.minWidth){
29444                 if(this.hidden){
29445                     this.el.beginMeasure();
29446                 }
29447                 if(this.el.getWidth() < this.minWidth){
29448                     this.el.setWidth(this.minWidth);
29449                 }
29450                 if(this.hidden){
29451                     this.el.endMeasure();
29452                 }
29453             }
29454         }
29455     },
29456
29457     /**
29458      * Assigns this button's click handler
29459      * @param {Function} handler The function to call when the button is clicked
29460      * @param {Object} scope (optional) Scope for the function passed in
29461      */
29462     setHandler : function(handler, scope){
29463         this.handler = handler;
29464         this.scope = scope;  
29465     },
29466     
29467     /**
29468      * Sets this button's text
29469      * @param {String} text The button text
29470      */
29471     setText : function(text){
29472         this.text = text;
29473         if(this.el){
29474             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29475         }
29476         this.autoWidth();
29477     },
29478     
29479     /**
29480      * Gets the text for this button
29481      * @return {String} The button text
29482      */
29483     getText : function(){
29484         return this.text;  
29485     },
29486     
29487     /**
29488      * Show this button
29489      */
29490     show: function(){
29491         this.hidden = false;
29492         if(this.el){
29493             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29494         }
29495     },
29496     
29497     /**
29498      * Hide this button
29499      */
29500     hide: function(){
29501         this.hidden = true;
29502         if(this.el){
29503             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29504         }
29505     },
29506     
29507     /**
29508      * Convenience function for boolean show/hide
29509      * @param {Boolean} visible True to show, false to hide
29510      */
29511     setVisible: function(visible){
29512         if(visible) {
29513             this.show();
29514         }else{
29515             this.hide();
29516         }
29517     },
29518     
29519     /**
29520      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29521      * @param {Boolean} state (optional) Force a particular state
29522      */
29523     toggle : function(state){
29524         state = state === undefined ? !this.pressed : state;
29525         if(state != this.pressed){
29526             if(state){
29527                 this.el.addClass("x-btn-pressed");
29528                 this.pressed = true;
29529                 this.fireEvent("toggle", this, true);
29530             }else{
29531                 this.el.removeClass("x-btn-pressed");
29532                 this.pressed = false;
29533                 this.fireEvent("toggle", this, false);
29534             }
29535             if(this.toggleHandler){
29536                 this.toggleHandler.call(this.scope || this, this, state);
29537             }
29538         }
29539     },
29540     
29541     /**
29542      * Focus the button
29543      */
29544     focus : function(){
29545         this.el.child('button:first').focus();
29546     },
29547     
29548     /**
29549      * Disable this button
29550      */
29551     disable : function(){
29552         if(this.el){
29553             this.el.addClass("x-btn-disabled");
29554         }
29555         this.disabled = true;
29556     },
29557     
29558     /**
29559      * Enable this button
29560      */
29561     enable : function(){
29562         if(this.el){
29563             this.el.removeClass("x-btn-disabled");
29564         }
29565         this.disabled = false;
29566     },
29567
29568     /**
29569      * Convenience function for boolean enable/disable
29570      * @param {Boolean} enabled True to enable, false to disable
29571      */
29572     setDisabled : function(v){
29573         this[v !== true ? "enable" : "disable"]();
29574     },
29575
29576     // private
29577     onClick : function(e)
29578     {
29579         if(e){
29580             e.preventDefault();
29581         }
29582         if(e.button != 0){
29583             return;
29584         }
29585         if(!this.disabled){
29586             if(this.enableToggle){
29587                 this.toggle();
29588             }
29589             if(this.menu && !this.menu.isVisible()){
29590                 this.menu.show(this.el, this.menuAlign);
29591             }
29592             this.fireEvent("click", this, e);
29593             if(this.handler){
29594                 this.el.removeClass("x-btn-over");
29595                 this.handler.call(this.scope || this, this, e);
29596             }
29597         }
29598     },
29599     // private
29600     onMouseOver : function(e){
29601         if(!this.disabled){
29602             this.el.addClass("x-btn-over");
29603             this.fireEvent('mouseover', this, e);
29604         }
29605     },
29606     // private
29607     onMouseOut : function(e){
29608         if(!e.within(this.el,  true)){
29609             this.el.removeClass("x-btn-over");
29610             this.fireEvent('mouseout', this, e);
29611         }
29612     },
29613     // private
29614     onFocus : function(e){
29615         if(!this.disabled){
29616             this.el.addClass("x-btn-focus");
29617         }
29618     },
29619     // private
29620     onBlur : function(e){
29621         this.el.removeClass("x-btn-focus");
29622     },
29623     // private
29624     onMouseDown : function(e){
29625         if(!this.disabled && e.button == 0){
29626             this.el.addClass("x-btn-click");
29627             Roo.get(document).on('mouseup', this.onMouseUp, this);
29628         }
29629     },
29630     // private
29631     onMouseUp : function(e){
29632         if(e.button == 0){
29633             this.el.removeClass("x-btn-click");
29634             Roo.get(document).un('mouseup', this.onMouseUp, this);
29635         }
29636     },
29637     // private
29638     onMenuShow : function(e){
29639         this.el.addClass("x-btn-menu-active");
29640     },
29641     // private
29642     onMenuHide : function(e){
29643         this.el.removeClass("x-btn-menu-active");
29644     }   
29645 });
29646
29647 // Private utility class used by Button
29648 Roo.ButtonToggleMgr = function(){
29649    var groups = {};
29650    
29651    function toggleGroup(btn, state){
29652        if(state){
29653            var g = groups[btn.toggleGroup];
29654            for(var i = 0, l = g.length; i < l; i++){
29655                if(g[i] != btn){
29656                    g[i].toggle(false);
29657                }
29658            }
29659        }
29660    }
29661    
29662    return {
29663        register : function(btn){
29664            if(!btn.toggleGroup){
29665                return;
29666            }
29667            var g = groups[btn.toggleGroup];
29668            if(!g){
29669                g = groups[btn.toggleGroup] = [];
29670            }
29671            g.push(btn);
29672            btn.on("toggle", toggleGroup);
29673        },
29674        
29675        unregister : function(btn){
29676            if(!btn.toggleGroup){
29677                return;
29678            }
29679            var g = groups[btn.toggleGroup];
29680            if(g){
29681                g.remove(btn);
29682                btn.un("toggle", toggleGroup);
29683            }
29684        }
29685    };
29686 }();/*
29687  * Based on:
29688  * Ext JS Library 1.1.1
29689  * Copyright(c) 2006-2007, Ext JS, LLC.
29690  *
29691  * Originally Released Under LGPL - original licence link has changed is not relivant.
29692  *
29693  * Fork - LGPL
29694  * <script type="text/javascript">
29695  */
29696  
29697 /**
29698  * @class Roo.SplitButton
29699  * @extends Roo.Button
29700  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29701  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29702  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29703  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29704  * @cfg {String} arrowTooltip The title attribute of the arrow
29705  * @constructor
29706  * Create a new menu button
29707  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29708  * @param {Object} config The config object
29709  */
29710 Roo.SplitButton = function(renderTo, config){
29711     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29712     /**
29713      * @event arrowclick
29714      * Fires when this button's arrow is clicked
29715      * @param {SplitButton} this
29716      * @param {EventObject} e The click event
29717      */
29718     this.addEvents({"arrowclick":true});
29719 };
29720
29721 Roo.extend(Roo.SplitButton, Roo.Button, {
29722     render : function(renderTo){
29723         // this is one sweet looking template!
29724         var tpl = new Roo.Template(
29725             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29726             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29727             '<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>',
29728             "</tbody></table></td><td>",
29729             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29730             '<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>',
29731             "</tbody></table></td></tr></table>"
29732         );
29733         var btn = tpl.append(renderTo, [this.text, this.type], true);
29734         var btnEl = btn.child("button");
29735         if(this.cls){
29736             btn.addClass(this.cls);
29737         }
29738         if(this.icon){
29739             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29740         }
29741         if(this.iconCls){
29742             btnEl.addClass(this.iconCls);
29743             if(!this.cls){
29744                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29745             }
29746         }
29747         this.el = btn;
29748         if(this.handleMouseEvents){
29749             btn.on("mouseover", this.onMouseOver, this);
29750             btn.on("mouseout", this.onMouseOut, this);
29751             btn.on("mousedown", this.onMouseDown, this);
29752             btn.on("mouseup", this.onMouseUp, this);
29753         }
29754         btn.on(this.clickEvent, this.onClick, this);
29755         if(this.tooltip){
29756             if(typeof this.tooltip == 'object'){
29757                 Roo.QuickTips.tips(Roo.apply({
29758                       target: btnEl.id
29759                 }, this.tooltip));
29760             } else {
29761                 btnEl.dom[this.tooltipType] = this.tooltip;
29762             }
29763         }
29764         if(this.arrowTooltip){
29765             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29766         }
29767         if(this.hidden){
29768             this.hide();
29769         }
29770         if(this.disabled){
29771             this.disable();
29772         }
29773         if(this.pressed){
29774             this.el.addClass("x-btn-pressed");
29775         }
29776         if(Roo.isIE && !Roo.isIE7){
29777             this.autoWidth.defer(1, this);
29778         }else{
29779             this.autoWidth();
29780         }
29781         if(this.menu){
29782             this.menu.on("show", this.onMenuShow, this);
29783             this.menu.on("hide", this.onMenuHide, this);
29784         }
29785         this.fireEvent('render', this);
29786     },
29787
29788     // private
29789     autoWidth : function(){
29790         if(this.el){
29791             var tbl = this.el.child("table:first");
29792             var tbl2 = this.el.child("table:last");
29793             this.el.setWidth("auto");
29794             tbl.setWidth("auto");
29795             if(Roo.isIE7 && Roo.isStrict){
29796                 var ib = this.el.child('button:first');
29797                 if(ib && ib.getWidth() > 20){
29798                     ib.clip();
29799                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29800                 }
29801             }
29802             if(this.minWidth){
29803                 if(this.hidden){
29804                     this.el.beginMeasure();
29805                 }
29806                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29807                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29808                 }
29809                 if(this.hidden){
29810                     this.el.endMeasure();
29811                 }
29812             }
29813             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29814         } 
29815     },
29816     /**
29817      * Sets this button's click handler
29818      * @param {Function} handler The function to call when the button is clicked
29819      * @param {Object} scope (optional) Scope for the function passed above
29820      */
29821     setHandler : function(handler, scope){
29822         this.handler = handler;
29823         this.scope = scope;  
29824     },
29825     
29826     /**
29827      * Sets this button's arrow click handler
29828      * @param {Function} handler The function to call when the arrow is clicked
29829      * @param {Object} scope (optional) Scope for the function passed above
29830      */
29831     setArrowHandler : function(handler, scope){
29832         this.arrowHandler = handler;
29833         this.scope = scope;  
29834     },
29835     
29836     /**
29837      * Focus the button
29838      */
29839     focus : function(){
29840         if(this.el){
29841             this.el.child("button:first").focus();
29842         }
29843     },
29844
29845     // private
29846     onClick : function(e){
29847         e.preventDefault();
29848         if(!this.disabled){
29849             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29850                 if(this.menu && !this.menu.isVisible()){
29851                     this.menu.show(this.el, this.menuAlign);
29852                 }
29853                 this.fireEvent("arrowclick", this, e);
29854                 if(this.arrowHandler){
29855                     this.arrowHandler.call(this.scope || this, this, e);
29856                 }
29857             }else{
29858                 this.fireEvent("click", this, e);
29859                 if(this.handler){
29860                     this.handler.call(this.scope || this, this, e);
29861                 }
29862             }
29863         }
29864     },
29865     // private
29866     onMouseDown : function(e){
29867         if(!this.disabled){
29868             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29869         }
29870     },
29871     // private
29872     onMouseUp : function(e){
29873         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29874     }   
29875 });
29876
29877
29878 // backwards compat
29879 Roo.MenuButton = Roo.SplitButton;/*
29880  * Based on:
29881  * Ext JS Library 1.1.1
29882  * Copyright(c) 2006-2007, Ext JS, LLC.
29883  *
29884  * Originally Released Under LGPL - original licence link has changed is not relivant.
29885  *
29886  * Fork - LGPL
29887  * <script type="text/javascript">
29888  */
29889
29890 /**
29891  * @class Roo.Toolbar
29892  * Basic Toolbar class.
29893  * @constructor
29894  * Creates a new Toolbar
29895  * @param {Object} container The config object
29896  */ 
29897 Roo.Toolbar = function(container, buttons, config)
29898 {
29899     /// old consturctor format still supported..
29900     if(container instanceof Array){ // omit the container for later rendering
29901         buttons = container;
29902         config = buttons;
29903         container = null;
29904     }
29905     if (typeof(container) == 'object' && container.xtype) {
29906         config = container;
29907         container = config.container;
29908         buttons = config.buttons || []; // not really - use items!!
29909     }
29910     var xitems = [];
29911     if (config && config.items) {
29912         xitems = config.items;
29913         delete config.items;
29914     }
29915     Roo.apply(this, config);
29916     this.buttons = buttons;
29917     
29918     if(container){
29919         this.render(container);
29920     }
29921     this.xitems = xitems;
29922     Roo.each(xitems, function(b) {
29923         this.add(b);
29924     }, this);
29925     
29926 };
29927
29928 Roo.Toolbar.prototype = {
29929     /**
29930      * @cfg {Array} items
29931      * array of button configs or elements to add (will be converted to a MixedCollection)
29932      */
29933     
29934     /**
29935      * @cfg {String/HTMLElement/Element} container
29936      * The id or element that will contain the toolbar
29937      */
29938     // private
29939     render : function(ct){
29940         this.el = Roo.get(ct);
29941         if(this.cls){
29942             this.el.addClass(this.cls);
29943         }
29944         // using a table allows for vertical alignment
29945         // 100% width is needed by Safari...
29946         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29947         this.tr = this.el.child("tr", true);
29948         var autoId = 0;
29949         this.items = new Roo.util.MixedCollection(false, function(o){
29950             return o.id || ("item" + (++autoId));
29951         });
29952         if(this.buttons){
29953             this.add.apply(this, this.buttons);
29954             delete this.buttons;
29955         }
29956     },
29957
29958     /**
29959      * Adds element(s) to the toolbar -- this function takes a variable number of 
29960      * arguments of mixed type and adds them to the toolbar.
29961      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
29962      * <ul>
29963      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
29964      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
29965      * <li>Field: Any form field (equivalent to {@link #addField})</li>
29966      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
29967      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
29968      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
29969      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
29970      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
29971      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
29972      * </ul>
29973      * @param {Mixed} arg2
29974      * @param {Mixed} etc.
29975      */
29976     add : function(){
29977         var a = arguments, l = a.length;
29978         for(var i = 0; i < l; i++){
29979             this._add(a[i]);
29980         }
29981     },
29982     // private..
29983     _add : function(el) {
29984         
29985         if (el.xtype) {
29986             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
29987         }
29988         
29989         if (el.applyTo){ // some kind of form field
29990             return this.addField(el);
29991         } 
29992         if (el.render){ // some kind of Toolbar.Item
29993             return this.addItem(el);
29994         }
29995         if (typeof el == "string"){ // string
29996             if(el == "separator" || el == "-"){
29997                 return this.addSeparator();
29998             }
29999             if (el == " "){
30000                 return this.addSpacer();
30001             }
30002             if(el == "->"){
30003                 return this.addFill();
30004             }
30005             return this.addText(el);
30006             
30007         }
30008         if(el.tagName){ // element
30009             return this.addElement(el);
30010         }
30011         if(typeof el == "object"){ // must be button config?
30012             return this.addButton(el);
30013         }
30014         // and now what?!?!
30015         return false;
30016         
30017     },
30018     
30019     /**
30020      * Add an Xtype element
30021      * @param {Object} xtype Xtype Object
30022      * @return {Object} created Object
30023      */
30024     addxtype : function(e){
30025         return this.add(e);  
30026     },
30027     
30028     /**
30029      * Returns the Element for this toolbar.
30030      * @return {Roo.Element}
30031      */
30032     getEl : function(){
30033         return this.el;  
30034     },
30035     
30036     /**
30037      * Adds a separator
30038      * @return {Roo.Toolbar.Item} The separator item
30039      */
30040     addSeparator : function(){
30041         return this.addItem(new Roo.Toolbar.Separator());
30042     },
30043
30044     /**
30045      * Adds a spacer element
30046      * @return {Roo.Toolbar.Spacer} The spacer item
30047      */
30048     addSpacer : function(){
30049         return this.addItem(new Roo.Toolbar.Spacer());
30050     },
30051
30052     /**
30053      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30054      * @return {Roo.Toolbar.Fill} The fill item
30055      */
30056     addFill : function(){
30057         return this.addItem(new Roo.Toolbar.Fill());
30058     },
30059
30060     /**
30061      * Adds any standard HTML element to the toolbar
30062      * @param {String/HTMLElement/Element} el The element or id of the element to add
30063      * @return {Roo.Toolbar.Item} The element's item
30064      */
30065     addElement : function(el){
30066         return this.addItem(new Roo.Toolbar.Item(el));
30067     },
30068     /**
30069      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30070      * @type Roo.util.MixedCollection  
30071      */
30072     items : false,
30073      
30074     /**
30075      * Adds any Toolbar.Item or subclass
30076      * @param {Roo.Toolbar.Item} item
30077      * @return {Roo.Toolbar.Item} The item
30078      */
30079     addItem : function(item){
30080         var td = this.nextBlock();
30081         item.render(td);
30082         this.items.add(item);
30083         return item;
30084     },
30085     
30086     /**
30087      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30088      * @param {Object/Array} config A button config or array of configs
30089      * @return {Roo.Toolbar.Button/Array}
30090      */
30091     addButton : function(config){
30092         if(config instanceof Array){
30093             var buttons = [];
30094             for(var i = 0, len = config.length; i < len; i++) {
30095                 buttons.push(this.addButton(config[i]));
30096             }
30097             return buttons;
30098         }
30099         var b = config;
30100         if(!(config instanceof Roo.Toolbar.Button)){
30101             b = config.split ?
30102                 new Roo.Toolbar.SplitButton(config) :
30103                 new Roo.Toolbar.Button(config);
30104         }
30105         var td = this.nextBlock();
30106         b.render(td);
30107         this.items.add(b);
30108         return b;
30109     },
30110     
30111     /**
30112      * Adds text to the toolbar
30113      * @param {String} text The text to add
30114      * @return {Roo.Toolbar.Item} The element's item
30115      */
30116     addText : function(text){
30117         return this.addItem(new Roo.Toolbar.TextItem(text));
30118     },
30119     
30120     /**
30121      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30122      * @param {Number} index The index where the item is to be inserted
30123      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30124      * @return {Roo.Toolbar.Button/Item}
30125      */
30126     insertButton : function(index, item){
30127         if(item instanceof Array){
30128             var buttons = [];
30129             for(var i = 0, len = item.length; i < len; i++) {
30130                buttons.push(this.insertButton(index + i, item[i]));
30131             }
30132             return buttons;
30133         }
30134         if (!(item instanceof Roo.Toolbar.Button)){
30135            item = new Roo.Toolbar.Button(item);
30136         }
30137         var td = document.createElement("td");
30138         this.tr.insertBefore(td, this.tr.childNodes[index]);
30139         item.render(td);
30140         this.items.insert(index, item);
30141         return item;
30142     },
30143     
30144     /**
30145      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30146      * @param {Object} config
30147      * @return {Roo.Toolbar.Item} The element's item
30148      */
30149     addDom : function(config, returnEl){
30150         var td = this.nextBlock();
30151         Roo.DomHelper.overwrite(td, config);
30152         var ti = new Roo.Toolbar.Item(td.firstChild);
30153         ti.render(td);
30154         this.items.add(ti);
30155         return ti;
30156     },
30157
30158     /**
30159      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30160      * @type Roo.util.MixedCollection  
30161      */
30162     fields : false,
30163     
30164     /**
30165      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30166      * Note: the field should not have been rendered yet. For a field that has already been
30167      * rendered, use {@link #addElement}.
30168      * @param {Roo.form.Field} field
30169      * @return {Roo.ToolbarItem}
30170      */
30171      
30172       
30173     addField : function(field) {
30174         if (!this.fields) {
30175             var autoId = 0;
30176             this.fields = new Roo.util.MixedCollection(false, function(o){
30177                 return o.id || ("item" + (++autoId));
30178             });
30179
30180         }
30181         
30182         var td = this.nextBlock();
30183         field.render(td);
30184         var ti = new Roo.Toolbar.Item(td.firstChild);
30185         ti.render(td);
30186         this.items.add(ti);
30187         this.fields.add(field);
30188         return ti;
30189     },
30190     /**
30191      * Hide the toolbar
30192      * @method hide
30193      */
30194      
30195       
30196     hide : function()
30197     {
30198         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30199         this.el.child('div').hide();
30200     },
30201     /**
30202      * Show the toolbar
30203      * @method show
30204      */
30205     show : function()
30206     {
30207         this.el.child('div').show();
30208     },
30209       
30210     // private
30211     nextBlock : function(){
30212         var td = document.createElement("td");
30213         this.tr.appendChild(td);
30214         return td;
30215     },
30216
30217     // private
30218     destroy : function(){
30219         if(this.items){ // rendered?
30220             Roo.destroy.apply(Roo, this.items.items);
30221         }
30222         if(this.fields){ // rendered?
30223             Roo.destroy.apply(Roo, this.fields.items);
30224         }
30225         Roo.Element.uncache(this.el, this.tr);
30226     }
30227 };
30228
30229 /**
30230  * @class Roo.Toolbar.Item
30231  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30232  * @constructor
30233  * Creates a new Item
30234  * @param {HTMLElement} el 
30235  */
30236 Roo.Toolbar.Item = function(el){
30237     var cfg = {};
30238     if (typeof (el.xtype) != 'undefined') {
30239         cfg = el;
30240         el = cfg.el;
30241     }
30242     
30243     this.el = Roo.getDom(el);
30244     this.id = Roo.id(this.el);
30245     this.hidden = false;
30246     
30247     this.addEvents({
30248          /**
30249              * @event render
30250              * Fires when the button is rendered
30251              * @param {Button} this
30252              */
30253         'render': true
30254     });
30255     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30256 };
30257 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30258 //Roo.Toolbar.Item.prototype = {
30259     
30260     /**
30261      * Get this item's HTML Element
30262      * @return {HTMLElement}
30263      */
30264     getEl : function(){
30265        return this.el;  
30266     },
30267
30268     // private
30269     render : function(td){
30270         
30271          this.td = td;
30272         td.appendChild(this.el);
30273         
30274         this.fireEvent('render', this);
30275     },
30276     
30277     /**
30278      * Removes and destroys this item.
30279      */
30280     destroy : function(){
30281         this.td.parentNode.removeChild(this.td);
30282     },
30283     
30284     /**
30285      * Shows this item.
30286      */
30287     show: function(){
30288         this.hidden = false;
30289         this.td.style.display = "";
30290     },
30291     
30292     /**
30293      * Hides this item.
30294      */
30295     hide: function(){
30296         this.hidden = true;
30297         this.td.style.display = "none";
30298     },
30299     
30300     /**
30301      * Convenience function for boolean show/hide.
30302      * @param {Boolean} visible true to show/false to hide
30303      */
30304     setVisible: function(visible){
30305         if(visible) {
30306             this.show();
30307         }else{
30308             this.hide();
30309         }
30310     },
30311     
30312     /**
30313      * Try to focus this item.
30314      */
30315     focus : function(){
30316         Roo.fly(this.el).focus();
30317     },
30318     
30319     /**
30320      * Disables this item.
30321      */
30322     disable : function(){
30323         Roo.fly(this.td).addClass("x-item-disabled");
30324         this.disabled = true;
30325         this.el.disabled = true;
30326     },
30327     
30328     /**
30329      * Enables this item.
30330      */
30331     enable : function(){
30332         Roo.fly(this.td).removeClass("x-item-disabled");
30333         this.disabled = false;
30334         this.el.disabled = false;
30335     }
30336 });
30337
30338
30339 /**
30340  * @class Roo.Toolbar.Separator
30341  * @extends Roo.Toolbar.Item
30342  * A simple toolbar separator class
30343  * @constructor
30344  * Creates a new Separator
30345  */
30346 Roo.Toolbar.Separator = function(cfg){
30347     
30348     var s = document.createElement("span");
30349     s.className = "ytb-sep";
30350     if (cfg) {
30351         cfg.el = s;
30352     }
30353     
30354     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30355 };
30356 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30357     enable:Roo.emptyFn,
30358     disable:Roo.emptyFn,
30359     focus:Roo.emptyFn
30360 });
30361
30362 /**
30363  * @class Roo.Toolbar.Spacer
30364  * @extends Roo.Toolbar.Item
30365  * A simple element that adds extra horizontal space to a toolbar.
30366  * @constructor
30367  * Creates a new Spacer
30368  */
30369 Roo.Toolbar.Spacer = function(cfg){
30370     var s = document.createElement("div");
30371     s.className = "ytb-spacer";
30372     if (cfg) {
30373         cfg.el = s;
30374     }
30375     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30376 };
30377 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30378     enable:Roo.emptyFn,
30379     disable:Roo.emptyFn,
30380     focus:Roo.emptyFn
30381 });
30382
30383 /**
30384  * @class Roo.Toolbar.Fill
30385  * @extends Roo.Toolbar.Spacer
30386  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30387  * @constructor
30388  * Creates a new Spacer
30389  */
30390 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30391     // private
30392     render : function(td){
30393         td.style.width = '100%';
30394         Roo.Toolbar.Fill.superclass.render.call(this, td);
30395     }
30396 });
30397
30398 /**
30399  * @class Roo.Toolbar.TextItem
30400  * @extends Roo.Toolbar.Item
30401  * A simple class that renders text directly into a toolbar.
30402  * @constructor
30403  * Creates a new TextItem
30404  * @param {String} text
30405  */
30406 Roo.Toolbar.TextItem = function(cfg){
30407     var  text = cfg || "";
30408     if (typeof(cfg) == 'object') {
30409         text = cfg.text || "";
30410     }  else {
30411         cfg = null;
30412     }
30413     var s = document.createElement("span");
30414     s.className = "ytb-text";
30415     s.innerHTML = text;
30416     if (cfg) {
30417         cfg.el  = s;
30418     }
30419     
30420     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30421 };
30422 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30423     
30424      
30425     enable:Roo.emptyFn,
30426     disable:Roo.emptyFn,
30427     focus:Roo.emptyFn
30428 });
30429
30430 /**
30431  * @class Roo.Toolbar.Button
30432  * @extends Roo.Button
30433  * A button that renders into a toolbar.
30434  * @constructor
30435  * Creates a new Button
30436  * @param {Object} config A standard {@link Roo.Button} config object
30437  */
30438 Roo.Toolbar.Button = function(config){
30439     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30440 };
30441 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30442     render : function(td){
30443         this.td = td;
30444         Roo.Toolbar.Button.superclass.render.call(this, td);
30445     },
30446     
30447     /**
30448      * Removes and destroys this button
30449      */
30450     destroy : function(){
30451         Roo.Toolbar.Button.superclass.destroy.call(this);
30452         this.td.parentNode.removeChild(this.td);
30453     },
30454     
30455     /**
30456      * Shows this button
30457      */
30458     show: function(){
30459         this.hidden = false;
30460         this.td.style.display = "";
30461     },
30462     
30463     /**
30464      * Hides this button
30465      */
30466     hide: function(){
30467         this.hidden = true;
30468         this.td.style.display = "none";
30469     },
30470
30471     /**
30472      * Disables this item
30473      */
30474     disable : function(){
30475         Roo.fly(this.td).addClass("x-item-disabled");
30476         this.disabled = true;
30477     },
30478
30479     /**
30480      * Enables this item
30481      */
30482     enable : function(){
30483         Roo.fly(this.td).removeClass("x-item-disabled");
30484         this.disabled = false;
30485     }
30486 });
30487 // backwards compat
30488 Roo.ToolbarButton = Roo.Toolbar.Button;
30489
30490 /**
30491  * @class Roo.Toolbar.SplitButton
30492  * @extends Roo.SplitButton
30493  * A menu button that renders into a toolbar.
30494  * @constructor
30495  * Creates a new SplitButton
30496  * @param {Object} config A standard {@link Roo.SplitButton} config object
30497  */
30498 Roo.Toolbar.SplitButton = function(config){
30499     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30500 };
30501 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30502     render : function(td){
30503         this.td = td;
30504         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30505     },
30506     
30507     /**
30508      * Removes and destroys this button
30509      */
30510     destroy : function(){
30511         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30512         this.td.parentNode.removeChild(this.td);
30513     },
30514     
30515     /**
30516      * Shows this button
30517      */
30518     show: function(){
30519         this.hidden = false;
30520         this.td.style.display = "";
30521     },
30522     
30523     /**
30524      * Hides this button
30525      */
30526     hide: function(){
30527         this.hidden = true;
30528         this.td.style.display = "none";
30529     }
30530 });
30531
30532 // backwards compat
30533 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30534  * Based on:
30535  * Ext JS Library 1.1.1
30536  * Copyright(c) 2006-2007, Ext JS, LLC.
30537  *
30538  * Originally Released Under LGPL - original licence link has changed is not relivant.
30539  *
30540  * Fork - LGPL
30541  * <script type="text/javascript">
30542  */
30543  
30544 /**
30545  * @class Roo.PagingToolbar
30546  * @extends Roo.Toolbar
30547  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30548  * @constructor
30549  * Create a new PagingToolbar
30550  * @param {Object} config The config object
30551  */
30552 Roo.PagingToolbar = function(el, ds, config)
30553 {
30554     // old args format still supported... - xtype is prefered..
30555     if (typeof(el) == 'object' && el.xtype) {
30556         // created from xtype...
30557         config = el;
30558         ds = el.dataSource;
30559         el = config.container;
30560     }
30561     var items = [];
30562     if (config.items) {
30563         items = config.items;
30564         config.items = [];
30565     }
30566     
30567     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30568     this.ds = ds;
30569     this.cursor = 0;
30570     this.renderButtons(this.el);
30571     this.bind(ds);
30572     
30573     // supprot items array.
30574    
30575     Roo.each(items, function(e) {
30576         this.add(Roo.factory(e));
30577     },this);
30578     
30579 };
30580
30581 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30582     /**
30583      * @cfg {Roo.data.Store} dataSource
30584      * The underlying data store providing the paged data
30585      */
30586     /**
30587      * @cfg {String/HTMLElement/Element} container
30588      * container The id or element that will contain the toolbar
30589      */
30590     /**
30591      * @cfg {Boolean} displayInfo
30592      * True to display the displayMsg (defaults to false)
30593      */
30594     /**
30595      * @cfg {Number} pageSize
30596      * The number of records to display per page (defaults to 20)
30597      */
30598     pageSize: 20,
30599     /**
30600      * @cfg {String} displayMsg
30601      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30602      */
30603     displayMsg : 'Displaying {0} - {1} of {2}',
30604     /**
30605      * @cfg {String} emptyMsg
30606      * The message to display when no records are found (defaults to "No data to display")
30607      */
30608     emptyMsg : 'No data to display',
30609     /**
30610      * Customizable piece of the default paging text (defaults to "Page")
30611      * @type String
30612      */
30613     beforePageText : "Page",
30614     /**
30615      * Customizable piece of the default paging text (defaults to "of %0")
30616      * @type String
30617      */
30618     afterPageText : "of {0}",
30619     /**
30620      * Customizable piece of the default paging text (defaults to "First Page")
30621      * @type String
30622      */
30623     firstText : "First Page",
30624     /**
30625      * Customizable piece of the default paging text (defaults to "Previous Page")
30626      * @type String
30627      */
30628     prevText : "Previous Page",
30629     /**
30630      * Customizable piece of the default paging text (defaults to "Next Page")
30631      * @type String
30632      */
30633     nextText : "Next Page",
30634     /**
30635      * Customizable piece of the default paging text (defaults to "Last Page")
30636      * @type String
30637      */
30638     lastText : "Last Page",
30639     /**
30640      * Customizable piece of the default paging text (defaults to "Refresh")
30641      * @type String
30642      */
30643     refreshText : "Refresh",
30644
30645     // private
30646     renderButtons : function(el){
30647         Roo.PagingToolbar.superclass.render.call(this, el);
30648         this.first = this.addButton({
30649             tooltip: this.firstText,
30650             cls: "x-btn-icon x-grid-page-first",
30651             disabled: true,
30652             handler: this.onClick.createDelegate(this, ["first"])
30653         });
30654         this.prev = this.addButton({
30655             tooltip: this.prevText,
30656             cls: "x-btn-icon x-grid-page-prev",
30657             disabled: true,
30658             handler: this.onClick.createDelegate(this, ["prev"])
30659         });
30660         //this.addSeparator();
30661         this.add(this.beforePageText);
30662         this.field = Roo.get(this.addDom({
30663            tag: "input",
30664            type: "text",
30665            size: "3",
30666            value: "1",
30667            cls: "x-grid-page-number"
30668         }).el);
30669         this.field.on("keydown", this.onPagingKeydown, this);
30670         this.field.on("focus", function(){this.dom.select();});
30671         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30672         this.field.setHeight(18);
30673         //this.addSeparator();
30674         this.next = this.addButton({
30675             tooltip: this.nextText,
30676             cls: "x-btn-icon x-grid-page-next",
30677             disabled: true,
30678             handler: this.onClick.createDelegate(this, ["next"])
30679         });
30680         this.last = this.addButton({
30681             tooltip: this.lastText,
30682             cls: "x-btn-icon x-grid-page-last",
30683             disabled: true,
30684             handler: this.onClick.createDelegate(this, ["last"])
30685         });
30686         //this.addSeparator();
30687         this.loading = this.addButton({
30688             tooltip: this.refreshText,
30689             cls: "x-btn-icon x-grid-loading",
30690             handler: this.onClick.createDelegate(this, ["refresh"])
30691         });
30692
30693         if(this.displayInfo){
30694             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30695         }
30696     },
30697
30698     // private
30699     updateInfo : function(){
30700         if(this.displayEl){
30701             var count = this.ds.getCount();
30702             var msg = count == 0 ?
30703                 this.emptyMsg :
30704                 String.format(
30705                     this.displayMsg,
30706                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30707                 );
30708             this.displayEl.update(msg);
30709         }
30710     },
30711
30712     // private
30713     onLoad : function(ds, r, o){
30714        this.cursor = o.params ? o.params.start : 0;
30715        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30716
30717        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30718        this.field.dom.value = ap;
30719        this.first.setDisabled(ap == 1);
30720        this.prev.setDisabled(ap == 1);
30721        this.next.setDisabled(ap == ps);
30722        this.last.setDisabled(ap == ps);
30723        this.loading.enable();
30724        this.updateInfo();
30725     },
30726
30727     // private
30728     getPageData : function(){
30729         var total = this.ds.getTotalCount();
30730         return {
30731             total : total,
30732             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30733             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30734         };
30735     },
30736
30737     // private
30738     onLoadError : function(){
30739         this.loading.enable();
30740     },
30741
30742     // private
30743     onPagingKeydown : function(e){
30744         var k = e.getKey();
30745         var d = this.getPageData();
30746         if(k == e.RETURN){
30747             var v = this.field.dom.value, pageNum;
30748             if(!v || isNaN(pageNum = parseInt(v, 10))){
30749                 this.field.dom.value = d.activePage;
30750                 return;
30751             }
30752             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30753             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30754             e.stopEvent();
30755         }
30756         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))
30757         {
30758           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30759           this.field.dom.value = pageNum;
30760           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30761           e.stopEvent();
30762         }
30763         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30764         {
30765           var v = this.field.dom.value, pageNum; 
30766           var increment = (e.shiftKey) ? 10 : 1;
30767           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30768             increment *= -1;
30769           }
30770           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30771             this.field.dom.value = d.activePage;
30772             return;
30773           }
30774           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30775           {
30776             this.field.dom.value = parseInt(v, 10) + increment;
30777             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30778             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30779           }
30780           e.stopEvent();
30781         }
30782     },
30783
30784     // private
30785     beforeLoad : function(){
30786         if(this.loading){
30787             this.loading.disable();
30788         }
30789     },
30790
30791     // private
30792     onClick : function(which){
30793         var ds = this.ds;
30794         switch(which){
30795             case "first":
30796                 ds.load({params:{start: 0, limit: this.pageSize}});
30797             break;
30798             case "prev":
30799                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30800             break;
30801             case "next":
30802                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30803             break;
30804             case "last":
30805                 var total = ds.getTotalCount();
30806                 var extra = total % this.pageSize;
30807                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30808                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30809             break;
30810             case "refresh":
30811                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30812             break;
30813         }
30814     },
30815
30816     /**
30817      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30818      * @param {Roo.data.Store} store The data store to unbind
30819      */
30820     unbind : function(ds){
30821         ds.un("beforeload", this.beforeLoad, this);
30822         ds.un("load", this.onLoad, this);
30823         ds.un("loadexception", this.onLoadError, this);
30824         ds.un("remove", this.updateInfo, this);
30825         ds.un("add", this.updateInfo, this);
30826         this.ds = undefined;
30827     },
30828
30829     /**
30830      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30831      * @param {Roo.data.Store} store The data store to bind
30832      */
30833     bind : function(ds){
30834         ds.on("beforeload", this.beforeLoad, this);
30835         ds.on("load", this.onLoad, this);
30836         ds.on("loadexception", this.onLoadError, this);
30837         ds.on("remove", this.updateInfo, this);
30838         ds.on("add", this.updateInfo, this);
30839         this.ds = ds;
30840     }
30841 });/*
30842  * Based on:
30843  * Ext JS Library 1.1.1
30844  * Copyright(c) 2006-2007, Ext JS, LLC.
30845  *
30846  * Originally Released Under LGPL - original licence link has changed is not relivant.
30847  *
30848  * Fork - LGPL
30849  * <script type="text/javascript">
30850  */
30851
30852 /**
30853  * @class Roo.Resizable
30854  * @extends Roo.util.Observable
30855  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30856  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30857  * 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
30858  * the element will be wrapped for you automatically.</p>
30859  * <p>Here is the list of valid resize handles:</p>
30860  * <pre>
30861 Value   Description
30862 ------  -------------------
30863  'n'     north
30864  's'     south
30865  'e'     east
30866  'w'     west
30867  'nw'    northwest
30868  'sw'    southwest
30869  'se'    southeast
30870  'ne'    northeast
30871  'hd'    horizontal drag
30872  'all'   all
30873 </pre>
30874  * <p>Here's an example showing the creation of a typical Resizable:</p>
30875  * <pre><code>
30876 var resizer = new Roo.Resizable("element-id", {
30877     handles: 'all',
30878     minWidth: 200,
30879     minHeight: 100,
30880     maxWidth: 500,
30881     maxHeight: 400,
30882     pinned: true
30883 });
30884 resizer.on("resize", myHandler);
30885 </code></pre>
30886  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30887  * resizer.east.setDisplayed(false);</p>
30888  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30889  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30890  * resize operation's new size (defaults to [0, 0])
30891  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30892  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30893  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30894  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30895  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30896  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30897  * @cfg {Number} width The width of the element in pixels (defaults to null)
30898  * @cfg {Number} height The height of the element in pixels (defaults to null)
30899  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30900  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30901  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30902  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30903  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30904  * in favor of the handles config option (defaults to false)
30905  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30906  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30907  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30908  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30909  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30910  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30911  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30912  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30913  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30914  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30915  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30916  * @constructor
30917  * Create a new resizable component
30918  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30919  * @param {Object} config configuration options
30920   */
30921 Roo.Resizable = function(el, config)
30922 {
30923     this.el = Roo.get(el);
30924
30925     if(config && config.wrap){
30926         config.resizeChild = this.el;
30927         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30928         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30929         this.el.setStyle("overflow", "hidden");
30930         this.el.setPositioning(config.resizeChild.getPositioning());
30931         config.resizeChild.clearPositioning();
30932         if(!config.width || !config.height){
30933             var csize = config.resizeChild.getSize();
30934             this.el.setSize(csize.width, csize.height);
30935         }
30936         if(config.pinned && !config.adjustments){
30937             config.adjustments = "auto";
30938         }
30939     }
30940
30941     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30942     this.proxy.unselectable();
30943     this.proxy.enableDisplayMode('block');
30944
30945     Roo.apply(this, config);
30946
30947     if(this.pinned){
30948         this.disableTrackOver = true;
30949         this.el.addClass("x-resizable-pinned");
30950     }
30951     // if the element isn't positioned, make it relative
30952     var position = this.el.getStyle("position");
30953     if(position != "absolute" && position != "fixed"){
30954         this.el.setStyle("position", "relative");
30955     }
30956     if(!this.handles){ // no handles passed, must be legacy style
30957         this.handles = 's,e,se';
30958         if(this.multiDirectional){
30959             this.handles += ',n,w';
30960         }
30961     }
30962     if(this.handles == "all"){
30963         this.handles = "n s e w ne nw se sw";
30964     }
30965     var hs = this.handles.split(/\s*?[,;]\s*?| /);
30966     var ps = Roo.Resizable.positions;
30967     for(var i = 0, len = hs.length; i < len; i++){
30968         if(hs[i] && ps[hs[i]]){
30969             var pos = ps[hs[i]];
30970             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
30971         }
30972     }
30973     // legacy
30974     this.corner = this.southeast;
30975     
30976     // updateBox = the box can move..
30977     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
30978         this.updateBox = true;
30979     }
30980
30981     this.activeHandle = null;
30982
30983     if(this.resizeChild){
30984         if(typeof this.resizeChild == "boolean"){
30985             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
30986         }else{
30987             this.resizeChild = Roo.get(this.resizeChild, true);
30988         }
30989     }
30990     
30991     if(this.adjustments == "auto"){
30992         var rc = this.resizeChild;
30993         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
30994         if(rc && (hw || hn)){
30995             rc.position("relative");
30996             rc.setLeft(hw ? hw.el.getWidth() : 0);
30997             rc.setTop(hn ? hn.el.getHeight() : 0);
30998         }
30999         this.adjustments = [
31000             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31001             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31002         ];
31003     }
31004
31005     if(this.draggable){
31006         this.dd = this.dynamic ?
31007             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31008         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31009     }
31010
31011     // public events
31012     this.addEvents({
31013         /**
31014          * @event beforeresize
31015          * Fired before resize is allowed. Set enabled to false to cancel resize.
31016          * @param {Roo.Resizable} this
31017          * @param {Roo.EventObject} e The mousedown event
31018          */
31019         "beforeresize" : true,
31020         /**
31021          * @event resizing
31022          * Fired a resizing.
31023          * @param {Roo.Resizable} this
31024          * @param {Number} x The new x position
31025          * @param {Number} y The new y position
31026          * @param {Number} w The new w width
31027          * @param {Number} h The new h hight
31028          * @param {Roo.EventObject} e The mouseup event
31029          */
31030         "resizing" : true,
31031         /**
31032          * @event resize
31033          * Fired after a resize.
31034          * @param {Roo.Resizable} this
31035          * @param {Number} width The new width
31036          * @param {Number} height The new height
31037          * @param {Roo.EventObject} e The mouseup event
31038          */
31039         "resize" : true
31040     });
31041
31042     if(this.width !== null && this.height !== null){
31043         this.resizeTo(this.width, this.height);
31044     }else{
31045         this.updateChildSize();
31046     }
31047     if(Roo.isIE){
31048         this.el.dom.style.zoom = 1;
31049     }
31050     Roo.Resizable.superclass.constructor.call(this);
31051 };
31052
31053 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31054         resizeChild : false,
31055         adjustments : [0, 0],
31056         minWidth : 5,
31057         minHeight : 5,
31058         maxWidth : 10000,
31059         maxHeight : 10000,
31060         enabled : true,
31061         animate : false,
31062         duration : .35,
31063         dynamic : false,
31064         handles : false,
31065         multiDirectional : false,
31066         disableTrackOver : false,
31067         easing : 'easeOutStrong',
31068         widthIncrement : 0,
31069         heightIncrement : 0,
31070         pinned : false,
31071         width : null,
31072         height : null,
31073         preserveRatio : false,
31074         transparent: false,
31075         minX: 0,
31076         minY: 0,
31077         draggable: false,
31078
31079         /**
31080          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31081          */
31082         constrainTo: undefined,
31083         /**
31084          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31085          */
31086         resizeRegion: undefined,
31087
31088
31089     /**
31090      * Perform a manual resize
31091      * @param {Number} width
31092      * @param {Number} height
31093      */
31094     resizeTo : function(width, height){
31095         this.el.setSize(width, height);
31096         this.updateChildSize();
31097         this.fireEvent("resize", this, width, height, null);
31098     },
31099
31100     // private
31101     startSizing : function(e, handle){
31102         this.fireEvent("beforeresize", this, e);
31103         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31104
31105             if(!this.overlay){
31106                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31107                 this.overlay.unselectable();
31108                 this.overlay.enableDisplayMode("block");
31109                 this.overlay.on("mousemove", this.onMouseMove, this);
31110                 this.overlay.on("mouseup", this.onMouseUp, this);
31111             }
31112             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31113
31114             this.resizing = true;
31115             this.startBox = this.el.getBox();
31116             this.startPoint = e.getXY();
31117             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31118                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31119
31120             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31121             this.overlay.show();
31122
31123             if(this.constrainTo) {
31124                 var ct = Roo.get(this.constrainTo);
31125                 this.resizeRegion = ct.getRegion().adjust(
31126                     ct.getFrameWidth('t'),
31127                     ct.getFrameWidth('l'),
31128                     -ct.getFrameWidth('b'),
31129                     -ct.getFrameWidth('r')
31130                 );
31131             }
31132
31133             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31134             this.proxy.show();
31135             this.proxy.setBox(this.startBox);
31136             if(!this.dynamic){
31137                 this.proxy.setStyle('visibility', 'visible');
31138             }
31139         }
31140     },
31141
31142     // private
31143     onMouseDown : function(handle, e){
31144         if(this.enabled){
31145             e.stopEvent();
31146             this.activeHandle = handle;
31147             this.startSizing(e, handle);
31148         }
31149     },
31150
31151     // private
31152     onMouseUp : function(e){
31153         var size = this.resizeElement();
31154         this.resizing = false;
31155         this.handleOut();
31156         this.overlay.hide();
31157         this.proxy.hide();
31158         this.fireEvent("resize", this, size.width, size.height, e);
31159     },
31160
31161     // private
31162     updateChildSize : function(){
31163         
31164         if(this.resizeChild){
31165             var el = this.el;
31166             var child = this.resizeChild;
31167             var adj = this.adjustments;
31168             if(el.dom.offsetWidth){
31169                 var b = el.getSize(true);
31170                 child.setSize(b.width+adj[0], b.height+adj[1]);
31171             }
31172             // Second call here for IE
31173             // The first call enables instant resizing and
31174             // the second call corrects scroll bars if they
31175             // exist
31176             if(Roo.isIE){
31177                 setTimeout(function(){
31178                     if(el.dom.offsetWidth){
31179                         var b = el.getSize(true);
31180                         child.setSize(b.width+adj[0], b.height+adj[1]);
31181                     }
31182                 }, 10);
31183             }
31184         }
31185     },
31186
31187     // private
31188     snap : function(value, inc, min){
31189         if(!inc || !value) {
31190             return value;
31191         }
31192         var newValue = value;
31193         var m = value % inc;
31194         if(m > 0){
31195             if(m > (inc/2)){
31196                 newValue = value + (inc-m);
31197             }else{
31198                 newValue = value - m;
31199             }
31200         }
31201         return Math.max(min, newValue);
31202     },
31203
31204     // private
31205     resizeElement : function(){
31206         var box = this.proxy.getBox();
31207         if(this.updateBox){
31208             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31209         }else{
31210             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31211         }
31212         this.updateChildSize();
31213         if(!this.dynamic){
31214             this.proxy.hide();
31215         }
31216         return box;
31217     },
31218
31219     // private
31220     constrain : function(v, diff, m, mx){
31221         if(v - diff < m){
31222             diff = v - m;
31223         }else if(v - diff > mx){
31224             diff = mx - v;
31225         }
31226         return diff;
31227     },
31228
31229     // private
31230     onMouseMove : function(e){
31231         
31232         if(this.enabled){
31233             try{// try catch so if something goes wrong the user doesn't get hung
31234
31235             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31236                 return;
31237             }
31238
31239             //var curXY = this.startPoint;
31240             var curSize = this.curSize || this.startBox;
31241             var x = this.startBox.x, y = this.startBox.y;
31242             var ox = x, oy = y;
31243             var w = curSize.width, h = curSize.height;
31244             var ow = w, oh = h;
31245             var mw = this.minWidth, mh = this.minHeight;
31246             var mxw = this.maxWidth, mxh = this.maxHeight;
31247             var wi = this.widthIncrement;
31248             var hi = this.heightIncrement;
31249
31250             var eventXY = e.getXY();
31251             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31252             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31253
31254             var pos = this.activeHandle.position;
31255
31256             switch(pos){
31257                 case "east":
31258                     w += diffX;
31259                     w = Math.min(Math.max(mw, w), mxw);
31260                     break;
31261              
31262                 case "south":
31263                     h += diffY;
31264                     h = Math.min(Math.max(mh, h), mxh);
31265                     break;
31266                 case "southeast":
31267                     w += diffX;
31268                     h += diffY;
31269                     w = Math.min(Math.max(mw, w), mxw);
31270                     h = Math.min(Math.max(mh, h), mxh);
31271                     break;
31272                 case "north":
31273                     diffY = this.constrain(h, diffY, mh, mxh);
31274                     y += diffY;
31275                     h -= diffY;
31276                     break;
31277                 case "hdrag":
31278                     
31279                     if (wi) {
31280                         var adiffX = Math.abs(diffX);
31281                         var sub = (adiffX % wi); // how much 
31282                         if (sub > (wi/2)) { // far enough to snap
31283                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31284                         } else {
31285                             // remove difference.. 
31286                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31287                         }
31288                     }
31289                     x += diffX;
31290                     x = Math.max(this.minX, x);
31291                     break;
31292                 case "west":
31293                     diffX = this.constrain(w, diffX, mw, mxw);
31294                     x += diffX;
31295                     w -= diffX;
31296                     break;
31297                 case "northeast":
31298                     w += diffX;
31299                     w = Math.min(Math.max(mw, w), mxw);
31300                     diffY = this.constrain(h, diffY, mh, mxh);
31301                     y += diffY;
31302                     h -= diffY;
31303                     break;
31304                 case "northwest":
31305                     diffX = this.constrain(w, diffX, mw, mxw);
31306                     diffY = this.constrain(h, diffY, mh, mxh);
31307                     y += diffY;
31308                     h -= diffY;
31309                     x += diffX;
31310                     w -= diffX;
31311                     break;
31312                case "southwest":
31313                     diffX = this.constrain(w, diffX, mw, mxw);
31314                     h += diffY;
31315                     h = Math.min(Math.max(mh, h), mxh);
31316                     x += diffX;
31317                     w -= diffX;
31318                     break;
31319             }
31320
31321             var sw = this.snap(w, wi, mw);
31322             var sh = this.snap(h, hi, mh);
31323             if(sw != w || sh != h){
31324                 switch(pos){
31325                     case "northeast":
31326                         y -= sh - h;
31327                     break;
31328                     case "north":
31329                         y -= sh - h;
31330                         break;
31331                     case "southwest":
31332                         x -= sw - w;
31333                     break;
31334                     case "west":
31335                         x -= sw - w;
31336                         break;
31337                     case "northwest":
31338                         x -= sw - w;
31339                         y -= sh - h;
31340                     break;
31341                 }
31342                 w = sw;
31343                 h = sh;
31344             }
31345
31346             if(this.preserveRatio){
31347                 switch(pos){
31348                     case "southeast":
31349                     case "east":
31350                         h = oh * (w/ow);
31351                         h = Math.min(Math.max(mh, h), mxh);
31352                         w = ow * (h/oh);
31353                        break;
31354                     case "south":
31355                         w = ow * (h/oh);
31356                         w = Math.min(Math.max(mw, w), mxw);
31357                         h = oh * (w/ow);
31358                         break;
31359                     case "northeast":
31360                         w = ow * (h/oh);
31361                         w = Math.min(Math.max(mw, w), mxw);
31362                         h = oh * (w/ow);
31363                     break;
31364                     case "north":
31365                         var tw = w;
31366                         w = ow * (h/oh);
31367                         w = Math.min(Math.max(mw, w), mxw);
31368                         h = oh * (w/ow);
31369                         x += (tw - w) / 2;
31370                         break;
31371                     case "southwest":
31372                         h = oh * (w/ow);
31373                         h = Math.min(Math.max(mh, h), mxh);
31374                         var tw = w;
31375                         w = ow * (h/oh);
31376                         x += tw - w;
31377                         break;
31378                     case "west":
31379                         var th = h;
31380                         h = oh * (w/ow);
31381                         h = Math.min(Math.max(mh, h), mxh);
31382                         y += (th - h) / 2;
31383                         var tw = w;
31384                         w = ow * (h/oh);
31385                         x += tw - w;
31386                        break;
31387                     case "northwest":
31388                         var tw = w;
31389                         var th = h;
31390                         h = oh * (w/ow);
31391                         h = Math.min(Math.max(mh, h), mxh);
31392                         w = ow * (h/oh);
31393                         y += th - h;
31394                         x += tw - w;
31395                        break;
31396
31397                 }
31398             }
31399             if (pos == 'hdrag') {
31400                 w = ow;
31401             }
31402             this.proxy.setBounds(x, y, w, h);
31403             if(this.dynamic){
31404                 this.resizeElement();
31405             }
31406             }catch(e){}
31407         }
31408         this.fireEvent("resizing", this, x, y, w, h, e);
31409     },
31410
31411     // private
31412     handleOver : function(){
31413         if(this.enabled){
31414             this.el.addClass("x-resizable-over");
31415         }
31416     },
31417
31418     // private
31419     handleOut : function(){
31420         if(!this.resizing){
31421             this.el.removeClass("x-resizable-over");
31422         }
31423     },
31424
31425     /**
31426      * Returns the element this component is bound to.
31427      * @return {Roo.Element}
31428      */
31429     getEl : function(){
31430         return this.el;
31431     },
31432
31433     /**
31434      * Returns the resizeChild element (or null).
31435      * @return {Roo.Element}
31436      */
31437     getResizeChild : function(){
31438         return this.resizeChild;
31439     },
31440     groupHandler : function()
31441     {
31442         
31443     },
31444     /**
31445      * Destroys this resizable. If the element was wrapped and
31446      * removeEl is not true then the element remains.
31447      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31448      */
31449     destroy : function(removeEl){
31450         this.proxy.remove();
31451         if(this.overlay){
31452             this.overlay.removeAllListeners();
31453             this.overlay.remove();
31454         }
31455         var ps = Roo.Resizable.positions;
31456         for(var k in ps){
31457             if(typeof ps[k] != "function" && this[ps[k]]){
31458                 var h = this[ps[k]];
31459                 h.el.removeAllListeners();
31460                 h.el.remove();
31461             }
31462         }
31463         if(removeEl){
31464             this.el.update("");
31465             this.el.remove();
31466         }
31467     }
31468 });
31469
31470 // private
31471 // hash to map config positions to true positions
31472 Roo.Resizable.positions = {
31473     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31474     hd: "hdrag"
31475 };
31476
31477 // private
31478 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31479     if(!this.tpl){
31480         // only initialize the template if resizable is used
31481         var tpl = Roo.DomHelper.createTemplate(
31482             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31483         );
31484         tpl.compile();
31485         Roo.Resizable.Handle.prototype.tpl = tpl;
31486     }
31487     this.position = pos;
31488     this.rz = rz;
31489     // show north drag fro topdra
31490     var handlepos = pos == 'hdrag' ? 'north' : pos;
31491     
31492     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31493     if (pos == 'hdrag') {
31494         this.el.setStyle('cursor', 'pointer');
31495     }
31496     this.el.unselectable();
31497     if(transparent){
31498         this.el.setOpacity(0);
31499     }
31500     this.el.on("mousedown", this.onMouseDown, this);
31501     if(!disableTrackOver){
31502         this.el.on("mouseover", this.onMouseOver, this);
31503         this.el.on("mouseout", this.onMouseOut, this);
31504     }
31505 };
31506
31507 // private
31508 Roo.Resizable.Handle.prototype = {
31509     afterResize : function(rz){
31510         Roo.log('after?');
31511         // do nothing
31512     },
31513     // private
31514     onMouseDown : function(e){
31515         this.rz.onMouseDown(this, e);
31516     },
31517     // private
31518     onMouseOver : function(e){
31519         this.rz.handleOver(this, e);
31520     },
31521     // private
31522     onMouseOut : function(e){
31523         this.rz.handleOut(this, e);
31524     }
31525 };/*
31526  * Based on:
31527  * Ext JS Library 1.1.1
31528  * Copyright(c) 2006-2007, Ext JS, LLC.
31529  *
31530  * Originally Released Under LGPL - original licence link has changed is not relivant.
31531  *
31532  * Fork - LGPL
31533  * <script type="text/javascript">
31534  */
31535
31536 /**
31537  * @class Roo.Editor
31538  * @extends Roo.Component
31539  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31540  * @constructor
31541  * Create a new Editor
31542  * @param {Roo.form.Field} field The Field object (or descendant)
31543  * @param {Object} config The config object
31544  */
31545 Roo.Editor = function(field, config){
31546     Roo.Editor.superclass.constructor.call(this, config);
31547     this.field = field;
31548     this.addEvents({
31549         /**
31550              * @event beforestartedit
31551              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31552              * false from the handler of this event.
31553              * @param {Editor} this
31554              * @param {Roo.Element} boundEl The underlying element bound to this editor
31555              * @param {Mixed} value The field value being set
31556              */
31557         "beforestartedit" : true,
31558         /**
31559              * @event startedit
31560              * Fires when this editor is displayed
31561              * @param {Roo.Element} boundEl The underlying element bound to this editor
31562              * @param {Mixed} value The starting field value
31563              */
31564         "startedit" : true,
31565         /**
31566              * @event beforecomplete
31567              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31568              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31569              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31570              * event will not fire since no edit actually occurred.
31571              * @param {Editor} this
31572              * @param {Mixed} value The current field value
31573              * @param {Mixed} startValue The original field value
31574              */
31575         "beforecomplete" : true,
31576         /**
31577              * @event complete
31578              * Fires after editing is complete and any changed value has been written to the underlying field.
31579              * @param {Editor} this
31580              * @param {Mixed} value The current field value
31581              * @param {Mixed} startValue The original field value
31582              */
31583         "complete" : true,
31584         /**
31585          * @event specialkey
31586          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31587          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31588          * @param {Roo.form.Field} this
31589          * @param {Roo.EventObject} e The event object
31590          */
31591         "specialkey" : true
31592     });
31593 };
31594
31595 Roo.extend(Roo.Editor, Roo.Component, {
31596     /**
31597      * @cfg {Boolean/String} autosize
31598      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31599      * or "height" to adopt the height only (defaults to false)
31600      */
31601     /**
31602      * @cfg {Boolean} revertInvalid
31603      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31604      * validation fails (defaults to true)
31605      */
31606     /**
31607      * @cfg {Boolean} ignoreNoChange
31608      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31609      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31610      * will never be ignored.
31611      */
31612     /**
31613      * @cfg {Boolean} hideEl
31614      * False to keep the bound element visible while the editor is displayed (defaults to true)
31615      */
31616     /**
31617      * @cfg {Mixed} value
31618      * The data value of the underlying field (defaults to "")
31619      */
31620     value : "",
31621     /**
31622      * @cfg {String} alignment
31623      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31624      */
31625     alignment: "c-c?",
31626     /**
31627      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31628      * for bottom-right shadow (defaults to "frame")
31629      */
31630     shadow : "frame",
31631     /**
31632      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31633      */
31634     constrain : false,
31635     /**
31636      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31637      */
31638     completeOnEnter : false,
31639     /**
31640      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31641      */
31642     cancelOnEsc : false,
31643     /**
31644      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31645      */
31646     updateEl : false,
31647
31648     // private
31649     onRender : function(ct, position){
31650         this.el = new Roo.Layer({
31651             shadow: this.shadow,
31652             cls: "x-editor",
31653             parentEl : ct,
31654             shim : this.shim,
31655             shadowOffset:4,
31656             id: this.id,
31657             constrain: this.constrain
31658         });
31659         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31660         if(this.field.msgTarget != 'title'){
31661             this.field.msgTarget = 'qtip';
31662         }
31663         this.field.render(this.el);
31664         if(Roo.isGecko){
31665             this.field.el.dom.setAttribute('autocomplete', 'off');
31666         }
31667         this.field.on("specialkey", this.onSpecialKey, this);
31668         if(this.swallowKeys){
31669             this.field.el.swallowEvent(['keydown','keypress']);
31670         }
31671         this.field.show();
31672         this.field.on("blur", this.onBlur, this);
31673         if(this.field.grow){
31674             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31675         }
31676     },
31677
31678     onSpecialKey : function(field, e)
31679     {
31680         //Roo.log('editor onSpecialKey');
31681         if(this.completeOnEnter && e.getKey() == e.ENTER){
31682             e.stopEvent();
31683             this.completeEdit();
31684             return;
31685         }
31686         // do not fire special key otherwise it might hide close the editor...
31687         if(e.getKey() == e.ENTER){    
31688             return;
31689         }
31690         if(this.cancelOnEsc && e.getKey() == e.ESC){
31691             this.cancelEdit();
31692             return;
31693         } 
31694         this.fireEvent('specialkey', field, e);
31695     
31696     },
31697
31698     /**
31699      * Starts the editing process and shows the editor.
31700      * @param {String/HTMLElement/Element} el The element to edit
31701      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31702       * to the innerHTML of el.
31703      */
31704     startEdit : function(el, value){
31705         if(this.editing){
31706             this.completeEdit();
31707         }
31708         this.boundEl = Roo.get(el);
31709         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31710         if(!this.rendered){
31711             this.render(this.parentEl || document.body);
31712         }
31713         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31714             return;
31715         }
31716         this.startValue = v;
31717         this.field.setValue(v);
31718         if(this.autoSize){
31719             var sz = this.boundEl.getSize();
31720             switch(this.autoSize){
31721                 case "width":
31722                 this.setSize(sz.width,  "");
31723                 break;
31724                 case "height":
31725                 this.setSize("",  sz.height);
31726                 break;
31727                 default:
31728                 this.setSize(sz.width,  sz.height);
31729             }
31730         }
31731         this.el.alignTo(this.boundEl, this.alignment);
31732         this.editing = true;
31733         if(Roo.QuickTips){
31734             Roo.QuickTips.disable();
31735         }
31736         this.show();
31737     },
31738
31739     /**
31740      * Sets the height and width of this editor.
31741      * @param {Number} width The new width
31742      * @param {Number} height The new height
31743      */
31744     setSize : function(w, h){
31745         this.field.setSize(w, h);
31746         if(this.el){
31747             this.el.sync();
31748         }
31749     },
31750
31751     /**
31752      * Realigns the editor to the bound field based on the current alignment config value.
31753      */
31754     realign : function(){
31755         this.el.alignTo(this.boundEl, this.alignment);
31756     },
31757
31758     /**
31759      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31760      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31761      */
31762     completeEdit : function(remainVisible){
31763         if(!this.editing){
31764             return;
31765         }
31766         var v = this.getValue();
31767         if(this.revertInvalid !== false && !this.field.isValid()){
31768             v = this.startValue;
31769             this.cancelEdit(true);
31770         }
31771         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31772             this.editing = false;
31773             this.hide();
31774             return;
31775         }
31776         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31777             this.editing = false;
31778             if(this.updateEl && this.boundEl){
31779                 this.boundEl.update(v);
31780             }
31781             if(remainVisible !== true){
31782                 this.hide();
31783             }
31784             this.fireEvent("complete", this, v, this.startValue);
31785         }
31786     },
31787
31788     // private
31789     onShow : function(){
31790         this.el.show();
31791         if(this.hideEl !== false){
31792             this.boundEl.hide();
31793         }
31794         this.field.show();
31795         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31796             this.fixIEFocus = true;
31797             this.deferredFocus.defer(50, this);
31798         }else{
31799             this.field.focus();
31800         }
31801         this.fireEvent("startedit", this.boundEl, this.startValue);
31802     },
31803
31804     deferredFocus : function(){
31805         if(this.editing){
31806             this.field.focus();
31807         }
31808     },
31809
31810     /**
31811      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31812      * reverted to the original starting value.
31813      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31814      * cancel (defaults to false)
31815      */
31816     cancelEdit : function(remainVisible){
31817         if(this.editing){
31818             this.setValue(this.startValue);
31819             if(remainVisible !== true){
31820                 this.hide();
31821             }
31822         }
31823     },
31824
31825     // private
31826     onBlur : function(){
31827         if(this.allowBlur !== true && this.editing){
31828             this.completeEdit();
31829         }
31830     },
31831
31832     // private
31833     onHide : function(){
31834         if(this.editing){
31835             this.completeEdit();
31836             return;
31837         }
31838         this.field.blur();
31839         if(this.field.collapse){
31840             this.field.collapse();
31841         }
31842         this.el.hide();
31843         if(this.hideEl !== false){
31844             this.boundEl.show();
31845         }
31846         if(Roo.QuickTips){
31847             Roo.QuickTips.enable();
31848         }
31849     },
31850
31851     /**
31852      * Sets the data value of the editor
31853      * @param {Mixed} value Any valid value supported by the underlying field
31854      */
31855     setValue : function(v){
31856         this.field.setValue(v);
31857     },
31858
31859     /**
31860      * Gets the data value of the editor
31861      * @return {Mixed} The data value
31862      */
31863     getValue : function(){
31864         return this.field.getValue();
31865     }
31866 });/*
31867  * Based on:
31868  * Ext JS Library 1.1.1
31869  * Copyright(c) 2006-2007, Ext JS, LLC.
31870  *
31871  * Originally Released Under LGPL - original licence link has changed is not relivant.
31872  *
31873  * Fork - LGPL
31874  * <script type="text/javascript">
31875  */
31876  
31877 /**
31878  * @class Roo.BasicDialog
31879  * @extends Roo.util.Observable
31880  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31881  * <pre><code>
31882 var dlg = new Roo.BasicDialog("my-dlg", {
31883     height: 200,
31884     width: 300,
31885     minHeight: 100,
31886     minWidth: 150,
31887     modal: true,
31888     proxyDrag: true,
31889     shadow: true
31890 });
31891 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31892 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31893 dlg.addButton('Cancel', dlg.hide, dlg);
31894 dlg.show();
31895 </code></pre>
31896   <b>A Dialog should always be a direct child of the body element.</b>
31897  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31898  * @cfg {String} title Default text to display in the title bar (defaults to null)
31899  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31900  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31901  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31902  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31903  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31904  * (defaults to null with no animation)
31905  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31906  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31907  * property for valid values (defaults to 'all')
31908  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31909  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31910  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31911  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31912  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31913  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31914  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31915  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31916  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31917  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31918  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31919  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31920  * draggable = true (defaults to false)
31921  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31922  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31923  * shadow (defaults to false)
31924  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31925  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31926  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31927  * @cfg {Array} buttons Array of buttons
31928  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31929  * @constructor
31930  * Create a new BasicDialog.
31931  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31932  * @param {Object} config Configuration options
31933  */
31934 Roo.BasicDialog = function(el, config){
31935     this.el = Roo.get(el);
31936     var dh = Roo.DomHelper;
31937     if(!this.el && config && config.autoCreate){
31938         if(typeof config.autoCreate == "object"){
31939             if(!config.autoCreate.id){
31940                 config.autoCreate.id = el;
31941             }
31942             this.el = dh.append(document.body,
31943                         config.autoCreate, true);
31944         }else{
31945             this.el = dh.append(document.body,
31946                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31947         }
31948     }
31949     el = this.el;
31950     el.setDisplayed(true);
31951     el.hide = this.hideAction;
31952     this.id = el.id;
31953     el.addClass("x-dlg");
31954
31955     Roo.apply(this, config);
31956
31957     this.proxy = el.createProxy("x-dlg-proxy");
31958     this.proxy.hide = this.hideAction;
31959     this.proxy.setOpacity(.5);
31960     this.proxy.hide();
31961
31962     if(config.width){
31963         el.setWidth(config.width);
31964     }
31965     if(config.height){
31966         el.setHeight(config.height);
31967     }
31968     this.size = el.getSize();
31969     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
31970         this.xy = [config.x,config.y];
31971     }else{
31972         this.xy = el.getCenterXY(true);
31973     }
31974     /** The header element @type Roo.Element */
31975     this.header = el.child("> .x-dlg-hd");
31976     /** The body element @type Roo.Element */
31977     this.body = el.child("> .x-dlg-bd");
31978     /** The footer element @type Roo.Element */
31979     this.footer = el.child("> .x-dlg-ft");
31980
31981     if(!this.header){
31982         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
31983     }
31984     if(!this.body){
31985         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
31986     }
31987
31988     this.header.unselectable();
31989     if(this.title){
31990         this.header.update(this.title);
31991     }
31992     // this element allows the dialog to be focused for keyboard event
31993     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
31994     this.focusEl.swallowEvent("click", true);
31995
31996     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
31997
31998     // wrap the body and footer for special rendering
31999     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32000     if(this.footer){
32001         this.bwrap.dom.appendChild(this.footer.dom);
32002     }
32003
32004     this.bg = this.el.createChild({
32005         tag: "div", cls:"x-dlg-bg",
32006         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32007     });
32008     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32009
32010
32011     if(this.autoScroll !== false && !this.autoTabs){
32012         this.body.setStyle("overflow", "auto");
32013     }
32014
32015     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32016
32017     if(this.closable !== false){
32018         this.el.addClass("x-dlg-closable");
32019         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32020         this.close.on("click", this.closeClick, this);
32021         this.close.addClassOnOver("x-dlg-close-over");
32022     }
32023     if(this.collapsible !== false){
32024         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32025         this.collapseBtn.on("click", this.collapseClick, this);
32026         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32027         this.header.on("dblclick", this.collapseClick, this);
32028     }
32029     if(this.resizable !== false){
32030         this.el.addClass("x-dlg-resizable");
32031         this.resizer = new Roo.Resizable(el, {
32032             minWidth: this.minWidth || 80,
32033             minHeight:this.minHeight || 80,
32034             handles: this.resizeHandles || "all",
32035             pinned: true
32036         });
32037         this.resizer.on("beforeresize", this.beforeResize, this);
32038         this.resizer.on("resize", this.onResize, this);
32039     }
32040     if(this.draggable !== false){
32041         el.addClass("x-dlg-draggable");
32042         if (!this.proxyDrag) {
32043             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32044         }
32045         else {
32046             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32047         }
32048         dd.setHandleElId(this.header.id);
32049         dd.endDrag = this.endMove.createDelegate(this);
32050         dd.startDrag = this.startMove.createDelegate(this);
32051         dd.onDrag = this.onDrag.createDelegate(this);
32052         dd.scroll = false;
32053         this.dd = dd;
32054     }
32055     if(this.modal){
32056         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32057         this.mask.enableDisplayMode("block");
32058         this.mask.hide();
32059         this.el.addClass("x-dlg-modal");
32060     }
32061     if(this.shadow){
32062         this.shadow = new Roo.Shadow({
32063             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32064             offset : this.shadowOffset
32065         });
32066     }else{
32067         this.shadowOffset = 0;
32068     }
32069     if(Roo.useShims && this.shim !== false){
32070         this.shim = this.el.createShim();
32071         this.shim.hide = this.hideAction;
32072         this.shim.hide();
32073     }else{
32074         this.shim = false;
32075     }
32076     if(this.autoTabs){
32077         this.initTabs();
32078     }
32079     if (this.buttons) { 
32080         var bts= this.buttons;
32081         this.buttons = [];
32082         Roo.each(bts, function(b) {
32083             this.addButton(b);
32084         }, this);
32085     }
32086     
32087     
32088     this.addEvents({
32089         /**
32090          * @event keydown
32091          * Fires when a key is pressed
32092          * @param {Roo.BasicDialog} this
32093          * @param {Roo.EventObject} e
32094          */
32095         "keydown" : true,
32096         /**
32097          * @event move
32098          * Fires when this dialog is moved by the user.
32099          * @param {Roo.BasicDialog} this
32100          * @param {Number} x The new page X
32101          * @param {Number} y The new page Y
32102          */
32103         "move" : true,
32104         /**
32105          * @event resize
32106          * Fires when this dialog is resized by the user.
32107          * @param {Roo.BasicDialog} this
32108          * @param {Number} width The new width
32109          * @param {Number} height The new height
32110          */
32111         "resize" : true,
32112         /**
32113          * @event beforehide
32114          * Fires before this dialog is hidden.
32115          * @param {Roo.BasicDialog} this
32116          */
32117         "beforehide" : true,
32118         /**
32119          * @event hide
32120          * Fires when this dialog is hidden.
32121          * @param {Roo.BasicDialog} this
32122          */
32123         "hide" : true,
32124         /**
32125          * @event beforeshow
32126          * Fires before this dialog is shown.
32127          * @param {Roo.BasicDialog} this
32128          */
32129         "beforeshow" : true,
32130         /**
32131          * @event show
32132          * Fires when this dialog is shown.
32133          * @param {Roo.BasicDialog} this
32134          */
32135         "show" : true
32136     });
32137     el.on("keydown", this.onKeyDown, this);
32138     el.on("mousedown", this.toFront, this);
32139     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32140     this.el.hide();
32141     Roo.DialogManager.register(this);
32142     Roo.BasicDialog.superclass.constructor.call(this);
32143 };
32144
32145 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32146     shadowOffset: Roo.isIE ? 6 : 5,
32147     minHeight: 80,
32148     minWidth: 200,
32149     minButtonWidth: 75,
32150     defaultButton: null,
32151     buttonAlign: "right",
32152     tabTag: 'div',
32153     firstShow: true,
32154
32155     /**
32156      * Sets the dialog title text
32157      * @param {String} text The title text to display
32158      * @return {Roo.BasicDialog} this
32159      */
32160     setTitle : function(text){
32161         this.header.update(text);
32162         return this;
32163     },
32164
32165     // private
32166     closeClick : function(){
32167         this.hide();
32168     },
32169
32170     // private
32171     collapseClick : function(){
32172         this[this.collapsed ? "expand" : "collapse"]();
32173     },
32174
32175     /**
32176      * Collapses the dialog to its minimized state (only the title bar is visible).
32177      * Equivalent to the user clicking the collapse dialog button.
32178      */
32179     collapse : function(){
32180         if(!this.collapsed){
32181             this.collapsed = true;
32182             this.el.addClass("x-dlg-collapsed");
32183             this.restoreHeight = this.el.getHeight();
32184             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32185         }
32186     },
32187
32188     /**
32189      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32190      * clicking the expand dialog button.
32191      */
32192     expand : function(){
32193         if(this.collapsed){
32194             this.collapsed = false;
32195             this.el.removeClass("x-dlg-collapsed");
32196             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32197         }
32198     },
32199
32200     /**
32201      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32202      * @return {Roo.TabPanel} The tabs component
32203      */
32204     initTabs : function(){
32205         var tabs = this.getTabs();
32206         while(tabs.getTab(0)){
32207             tabs.removeTab(0);
32208         }
32209         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32210             var dom = el.dom;
32211             tabs.addTab(Roo.id(dom), dom.title);
32212             dom.title = "";
32213         });
32214         tabs.activate(0);
32215         return tabs;
32216     },
32217
32218     // private
32219     beforeResize : function(){
32220         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32221     },
32222
32223     // private
32224     onResize : function(){
32225         this.refreshSize();
32226         this.syncBodyHeight();
32227         this.adjustAssets();
32228         this.focus();
32229         this.fireEvent("resize", this, this.size.width, this.size.height);
32230     },
32231
32232     // private
32233     onKeyDown : function(e){
32234         if(this.isVisible()){
32235             this.fireEvent("keydown", this, e);
32236         }
32237     },
32238
32239     /**
32240      * Resizes the dialog.
32241      * @param {Number} width
32242      * @param {Number} height
32243      * @return {Roo.BasicDialog} this
32244      */
32245     resizeTo : function(width, height){
32246         this.el.setSize(width, height);
32247         this.size = {width: width, height: height};
32248         this.syncBodyHeight();
32249         if(this.fixedcenter){
32250             this.center();
32251         }
32252         if(this.isVisible()){
32253             this.constrainXY();
32254             this.adjustAssets();
32255         }
32256         this.fireEvent("resize", this, width, height);
32257         return this;
32258     },
32259
32260
32261     /**
32262      * Resizes the dialog to fit the specified content size.
32263      * @param {Number} width
32264      * @param {Number} height
32265      * @return {Roo.BasicDialog} this
32266      */
32267     setContentSize : function(w, h){
32268         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32269         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32270         //if(!this.el.isBorderBox()){
32271             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32272             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32273         //}
32274         if(this.tabs){
32275             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32276             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32277         }
32278         this.resizeTo(w, h);
32279         return this;
32280     },
32281
32282     /**
32283      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32284      * executed in response to a particular key being pressed while the dialog is active.
32285      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32286      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32287      * @param {Function} fn The function to call
32288      * @param {Object} scope (optional) The scope of the function
32289      * @return {Roo.BasicDialog} this
32290      */
32291     addKeyListener : function(key, fn, scope){
32292         var keyCode, shift, ctrl, alt;
32293         if(typeof key == "object" && !(key instanceof Array)){
32294             keyCode = key["key"];
32295             shift = key["shift"];
32296             ctrl = key["ctrl"];
32297             alt = key["alt"];
32298         }else{
32299             keyCode = key;
32300         }
32301         var handler = function(dlg, e){
32302             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32303                 var k = e.getKey();
32304                 if(keyCode instanceof Array){
32305                     for(var i = 0, len = keyCode.length; i < len; i++){
32306                         if(keyCode[i] == k){
32307                           fn.call(scope || window, dlg, k, e);
32308                           return;
32309                         }
32310                     }
32311                 }else{
32312                     if(k == keyCode){
32313                         fn.call(scope || window, dlg, k, e);
32314                     }
32315                 }
32316             }
32317         };
32318         this.on("keydown", handler);
32319         return this;
32320     },
32321
32322     /**
32323      * Returns the TabPanel component (creates it if it doesn't exist).
32324      * Note: If you wish to simply check for the existence of tabs without creating them,
32325      * check for a null 'tabs' property.
32326      * @return {Roo.TabPanel} The tabs component
32327      */
32328     getTabs : function(){
32329         if(!this.tabs){
32330             this.el.addClass("x-dlg-auto-tabs");
32331             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32332             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32333         }
32334         return this.tabs;
32335     },
32336
32337     /**
32338      * Adds a button to the footer section of the dialog.
32339      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32340      * object or a valid Roo.DomHelper element config
32341      * @param {Function} handler The function called when the button is clicked
32342      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32343      * @return {Roo.Button} The new button
32344      */
32345     addButton : function(config, handler, scope){
32346         var dh = Roo.DomHelper;
32347         if(!this.footer){
32348             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32349         }
32350         if(!this.btnContainer){
32351             var tb = this.footer.createChild({
32352
32353                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32354                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32355             }, null, true);
32356             this.btnContainer = tb.firstChild.firstChild.firstChild;
32357         }
32358         var bconfig = {
32359             handler: handler,
32360             scope: scope,
32361             minWidth: this.minButtonWidth,
32362             hideParent:true
32363         };
32364         if(typeof config == "string"){
32365             bconfig.text = config;
32366         }else{
32367             if(config.tag){
32368                 bconfig.dhconfig = config;
32369             }else{
32370                 Roo.apply(bconfig, config);
32371             }
32372         }
32373         var fc = false;
32374         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32375             bconfig.position = Math.max(0, bconfig.position);
32376             fc = this.btnContainer.childNodes[bconfig.position];
32377         }
32378          
32379         var btn = new Roo.Button(
32380             fc ? 
32381                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32382                 : this.btnContainer.appendChild(document.createElement("td")),
32383             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32384             bconfig
32385         );
32386         this.syncBodyHeight();
32387         if(!this.buttons){
32388             /**
32389              * Array of all the buttons that have been added to this dialog via addButton
32390              * @type Array
32391              */
32392             this.buttons = [];
32393         }
32394         this.buttons.push(btn);
32395         return btn;
32396     },
32397
32398     /**
32399      * Sets the default button to be focused when the dialog is displayed.
32400      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32401      * @return {Roo.BasicDialog} this
32402      */
32403     setDefaultButton : function(btn){
32404         this.defaultButton = btn;
32405         return this;
32406     },
32407
32408     // private
32409     getHeaderFooterHeight : function(safe){
32410         var height = 0;
32411         if(this.header){
32412            height += this.header.getHeight();
32413         }
32414         if(this.footer){
32415            var fm = this.footer.getMargins();
32416             height += (this.footer.getHeight()+fm.top+fm.bottom);
32417         }
32418         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32419         height += this.centerBg.getPadding("tb");
32420         return height;
32421     },
32422
32423     // private
32424     syncBodyHeight : function()
32425     {
32426         var bd = this.body, // the text
32427             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32428             bw = this.bwrap;
32429         var height = this.size.height - this.getHeaderFooterHeight(false);
32430         bd.setHeight(height-bd.getMargins("tb"));
32431         var hh = this.header.getHeight();
32432         var h = this.size.height-hh;
32433         cb.setHeight(h);
32434         
32435         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32436         bw.setHeight(h-cb.getPadding("tb"));
32437         
32438         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32439         bd.setWidth(bw.getWidth(true));
32440         if(this.tabs){
32441             this.tabs.syncHeight();
32442             if(Roo.isIE){
32443                 this.tabs.el.repaint();
32444             }
32445         }
32446     },
32447
32448     /**
32449      * Restores the previous state of the dialog if Roo.state is configured.
32450      * @return {Roo.BasicDialog} this
32451      */
32452     restoreState : function(){
32453         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32454         if(box && box.width){
32455             this.xy = [box.x, box.y];
32456             this.resizeTo(box.width, box.height);
32457         }
32458         return this;
32459     },
32460
32461     // private
32462     beforeShow : function(){
32463         this.expand();
32464         if(this.fixedcenter){
32465             this.xy = this.el.getCenterXY(true);
32466         }
32467         if(this.modal){
32468             Roo.get(document.body).addClass("x-body-masked");
32469             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32470             this.mask.show();
32471         }
32472         this.constrainXY();
32473     },
32474
32475     // private
32476     animShow : function(){
32477         var b = Roo.get(this.animateTarget).getBox();
32478         this.proxy.setSize(b.width, b.height);
32479         this.proxy.setLocation(b.x, b.y);
32480         this.proxy.show();
32481         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32482                     true, .35, this.showEl.createDelegate(this));
32483     },
32484
32485     /**
32486      * Shows the dialog.
32487      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32488      * @return {Roo.BasicDialog} this
32489      */
32490     show : function(animateTarget){
32491         if (this.fireEvent("beforeshow", this) === false){
32492             return;
32493         }
32494         if(this.syncHeightBeforeShow){
32495             this.syncBodyHeight();
32496         }else if(this.firstShow){
32497             this.firstShow = false;
32498             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32499         }
32500         this.animateTarget = animateTarget || this.animateTarget;
32501         if(!this.el.isVisible()){
32502             this.beforeShow();
32503             if(this.animateTarget && Roo.get(this.animateTarget)){
32504                 this.animShow();
32505             }else{
32506                 this.showEl();
32507             }
32508         }
32509         return this;
32510     },
32511
32512     // private
32513     showEl : function(){
32514         this.proxy.hide();
32515         this.el.setXY(this.xy);
32516         this.el.show();
32517         this.adjustAssets(true);
32518         this.toFront();
32519         this.focus();
32520         // IE peekaboo bug - fix found by Dave Fenwick
32521         if(Roo.isIE){
32522             this.el.repaint();
32523         }
32524         this.fireEvent("show", this);
32525     },
32526
32527     /**
32528      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32529      * dialog itself will receive focus.
32530      */
32531     focus : function(){
32532         if(this.defaultButton){
32533             this.defaultButton.focus();
32534         }else{
32535             this.focusEl.focus();
32536         }
32537     },
32538
32539     // private
32540     constrainXY : function(){
32541         if(this.constraintoviewport !== false){
32542             if(!this.viewSize){
32543                 if(this.container){
32544                     var s = this.container.getSize();
32545                     this.viewSize = [s.width, s.height];
32546                 }else{
32547                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32548                 }
32549             }
32550             var s = Roo.get(this.container||document).getScroll();
32551
32552             var x = this.xy[0], y = this.xy[1];
32553             var w = this.size.width, h = this.size.height;
32554             var vw = this.viewSize[0], vh = this.viewSize[1];
32555             // only move it if it needs it
32556             var moved = false;
32557             // first validate right/bottom
32558             if(x + w > vw+s.left){
32559                 x = vw - w;
32560                 moved = true;
32561             }
32562             if(y + h > vh+s.top){
32563                 y = vh - h;
32564                 moved = true;
32565             }
32566             // then make sure top/left isn't negative
32567             if(x < s.left){
32568                 x = s.left;
32569                 moved = true;
32570             }
32571             if(y < s.top){
32572                 y = s.top;
32573                 moved = true;
32574             }
32575             if(moved){
32576                 // cache xy
32577                 this.xy = [x, y];
32578                 if(this.isVisible()){
32579                     this.el.setLocation(x, y);
32580                     this.adjustAssets();
32581                 }
32582             }
32583         }
32584     },
32585
32586     // private
32587     onDrag : function(){
32588         if(!this.proxyDrag){
32589             this.xy = this.el.getXY();
32590             this.adjustAssets();
32591         }
32592     },
32593
32594     // private
32595     adjustAssets : function(doShow){
32596         var x = this.xy[0], y = this.xy[1];
32597         var w = this.size.width, h = this.size.height;
32598         if(doShow === true){
32599             if(this.shadow){
32600                 this.shadow.show(this.el);
32601             }
32602             if(this.shim){
32603                 this.shim.show();
32604             }
32605         }
32606         if(this.shadow && this.shadow.isVisible()){
32607             this.shadow.show(this.el);
32608         }
32609         if(this.shim && this.shim.isVisible()){
32610             this.shim.setBounds(x, y, w, h);
32611         }
32612     },
32613
32614     // private
32615     adjustViewport : function(w, h){
32616         if(!w || !h){
32617             w = Roo.lib.Dom.getViewWidth();
32618             h = Roo.lib.Dom.getViewHeight();
32619         }
32620         // cache the size
32621         this.viewSize = [w, h];
32622         if(this.modal && this.mask.isVisible()){
32623             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32624             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32625         }
32626         if(this.isVisible()){
32627             this.constrainXY();
32628         }
32629     },
32630
32631     /**
32632      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32633      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32634      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32635      */
32636     destroy : function(removeEl){
32637         if(this.isVisible()){
32638             this.animateTarget = null;
32639             this.hide();
32640         }
32641         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32642         if(this.tabs){
32643             this.tabs.destroy(removeEl);
32644         }
32645         Roo.destroy(
32646              this.shim,
32647              this.proxy,
32648              this.resizer,
32649              this.close,
32650              this.mask
32651         );
32652         if(this.dd){
32653             this.dd.unreg();
32654         }
32655         if(this.buttons){
32656            for(var i = 0, len = this.buttons.length; i < len; i++){
32657                this.buttons[i].destroy();
32658            }
32659         }
32660         this.el.removeAllListeners();
32661         if(removeEl === true){
32662             this.el.update("");
32663             this.el.remove();
32664         }
32665         Roo.DialogManager.unregister(this);
32666     },
32667
32668     // private
32669     startMove : function(){
32670         if(this.proxyDrag){
32671             this.proxy.show();
32672         }
32673         if(this.constraintoviewport !== false){
32674             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32675         }
32676     },
32677
32678     // private
32679     endMove : function(){
32680         if(!this.proxyDrag){
32681             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32682         }else{
32683             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32684             this.proxy.hide();
32685         }
32686         this.refreshSize();
32687         this.adjustAssets();
32688         this.focus();
32689         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32690     },
32691
32692     /**
32693      * Brings this dialog to the front of any other visible dialogs
32694      * @return {Roo.BasicDialog} this
32695      */
32696     toFront : function(){
32697         Roo.DialogManager.bringToFront(this);
32698         return this;
32699     },
32700
32701     /**
32702      * Sends this dialog to the back (under) of any other visible dialogs
32703      * @return {Roo.BasicDialog} this
32704      */
32705     toBack : function(){
32706         Roo.DialogManager.sendToBack(this);
32707         return this;
32708     },
32709
32710     /**
32711      * Centers this dialog in the viewport
32712      * @return {Roo.BasicDialog} this
32713      */
32714     center : function(){
32715         var xy = this.el.getCenterXY(true);
32716         this.moveTo(xy[0], xy[1]);
32717         return this;
32718     },
32719
32720     /**
32721      * Moves the dialog's top-left corner to the specified point
32722      * @param {Number} x
32723      * @param {Number} y
32724      * @return {Roo.BasicDialog} this
32725      */
32726     moveTo : function(x, y){
32727         this.xy = [x,y];
32728         if(this.isVisible()){
32729             this.el.setXY(this.xy);
32730             this.adjustAssets();
32731         }
32732         return this;
32733     },
32734
32735     /**
32736      * Aligns the dialog to the specified element
32737      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32738      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32739      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32740      * @return {Roo.BasicDialog} this
32741      */
32742     alignTo : function(element, position, offsets){
32743         this.xy = this.el.getAlignToXY(element, position, offsets);
32744         if(this.isVisible()){
32745             this.el.setXY(this.xy);
32746             this.adjustAssets();
32747         }
32748         return this;
32749     },
32750
32751     /**
32752      * Anchors an element to another element and realigns it when the window is resized.
32753      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32754      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32755      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32756      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32757      * is a number, it is used as the buffer delay (defaults to 50ms).
32758      * @return {Roo.BasicDialog} this
32759      */
32760     anchorTo : function(el, alignment, offsets, monitorScroll){
32761         var action = function(){
32762             this.alignTo(el, alignment, offsets);
32763         };
32764         Roo.EventManager.onWindowResize(action, this);
32765         var tm = typeof monitorScroll;
32766         if(tm != 'undefined'){
32767             Roo.EventManager.on(window, 'scroll', action, this,
32768                 {buffer: tm == 'number' ? monitorScroll : 50});
32769         }
32770         action.call(this);
32771         return this;
32772     },
32773
32774     /**
32775      * Returns true if the dialog is visible
32776      * @return {Boolean}
32777      */
32778     isVisible : function(){
32779         return this.el.isVisible();
32780     },
32781
32782     // private
32783     animHide : function(callback){
32784         var b = Roo.get(this.animateTarget).getBox();
32785         this.proxy.show();
32786         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32787         this.el.hide();
32788         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32789                     this.hideEl.createDelegate(this, [callback]));
32790     },
32791
32792     /**
32793      * Hides the dialog.
32794      * @param {Function} callback (optional) Function to call when the dialog is hidden
32795      * @return {Roo.BasicDialog} this
32796      */
32797     hide : function(callback){
32798         if (this.fireEvent("beforehide", this) === false){
32799             return;
32800         }
32801         if(this.shadow){
32802             this.shadow.hide();
32803         }
32804         if(this.shim) {
32805           this.shim.hide();
32806         }
32807         // sometimes animateTarget seems to get set.. causing problems...
32808         // this just double checks..
32809         if(this.animateTarget && Roo.get(this.animateTarget)) {
32810            this.animHide(callback);
32811         }else{
32812             this.el.hide();
32813             this.hideEl(callback);
32814         }
32815         return this;
32816     },
32817
32818     // private
32819     hideEl : function(callback){
32820         this.proxy.hide();
32821         if(this.modal){
32822             this.mask.hide();
32823             Roo.get(document.body).removeClass("x-body-masked");
32824         }
32825         this.fireEvent("hide", this);
32826         if(typeof callback == "function"){
32827             callback();
32828         }
32829     },
32830
32831     // private
32832     hideAction : function(){
32833         this.setLeft("-10000px");
32834         this.setTop("-10000px");
32835         this.setStyle("visibility", "hidden");
32836     },
32837
32838     // private
32839     refreshSize : function(){
32840         this.size = this.el.getSize();
32841         this.xy = this.el.getXY();
32842         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32843     },
32844
32845     // private
32846     // z-index is managed by the DialogManager and may be overwritten at any time
32847     setZIndex : function(index){
32848         if(this.modal){
32849             this.mask.setStyle("z-index", index);
32850         }
32851         if(this.shim){
32852             this.shim.setStyle("z-index", ++index);
32853         }
32854         if(this.shadow){
32855             this.shadow.setZIndex(++index);
32856         }
32857         this.el.setStyle("z-index", ++index);
32858         if(this.proxy){
32859             this.proxy.setStyle("z-index", ++index);
32860         }
32861         if(this.resizer){
32862             this.resizer.proxy.setStyle("z-index", ++index);
32863         }
32864
32865         this.lastZIndex = index;
32866     },
32867
32868     /**
32869      * Returns the element for this dialog
32870      * @return {Roo.Element} The underlying dialog Element
32871      */
32872     getEl : function(){
32873         return this.el;
32874     }
32875 });
32876
32877 /**
32878  * @class Roo.DialogManager
32879  * Provides global access to BasicDialogs that have been created and
32880  * support for z-indexing (layering) multiple open dialogs.
32881  */
32882 Roo.DialogManager = function(){
32883     var list = {};
32884     var accessList = [];
32885     var front = null;
32886
32887     // private
32888     var sortDialogs = function(d1, d2){
32889         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32890     };
32891
32892     // private
32893     var orderDialogs = function(){
32894         accessList.sort(sortDialogs);
32895         var seed = Roo.DialogManager.zseed;
32896         for(var i = 0, len = accessList.length; i < len; i++){
32897             var dlg = accessList[i];
32898             if(dlg){
32899                 dlg.setZIndex(seed + (i*10));
32900             }
32901         }
32902     };
32903
32904     return {
32905         /**
32906          * The starting z-index for BasicDialogs (defaults to 9000)
32907          * @type Number The z-index value
32908          */
32909         zseed : 9000,
32910
32911         // private
32912         register : function(dlg){
32913             list[dlg.id] = dlg;
32914             accessList.push(dlg);
32915         },
32916
32917         // private
32918         unregister : function(dlg){
32919             delete list[dlg.id];
32920             var i=0;
32921             var len=0;
32922             if(!accessList.indexOf){
32923                 for(  i = 0, len = accessList.length; i < len; i++){
32924                     if(accessList[i] == dlg){
32925                         accessList.splice(i, 1);
32926                         return;
32927                     }
32928                 }
32929             }else{
32930                  i = accessList.indexOf(dlg);
32931                 if(i != -1){
32932                     accessList.splice(i, 1);
32933                 }
32934             }
32935         },
32936
32937         /**
32938          * Gets a registered dialog by id
32939          * @param {String/Object} id The id of the dialog or a dialog
32940          * @return {Roo.BasicDialog} this
32941          */
32942         get : function(id){
32943             return typeof id == "object" ? id : list[id];
32944         },
32945
32946         /**
32947          * Brings the specified dialog to the front
32948          * @param {String/Object} dlg The id of the dialog or a dialog
32949          * @return {Roo.BasicDialog} this
32950          */
32951         bringToFront : function(dlg){
32952             dlg = this.get(dlg);
32953             if(dlg != front){
32954                 front = dlg;
32955                 dlg._lastAccess = new Date().getTime();
32956                 orderDialogs();
32957             }
32958             return dlg;
32959         },
32960
32961         /**
32962          * Sends the specified dialog to the back
32963          * @param {String/Object} dlg The id of the dialog or a dialog
32964          * @return {Roo.BasicDialog} this
32965          */
32966         sendToBack : function(dlg){
32967             dlg = this.get(dlg);
32968             dlg._lastAccess = -(new Date().getTime());
32969             orderDialogs();
32970             return dlg;
32971         },
32972
32973         /**
32974          * Hides all dialogs
32975          */
32976         hideAll : function(){
32977             for(var id in list){
32978                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
32979                     list[id].hide();
32980                 }
32981             }
32982         }
32983     };
32984 }();
32985
32986 /**
32987  * @class Roo.LayoutDialog
32988  * @extends Roo.BasicDialog
32989  * Dialog which provides adjustments for working with a layout in a Dialog.
32990  * Add your necessary layout config options to the dialog's config.<br>
32991  * Example usage (including a nested layout):
32992  * <pre><code>
32993 if(!dialog){
32994     dialog = new Roo.LayoutDialog("download-dlg", {
32995         modal: true,
32996         width:600,
32997         height:450,
32998         shadow:true,
32999         minWidth:500,
33000         minHeight:350,
33001         autoTabs:true,
33002         proxyDrag:true,
33003         // layout config merges with the dialog config
33004         center:{
33005             tabPosition: "top",
33006             alwaysShowTabs: true
33007         }
33008     });
33009     dialog.addKeyListener(27, dialog.hide, dialog);
33010     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33011     dialog.addButton("Build It!", this.getDownload, this);
33012
33013     // we can even add nested layouts
33014     var innerLayout = new Roo.BorderLayout("dl-inner", {
33015         east: {
33016             initialSize: 200,
33017             autoScroll:true,
33018             split:true
33019         },
33020         center: {
33021             autoScroll:true
33022         }
33023     });
33024     innerLayout.beginUpdate();
33025     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33026     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33027     innerLayout.endUpdate(true);
33028
33029     var layout = dialog.getLayout();
33030     layout.beginUpdate();
33031     layout.add("center", new Roo.ContentPanel("standard-panel",
33032                         {title: "Download the Source", fitToFrame:true}));
33033     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33034                {title: "Build your own roo.js"}));
33035     layout.getRegion("center").showPanel(sp);
33036     layout.endUpdate();
33037 }
33038 </code></pre>
33039     * @constructor
33040     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33041     * @param {Object} config configuration options
33042   */
33043 Roo.LayoutDialog = function(el, cfg){
33044     
33045     var config=  cfg;
33046     if (typeof(cfg) == 'undefined') {
33047         config = Roo.apply({}, el);
33048         // not sure why we use documentElement here.. - it should always be body.
33049         // IE7 borks horribly if we use documentElement.
33050         // webkit also does not like documentElement - it creates a body element...
33051         el = Roo.get( document.body || document.documentElement ).createChild();
33052         //config.autoCreate = true;
33053     }
33054     
33055     
33056     config.autoTabs = false;
33057     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33058     this.body.setStyle({overflow:"hidden", position:"relative"});
33059     this.layout = new Roo.BorderLayout(this.body.dom, config);
33060     this.layout.monitorWindowResize = false;
33061     this.el.addClass("x-dlg-auto-layout");
33062     // fix case when center region overwrites center function
33063     this.center = Roo.BasicDialog.prototype.center;
33064     this.on("show", this.layout.layout, this.layout, true);
33065     if (config.items) {
33066         var xitems = config.items;
33067         delete config.items;
33068         Roo.each(xitems, this.addxtype, this);
33069     }
33070     
33071     
33072 };
33073 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33074     /**
33075      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33076      * @deprecated
33077      */
33078     endUpdate : function(){
33079         this.layout.endUpdate();
33080     },
33081
33082     /**
33083      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33084      *  @deprecated
33085      */
33086     beginUpdate : function(){
33087         this.layout.beginUpdate();
33088     },
33089
33090     /**
33091      * Get the BorderLayout for this dialog
33092      * @return {Roo.BorderLayout}
33093      */
33094     getLayout : function(){
33095         return this.layout;
33096     },
33097
33098     showEl : function(){
33099         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33100         if(Roo.isIE7){
33101             this.layout.layout();
33102         }
33103     },
33104
33105     // private
33106     // Use the syncHeightBeforeShow config option to control this automatically
33107     syncBodyHeight : function(){
33108         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33109         if(this.layout){this.layout.layout();}
33110     },
33111     
33112       /**
33113      * Add an xtype element (actually adds to the layout.)
33114      * @return {Object} xdata xtype object data.
33115      */
33116     
33117     addxtype : function(c) {
33118         return this.layout.addxtype(c);
33119     }
33120 });/*
33121  * Based on:
33122  * Ext JS Library 1.1.1
33123  * Copyright(c) 2006-2007, Ext JS, LLC.
33124  *
33125  * Originally Released Under LGPL - original licence link has changed is not relivant.
33126  *
33127  * Fork - LGPL
33128  * <script type="text/javascript">
33129  */
33130  
33131 /**
33132  * @class Roo.MessageBox
33133  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33134  * Example usage:
33135  *<pre><code>
33136 // Basic alert:
33137 Roo.Msg.alert('Status', 'Changes saved successfully.');
33138
33139 // Prompt for user data:
33140 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33141     if (btn == 'ok'){
33142         // process text value...
33143     }
33144 });
33145
33146 // Show a dialog using config options:
33147 Roo.Msg.show({
33148    title:'Save Changes?',
33149    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33150    buttons: Roo.Msg.YESNOCANCEL,
33151    fn: processResult,
33152    animEl: 'elId'
33153 });
33154 </code></pre>
33155  * @singleton
33156  */
33157 Roo.MessageBox = function(){
33158     var dlg, opt, mask, waitTimer;
33159     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33160     var buttons, activeTextEl, bwidth;
33161
33162     // private
33163     var handleButton = function(button){
33164         dlg.hide();
33165         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33166     };
33167
33168     // private
33169     var handleHide = function(){
33170         if(opt && opt.cls){
33171             dlg.el.removeClass(opt.cls);
33172         }
33173         if(waitTimer){
33174             Roo.TaskMgr.stop(waitTimer);
33175             waitTimer = null;
33176         }
33177     };
33178
33179     // private
33180     var updateButtons = function(b){
33181         var width = 0;
33182         if(!b){
33183             buttons["ok"].hide();
33184             buttons["cancel"].hide();
33185             buttons["yes"].hide();
33186             buttons["no"].hide();
33187             dlg.footer.dom.style.display = 'none';
33188             return width;
33189         }
33190         dlg.footer.dom.style.display = '';
33191         for(var k in buttons){
33192             if(typeof buttons[k] != "function"){
33193                 if(b[k]){
33194                     buttons[k].show();
33195                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33196                     width += buttons[k].el.getWidth()+15;
33197                 }else{
33198                     buttons[k].hide();
33199                 }
33200             }
33201         }
33202         return width;
33203     };
33204
33205     // private
33206     var handleEsc = function(d, k, e){
33207         if(opt && opt.closable !== false){
33208             dlg.hide();
33209         }
33210         if(e){
33211             e.stopEvent();
33212         }
33213     };
33214
33215     return {
33216         /**
33217          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33218          * @return {Roo.BasicDialog} The BasicDialog element
33219          */
33220         getDialog : function(){
33221            if(!dlg){
33222                 dlg = new Roo.BasicDialog("x-msg-box", {
33223                     autoCreate : true,
33224                     shadow: true,
33225                     draggable: true,
33226                     resizable:false,
33227                     constraintoviewport:false,
33228                     fixedcenter:true,
33229                     collapsible : false,
33230                     shim:true,
33231                     modal: true,
33232                     width:400, height:100,
33233                     buttonAlign:"center",
33234                     closeClick : function(){
33235                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33236                             handleButton("no");
33237                         }else{
33238                             handleButton("cancel");
33239                         }
33240                     }
33241                 });
33242                 dlg.on("hide", handleHide);
33243                 mask = dlg.mask;
33244                 dlg.addKeyListener(27, handleEsc);
33245                 buttons = {};
33246                 var bt = this.buttonText;
33247                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33248                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33249                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33250                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33251                 bodyEl = dlg.body.createChild({
33252
33253                     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>'
33254                 });
33255                 msgEl = bodyEl.dom.firstChild;
33256                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33257                 textboxEl.enableDisplayMode();
33258                 textboxEl.addKeyListener([10,13], function(){
33259                     if(dlg.isVisible() && opt && opt.buttons){
33260                         if(opt.buttons.ok){
33261                             handleButton("ok");
33262                         }else if(opt.buttons.yes){
33263                             handleButton("yes");
33264                         }
33265                     }
33266                 });
33267                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33268                 textareaEl.enableDisplayMode();
33269                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33270                 progressEl.enableDisplayMode();
33271                 var pf = progressEl.dom.firstChild;
33272                 if (pf) {
33273                     pp = Roo.get(pf.firstChild);
33274                     pp.setHeight(pf.offsetHeight);
33275                 }
33276                 
33277             }
33278             return dlg;
33279         },
33280
33281         /**
33282          * Updates the message box body text
33283          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33284          * the XHTML-compliant non-breaking space character '&amp;#160;')
33285          * @return {Roo.MessageBox} This message box
33286          */
33287         updateText : function(text){
33288             if(!dlg.isVisible() && !opt.width){
33289                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33290             }
33291             msgEl.innerHTML = text || '&#160;';
33292       
33293             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33294             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33295             var w = Math.max(
33296                     Math.min(opt.width || cw , this.maxWidth), 
33297                     Math.max(opt.minWidth || this.minWidth, bwidth)
33298             );
33299             if(opt.prompt){
33300                 activeTextEl.setWidth(w);
33301             }
33302             if(dlg.isVisible()){
33303                 dlg.fixedcenter = false;
33304             }
33305             // to big, make it scroll. = But as usual stupid IE does not support
33306             // !important..
33307             
33308             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33309                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33310                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33311             } else {
33312                 bodyEl.dom.style.height = '';
33313                 bodyEl.dom.style.overflowY = '';
33314             }
33315             if (cw > w) {
33316                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33317             } else {
33318                 bodyEl.dom.style.overflowX = '';
33319             }
33320             
33321             dlg.setContentSize(w, bodyEl.getHeight());
33322             if(dlg.isVisible()){
33323                 dlg.fixedcenter = true;
33324             }
33325             return this;
33326         },
33327
33328         /**
33329          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33330          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33331          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33332          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33333          * @return {Roo.MessageBox} This message box
33334          */
33335         updateProgress : function(value, text){
33336             if(text){
33337                 this.updateText(text);
33338             }
33339             if (pp) { // weird bug on my firefox - for some reason this is not defined
33340                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33341             }
33342             return this;
33343         },        
33344
33345         /**
33346          * Returns true if the message box is currently displayed
33347          * @return {Boolean} True if the message box is visible, else false
33348          */
33349         isVisible : function(){
33350             return dlg && dlg.isVisible();  
33351         },
33352
33353         /**
33354          * Hides the message box if it is displayed
33355          */
33356         hide : function(){
33357             if(this.isVisible()){
33358                 dlg.hide();
33359             }  
33360         },
33361
33362         /**
33363          * Displays a new message box, or reinitializes an existing message box, based on the config options
33364          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33365          * The following config object properties are supported:
33366          * <pre>
33367 Property    Type             Description
33368 ----------  ---------------  ------------------------------------------------------------------------------------
33369 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33370                                    closes (defaults to undefined)
33371 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33372                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33373 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33374                                    progress and wait dialogs will ignore this property and always hide the
33375                                    close button as they can only be closed programmatically.
33376 cls               String           A custom CSS class to apply to the message box element
33377 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33378                                    displayed (defaults to 75)
33379 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33380                                    function will be btn (the name of the button that was clicked, if applicable,
33381                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33382                                    Progress and wait dialogs will ignore this option since they do not respond to
33383                                    user actions and can only be closed programmatically, so any required function
33384                                    should be called by the same code after it closes the dialog.
33385 icon              String           A CSS class that provides a background image to be used as an icon for
33386                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33387 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33388 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33389 modal             Boolean          False to allow user interaction with the page while the message box is
33390                                    displayed (defaults to true)
33391 msg               String           A string that will replace the existing message box body text (defaults
33392                                    to the XHTML-compliant non-breaking space character '&#160;')
33393 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33394 progress          Boolean          True to display a progress bar (defaults to false)
33395 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33396 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33397 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33398 title             String           The title text
33399 value             String           The string value to set into the active textbox element if displayed
33400 wait              Boolean          True to display a progress bar (defaults to false)
33401 width             Number           The width of the dialog in pixels
33402 </pre>
33403          *
33404          * Example usage:
33405          * <pre><code>
33406 Roo.Msg.show({
33407    title: 'Address',
33408    msg: 'Please enter your address:',
33409    width: 300,
33410    buttons: Roo.MessageBox.OKCANCEL,
33411    multiline: true,
33412    fn: saveAddress,
33413    animEl: 'addAddressBtn'
33414 });
33415 </code></pre>
33416          * @param {Object} config Configuration options
33417          * @return {Roo.MessageBox} This message box
33418          */
33419         show : function(options)
33420         {
33421             
33422             // this causes nightmares if you show one dialog after another
33423             // especially on callbacks..
33424              
33425             if(this.isVisible()){
33426                 
33427                 this.hide();
33428                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33429                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33430                 Roo.log("New Dialog Message:" +  options.msg )
33431                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33432                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33433                 
33434             }
33435             var d = this.getDialog();
33436             opt = options;
33437             d.setTitle(opt.title || "&#160;");
33438             d.close.setDisplayed(opt.closable !== false);
33439             activeTextEl = textboxEl;
33440             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33441             if(opt.prompt){
33442                 if(opt.multiline){
33443                     textboxEl.hide();
33444                     textareaEl.show();
33445                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33446                         opt.multiline : this.defaultTextHeight);
33447                     activeTextEl = textareaEl;
33448                 }else{
33449                     textboxEl.show();
33450                     textareaEl.hide();
33451                 }
33452             }else{
33453                 textboxEl.hide();
33454                 textareaEl.hide();
33455             }
33456             progressEl.setDisplayed(opt.progress === true);
33457             this.updateProgress(0);
33458             activeTextEl.dom.value = opt.value || "";
33459             if(opt.prompt){
33460                 dlg.setDefaultButton(activeTextEl);
33461             }else{
33462                 var bs = opt.buttons;
33463                 var db = null;
33464                 if(bs && bs.ok){
33465                     db = buttons["ok"];
33466                 }else if(bs && bs.yes){
33467                     db = buttons["yes"];
33468                 }
33469                 dlg.setDefaultButton(db);
33470             }
33471             bwidth = updateButtons(opt.buttons);
33472             this.updateText(opt.msg);
33473             if(opt.cls){
33474                 d.el.addClass(opt.cls);
33475             }
33476             d.proxyDrag = opt.proxyDrag === true;
33477             d.modal = opt.modal !== false;
33478             d.mask = opt.modal !== false ? mask : false;
33479             if(!d.isVisible()){
33480                 // force it to the end of the z-index stack so it gets a cursor in FF
33481                 document.body.appendChild(dlg.el.dom);
33482                 d.animateTarget = null;
33483                 d.show(options.animEl);
33484             }
33485             return this;
33486         },
33487
33488         /**
33489          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33490          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33491          * and closing the message box when the process is complete.
33492          * @param {String} title The title bar text
33493          * @param {String} msg The message box body text
33494          * @return {Roo.MessageBox} This message box
33495          */
33496         progress : function(title, msg){
33497             this.show({
33498                 title : title,
33499                 msg : msg,
33500                 buttons: false,
33501                 progress:true,
33502                 closable:false,
33503                 minWidth: this.minProgressWidth,
33504                 modal : true
33505             });
33506             return this;
33507         },
33508
33509         /**
33510          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33511          * If a callback function is passed it will be called after the user clicks the button, and the
33512          * id of the button that was clicked will be passed as the only parameter to the callback
33513          * (could also be the top-right close button).
33514          * @param {String} title The title bar text
33515          * @param {String} msg The message box body text
33516          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33517          * @param {Object} scope (optional) The scope of the callback function
33518          * @return {Roo.MessageBox} This message box
33519          */
33520         alert : function(title, msg, fn, scope){
33521             this.show({
33522                 title : title,
33523                 msg : msg,
33524                 buttons: this.OK,
33525                 fn: fn,
33526                 scope : scope,
33527                 modal : true
33528             });
33529             return this;
33530         },
33531
33532         /**
33533          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33534          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33535          * You are responsible for closing the message box when the process is complete.
33536          * @param {String} msg The message box body text
33537          * @param {String} title (optional) The title bar text
33538          * @return {Roo.MessageBox} This message box
33539          */
33540         wait : function(msg, title){
33541             this.show({
33542                 title : title,
33543                 msg : msg,
33544                 buttons: false,
33545                 closable:false,
33546                 progress:true,
33547                 modal:true,
33548                 width:300,
33549                 wait:true
33550             });
33551             waitTimer = Roo.TaskMgr.start({
33552                 run: function(i){
33553                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33554                 },
33555                 interval: 1000
33556             });
33557             return this;
33558         },
33559
33560         /**
33561          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33562          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33563          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33564          * @param {String} title The title bar text
33565          * @param {String} msg The message box body text
33566          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33567          * @param {Object} scope (optional) The scope of the callback function
33568          * @return {Roo.MessageBox} This message box
33569          */
33570         confirm : function(title, msg, fn, scope){
33571             this.show({
33572                 title : title,
33573                 msg : msg,
33574                 buttons: this.YESNO,
33575                 fn: fn,
33576                 scope : scope,
33577                 modal : true
33578             });
33579             return this;
33580         },
33581
33582         /**
33583          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33584          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33585          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33586          * (could also be the top-right close button) and the text that was entered will be passed as the two
33587          * parameters to the callback.
33588          * @param {String} title The title bar text
33589          * @param {String} msg The message box body text
33590          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33591          * @param {Object} scope (optional) The scope of the callback function
33592          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33593          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33594          * @return {Roo.MessageBox} This message box
33595          */
33596         prompt : function(title, msg, fn, scope, multiline){
33597             this.show({
33598                 title : title,
33599                 msg : msg,
33600                 buttons: this.OKCANCEL,
33601                 fn: fn,
33602                 minWidth:250,
33603                 scope : scope,
33604                 prompt:true,
33605                 multiline: multiline,
33606                 modal : true
33607             });
33608             return this;
33609         },
33610
33611         /**
33612          * Button config that displays a single OK button
33613          * @type Object
33614          */
33615         OK : {ok:true},
33616         /**
33617          * Button config that displays Yes and No buttons
33618          * @type Object
33619          */
33620         YESNO : {yes:true, no:true},
33621         /**
33622          * Button config that displays OK and Cancel buttons
33623          * @type Object
33624          */
33625         OKCANCEL : {ok:true, cancel:true},
33626         /**
33627          * Button config that displays Yes, No and Cancel buttons
33628          * @type Object
33629          */
33630         YESNOCANCEL : {yes:true, no:true, cancel:true},
33631
33632         /**
33633          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33634          * @type Number
33635          */
33636         defaultTextHeight : 75,
33637         /**
33638          * The maximum width in pixels of the message box (defaults to 600)
33639          * @type Number
33640          */
33641         maxWidth : 600,
33642         /**
33643          * The minimum width in pixels of the message box (defaults to 100)
33644          * @type Number
33645          */
33646         minWidth : 100,
33647         /**
33648          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33649          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33650          * @type Number
33651          */
33652         minProgressWidth : 250,
33653         /**
33654          * An object containing the default button text strings that can be overriden for localized language support.
33655          * Supported properties are: ok, cancel, yes and no.
33656          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33657          * @type Object
33658          */
33659         buttonText : {
33660             ok : "OK",
33661             cancel : "Cancel",
33662             yes : "Yes",
33663             no : "No"
33664         }
33665     };
33666 }();
33667
33668 /**
33669  * Shorthand for {@link Roo.MessageBox}
33670  */
33671 Roo.Msg = Roo.MessageBox;/*
33672  * Based on:
33673  * Ext JS Library 1.1.1
33674  * Copyright(c) 2006-2007, Ext JS, LLC.
33675  *
33676  * Originally Released Under LGPL - original licence link has changed is not relivant.
33677  *
33678  * Fork - LGPL
33679  * <script type="text/javascript">
33680  */
33681 /**
33682  * @class Roo.QuickTips
33683  * Provides attractive and customizable tooltips for any element.
33684  * @singleton
33685  */
33686 Roo.QuickTips = function(){
33687     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33688     var ce, bd, xy, dd;
33689     var visible = false, disabled = true, inited = false;
33690     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33691     
33692     var onOver = function(e){
33693         if(disabled){
33694             return;
33695         }
33696         var t = e.getTarget();
33697         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33698             return;
33699         }
33700         if(ce && t == ce.el){
33701             clearTimeout(hideProc);
33702             return;
33703         }
33704         if(t && tagEls[t.id]){
33705             tagEls[t.id].el = t;
33706             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33707             return;
33708         }
33709         var ttp, et = Roo.fly(t);
33710         var ns = cfg.namespace;
33711         if(tm.interceptTitles && t.title){
33712             ttp = t.title;
33713             t.qtip = ttp;
33714             t.removeAttribute("title");
33715             e.preventDefault();
33716         }else{
33717             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33718         }
33719         if(ttp){
33720             showProc = show.defer(tm.showDelay, tm, [{
33721                 el: t, 
33722                 text: ttp.replace(/\\n/g,'<br/>'),
33723                 width: et.getAttributeNS(ns, cfg.width),
33724                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33725                 title: et.getAttributeNS(ns, cfg.title),
33726                     cls: et.getAttributeNS(ns, cfg.cls)
33727             }]);
33728         }
33729     };
33730     
33731     var onOut = function(e){
33732         clearTimeout(showProc);
33733         var t = e.getTarget();
33734         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33735             hideProc = setTimeout(hide, tm.hideDelay);
33736         }
33737     };
33738     
33739     var onMove = function(e){
33740         if(disabled){
33741             return;
33742         }
33743         xy = e.getXY();
33744         xy[1] += 18;
33745         if(tm.trackMouse && ce){
33746             el.setXY(xy);
33747         }
33748     };
33749     
33750     var onDown = function(e){
33751         clearTimeout(showProc);
33752         clearTimeout(hideProc);
33753         if(!e.within(el)){
33754             if(tm.hideOnClick){
33755                 hide();
33756                 tm.disable();
33757                 tm.enable.defer(100, tm);
33758             }
33759         }
33760     };
33761     
33762     var getPad = function(){
33763         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33764     };
33765
33766     var show = function(o){
33767         if(disabled){
33768             return;
33769         }
33770         clearTimeout(dismissProc);
33771         ce = o;
33772         if(removeCls){ // in case manually hidden
33773             el.removeClass(removeCls);
33774             removeCls = null;
33775         }
33776         if(ce.cls){
33777             el.addClass(ce.cls);
33778             removeCls = ce.cls;
33779         }
33780         if(ce.title){
33781             tipTitle.update(ce.title);
33782             tipTitle.show();
33783         }else{
33784             tipTitle.update('');
33785             tipTitle.hide();
33786         }
33787         el.dom.style.width  = tm.maxWidth+'px';
33788         //tipBody.dom.style.width = '';
33789         tipBodyText.update(o.text);
33790         var p = getPad(), w = ce.width;
33791         if(!w){
33792             var td = tipBodyText.dom;
33793             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33794             if(aw > tm.maxWidth){
33795                 w = tm.maxWidth;
33796             }else if(aw < tm.minWidth){
33797                 w = tm.minWidth;
33798             }else{
33799                 w = aw;
33800             }
33801         }
33802         //tipBody.setWidth(w);
33803         el.setWidth(parseInt(w, 10) + p);
33804         if(ce.autoHide === false){
33805             close.setDisplayed(true);
33806             if(dd){
33807                 dd.unlock();
33808             }
33809         }else{
33810             close.setDisplayed(false);
33811             if(dd){
33812                 dd.lock();
33813             }
33814         }
33815         if(xy){
33816             el.avoidY = xy[1]-18;
33817             el.setXY(xy);
33818         }
33819         if(tm.animate){
33820             el.setOpacity(.1);
33821             el.setStyle("visibility", "visible");
33822             el.fadeIn({callback: afterShow});
33823         }else{
33824             afterShow();
33825         }
33826     };
33827     
33828     var afterShow = function(){
33829         if(ce){
33830             el.show();
33831             esc.enable();
33832             if(tm.autoDismiss && ce.autoHide !== false){
33833                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33834             }
33835         }
33836     };
33837     
33838     var hide = function(noanim){
33839         clearTimeout(dismissProc);
33840         clearTimeout(hideProc);
33841         ce = null;
33842         if(el.isVisible()){
33843             esc.disable();
33844             if(noanim !== true && tm.animate){
33845                 el.fadeOut({callback: afterHide});
33846             }else{
33847                 afterHide();
33848             } 
33849         }
33850     };
33851     
33852     var afterHide = function(){
33853         el.hide();
33854         if(removeCls){
33855             el.removeClass(removeCls);
33856             removeCls = null;
33857         }
33858     };
33859     
33860     return {
33861         /**
33862         * @cfg {Number} minWidth
33863         * The minimum width of the quick tip (defaults to 40)
33864         */
33865        minWidth : 40,
33866         /**
33867         * @cfg {Number} maxWidth
33868         * The maximum width of the quick tip (defaults to 300)
33869         */
33870        maxWidth : 300,
33871         /**
33872         * @cfg {Boolean} interceptTitles
33873         * True to automatically use the element's DOM title value if available (defaults to false)
33874         */
33875        interceptTitles : false,
33876         /**
33877         * @cfg {Boolean} trackMouse
33878         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33879         */
33880        trackMouse : false,
33881         /**
33882         * @cfg {Boolean} hideOnClick
33883         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33884         */
33885        hideOnClick : true,
33886         /**
33887         * @cfg {Number} showDelay
33888         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33889         */
33890        showDelay : 500,
33891         /**
33892         * @cfg {Number} hideDelay
33893         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33894         */
33895        hideDelay : 200,
33896         /**
33897         * @cfg {Boolean} autoHide
33898         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33899         * Used in conjunction with hideDelay.
33900         */
33901        autoHide : true,
33902         /**
33903         * @cfg {Boolean}
33904         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33905         * (defaults to true).  Used in conjunction with autoDismissDelay.
33906         */
33907        autoDismiss : true,
33908         /**
33909         * @cfg {Number}
33910         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33911         */
33912        autoDismissDelay : 5000,
33913        /**
33914         * @cfg {Boolean} animate
33915         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33916         */
33917        animate : false,
33918
33919        /**
33920         * @cfg {String} title
33921         * Title text to display (defaults to '').  This can be any valid HTML markup.
33922         */
33923         title: '',
33924        /**
33925         * @cfg {String} text
33926         * Body text to display (defaults to '').  This can be any valid HTML markup.
33927         */
33928         text : '',
33929        /**
33930         * @cfg {String} cls
33931         * A CSS class to apply to the base quick tip element (defaults to '').
33932         */
33933         cls : '',
33934        /**
33935         * @cfg {Number} width
33936         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33937         * minWidth or maxWidth.
33938         */
33939         width : null,
33940
33941     /**
33942      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33943      * or display QuickTips in a page.
33944      */
33945        init : function(){
33946           tm = Roo.QuickTips;
33947           cfg = tm.tagConfig;
33948           if(!inited){
33949               if(!Roo.isReady){ // allow calling of init() before onReady
33950                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33951                   return;
33952               }
33953               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33954               el.fxDefaults = {stopFx: true};
33955               // maximum custom styling
33956               //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>');
33957               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>');              
33958               tipTitle = el.child('h3');
33959               tipTitle.enableDisplayMode("block");
33960               tipBody = el.child('div.x-tip-bd');
33961               tipBodyText = el.child('div.x-tip-bd-inner');
33962               //bdLeft = el.child('div.x-tip-bd-left');
33963               //bdRight = el.child('div.x-tip-bd-right');
33964               close = el.child('div.x-tip-close');
33965               close.enableDisplayMode("block");
33966               close.on("click", hide);
33967               var d = Roo.get(document);
33968               d.on("mousedown", onDown);
33969               d.on("mouseover", onOver);
33970               d.on("mouseout", onOut);
33971               d.on("mousemove", onMove);
33972               esc = d.addKeyListener(27, hide);
33973               esc.disable();
33974               if(Roo.dd.DD){
33975                   dd = el.initDD("default", null, {
33976                       onDrag : function(){
33977                           el.sync();  
33978                       }
33979                   });
33980                   dd.setHandleElId(tipTitle.id);
33981                   dd.lock();
33982               }
33983               inited = true;
33984           }
33985           this.enable(); 
33986        },
33987
33988     /**
33989      * Configures a new quick tip instance and assigns it to a target element.  The following config options
33990      * are supported:
33991      * <pre>
33992 Property    Type                   Description
33993 ----------  ---------------------  ------------------------------------------------------------------------
33994 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
33995      * </ul>
33996      * @param {Object} config The config object
33997      */
33998        register : function(config){
33999            var cs = config instanceof Array ? config : arguments;
34000            for(var i = 0, len = cs.length; i < len; i++) {
34001                var c = cs[i];
34002                var target = c.target;
34003                if(target){
34004                    if(target instanceof Array){
34005                        for(var j = 0, jlen = target.length; j < jlen; j++){
34006                            tagEls[target[j]] = c;
34007                        }
34008                    }else{
34009                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34010                    }
34011                }
34012            }
34013        },
34014
34015     /**
34016      * Removes this quick tip from its element and destroys it.
34017      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34018      */
34019        unregister : function(el){
34020            delete tagEls[Roo.id(el)];
34021        },
34022
34023     /**
34024      * Enable this quick tip.
34025      */
34026        enable : function(){
34027            if(inited && disabled){
34028                locks.pop();
34029                if(locks.length < 1){
34030                    disabled = false;
34031                }
34032            }
34033        },
34034
34035     /**
34036      * Disable this quick tip.
34037      */
34038        disable : function(){
34039           disabled = true;
34040           clearTimeout(showProc);
34041           clearTimeout(hideProc);
34042           clearTimeout(dismissProc);
34043           if(ce){
34044               hide(true);
34045           }
34046           locks.push(1);
34047        },
34048
34049     /**
34050      * Returns true if the quick tip is enabled, else false.
34051      */
34052        isEnabled : function(){
34053             return !disabled;
34054        },
34055
34056         // private
34057        tagConfig : {
34058            namespace : "roo", // was ext?? this may break..
34059            alt_namespace : "ext",
34060            attribute : "qtip",
34061            width : "width",
34062            target : "target",
34063            title : "qtitle",
34064            hide : "hide",
34065            cls : "qclass"
34066        }
34067    };
34068 }();
34069
34070 // backwards compat
34071 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34072  * Based on:
34073  * Ext JS Library 1.1.1
34074  * Copyright(c) 2006-2007, Ext JS, LLC.
34075  *
34076  * Originally Released Under LGPL - original licence link has changed is not relivant.
34077  *
34078  * Fork - LGPL
34079  * <script type="text/javascript">
34080  */
34081  
34082
34083 /**
34084  * @class Roo.tree.TreePanel
34085  * @extends Roo.data.Tree
34086
34087  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34088  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34089  * @cfg {Boolean} enableDD true to enable drag and drop
34090  * @cfg {Boolean} enableDrag true to enable just drag
34091  * @cfg {Boolean} enableDrop true to enable just drop
34092  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34093  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34094  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34095  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34096  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34097  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34098  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34099  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34100  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34101  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34102  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34103  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34104  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34105  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34106  * @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>
34107  * @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>
34108  * 
34109  * @constructor
34110  * @param {String/HTMLElement/Element} el The container element
34111  * @param {Object} config
34112  */
34113 Roo.tree.TreePanel = function(el, config){
34114     var root = false;
34115     var loader = false;
34116     if (config.root) {
34117         root = config.root;
34118         delete config.root;
34119     }
34120     if (config.loader) {
34121         loader = config.loader;
34122         delete config.loader;
34123     }
34124     
34125     Roo.apply(this, config);
34126     Roo.tree.TreePanel.superclass.constructor.call(this);
34127     this.el = Roo.get(el);
34128     this.el.addClass('x-tree');
34129     //console.log(root);
34130     if (root) {
34131         this.setRootNode( Roo.factory(root, Roo.tree));
34132     }
34133     if (loader) {
34134         this.loader = Roo.factory(loader, Roo.tree);
34135     }
34136    /**
34137     * Read-only. The id of the container element becomes this TreePanel's id.
34138     */
34139     this.id = this.el.id;
34140     this.addEvents({
34141         /**
34142         * @event beforeload
34143         * Fires before a node is loaded, return false to cancel
34144         * @param {Node} node The node being loaded
34145         */
34146         "beforeload" : true,
34147         /**
34148         * @event load
34149         * Fires when a node is loaded
34150         * @param {Node} node The node that was loaded
34151         */
34152         "load" : true,
34153         /**
34154         * @event textchange
34155         * Fires when the text for a node is changed
34156         * @param {Node} node The node
34157         * @param {String} text The new text
34158         * @param {String} oldText The old text
34159         */
34160         "textchange" : true,
34161         /**
34162         * @event beforeexpand
34163         * Fires before a node is expanded, return false to cancel.
34164         * @param {Node} node The node
34165         * @param {Boolean} deep
34166         * @param {Boolean} anim
34167         */
34168         "beforeexpand" : true,
34169         /**
34170         * @event beforecollapse
34171         * Fires before a node is collapsed, return false to cancel.
34172         * @param {Node} node The node
34173         * @param {Boolean} deep
34174         * @param {Boolean} anim
34175         */
34176         "beforecollapse" : true,
34177         /**
34178         * @event expand
34179         * Fires when a node is expanded
34180         * @param {Node} node The node
34181         */
34182         "expand" : true,
34183         /**
34184         * @event disabledchange
34185         * Fires when the disabled status of a node changes
34186         * @param {Node} node The node
34187         * @param {Boolean} disabled
34188         */
34189         "disabledchange" : true,
34190         /**
34191         * @event collapse
34192         * Fires when a node is collapsed
34193         * @param {Node} node The node
34194         */
34195         "collapse" : true,
34196         /**
34197         * @event beforeclick
34198         * Fires before click processing on a node. Return false to cancel the default action.
34199         * @param {Node} node The node
34200         * @param {Roo.EventObject} e The event object
34201         */
34202         "beforeclick":true,
34203         /**
34204         * @event checkchange
34205         * Fires when a node with a checkbox's checked property changes
34206         * @param {Node} this This node
34207         * @param {Boolean} checked
34208         */
34209         "checkchange":true,
34210         /**
34211         * @event click
34212         * Fires when a node is clicked
34213         * @param {Node} node The node
34214         * @param {Roo.EventObject} e The event object
34215         */
34216         "click":true,
34217         /**
34218         * @event dblclick
34219         * Fires when a node is double clicked
34220         * @param {Node} node The node
34221         * @param {Roo.EventObject} e The event object
34222         */
34223         "dblclick":true,
34224         /**
34225         * @event contextmenu
34226         * Fires when a node is right clicked
34227         * @param {Node} node The node
34228         * @param {Roo.EventObject} e The event object
34229         */
34230         "contextmenu":true,
34231         /**
34232         * @event beforechildrenrendered
34233         * Fires right before the child nodes for a node are rendered
34234         * @param {Node} node The node
34235         */
34236         "beforechildrenrendered":true,
34237         /**
34238         * @event startdrag
34239         * Fires when a node starts being dragged
34240         * @param {Roo.tree.TreePanel} this
34241         * @param {Roo.tree.TreeNode} node
34242         * @param {event} e The raw browser event
34243         */ 
34244        "startdrag" : true,
34245        /**
34246         * @event enddrag
34247         * Fires when a drag operation is complete
34248         * @param {Roo.tree.TreePanel} this
34249         * @param {Roo.tree.TreeNode} node
34250         * @param {event} e The raw browser event
34251         */
34252        "enddrag" : true,
34253        /**
34254         * @event dragdrop
34255         * Fires when a dragged node is dropped on a valid DD target
34256         * @param {Roo.tree.TreePanel} this
34257         * @param {Roo.tree.TreeNode} node
34258         * @param {DD} dd The dd it was dropped on
34259         * @param {event} e The raw browser event
34260         */
34261        "dragdrop" : true,
34262        /**
34263         * @event beforenodedrop
34264         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34265         * passed to handlers has the following properties:<br />
34266         * <ul style="padding:5px;padding-left:16px;">
34267         * <li>tree - The TreePanel</li>
34268         * <li>target - The node being targeted for the drop</li>
34269         * <li>data - The drag data from the drag source</li>
34270         * <li>point - The point of the drop - append, above or below</li>
34271         * <li>source - The drag source</li>
34272         * <li>rawEvent - Raw mouse event</li>
34273         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34274         * to be inserted by setting them on this object.</li>
34275         * <li>cancel - Set this to true to cancel the drop.</li>
34276         * </ul>
34277         * @param {Object} dropEvent
34278         */
34279        "beforenodedrop" : true,
34280        /**
34281         * @event nodedrop
34282         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34283         * passed to handlers has the following properties:<br />
34284         * <ul style="padding:5px;padding-left:16px;">
34285         * <li>tree - The TreePanel</li>
34286         * <li>target - The node being targeted for the drop</li>
34287         * <li>data - The drag data from the drag source</li>
34288         * <li>point - The point of the drop - append, above or below</li>
34289         * <li>source - The drag source</li>
34290         * <li>rawEvent - Raw mouse event</li>
34291         * <li>dropNode - Dropped node(s).</li>
34292         * </ul>
34293         * @param {Object} dropEvent
34294         */
34295        "nodedrop" : true,
34296         /**
34297         * @event nodedragover
34298         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34299         * passed to handlers has the following properties:<br />
34300         * <ul style="padding:5px;padding-left:16px;">
34301         * <li>tree - The TreePanel</li>
34302         * <li>target - The node being targeted for the drop</li>
34303         * <li>data - The drag data from the drag source</li>
34304         * <li>point - The point of the drop - append, above or below</li>
34305         * <li>source - The drag source</li>
34306         * <li>rawEvent - Raw mouse event</li>
34307         * <li>dropNode - Drop node(s) provided by the source.</li>
34308         * <li>cancel - Set this to true to signal drop not allowed.</li>
34309         * </ul>
34310         * @param {Object} dragOverEvent
34311         */
34312        "nodedragover" : true,
34313        /**
34314         * @event appendnode
34315         * Fires when append node to the tree
34316         * @param {Roo.tree.TreePanel} this
34317         * @param {Roo.tree.TreeNode} node
34318         * @param {Number} index The index of the newly appended node
34319         */
34320        "appendnode" : true
34321         
34322     });
34323     if(this.singleExpand){
34324        this.on("beforeexpand", this.restrictExpand, this);
34325     }
34326     if (this.editor) {
34327         this.editor.tree = this;
34328         this.editor = Roo.factory(this.editor, Roo.tree);
34329     }
34330     
34331     if (this.selModel) {
34332         this.selModel = Roo.factory(this.selModel, Roo.tree);
34333     }
34334    
34335 };
34336 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34337     rootVisible : true,
34338     animate: Roo.enableFx,
34339     lines : true,
34340     enableDD : false,
34341     hlDrop : Roo.enableFx,
34342   
34343     renderer: false,
34344     
34345     rendererTip: false,
34346     // private
34347     restrictExpand : function(node){
34348         var p = node.parentNode;
34349         if(p){
34350             if(p.expandedChild && p.expandedChild.parentNode == p){
34351                 p.expandedChild.collapse();
34352             }
34353             p.expandedChild = node;
34354         }
34355     },
34356
34357     // private override
34358     setRootNode : function(node){
34359         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34360         if(!this.rootVisible){
34361             node.ui = new Roo.tree.RootTreeNodeUI(node);
34362         }
34363         return node;
34364     },
34365
34366     /**
34367      * Returns the container element for this TreePanel
34368      */
34369     getEl : function(){
34370         return this.el;
34371     },
34372
34373     /**
34374      * Returns the default TreeLoader for this TreePanel
34375      */
34376     getLoader : function(){
34377         return this.loader;
34378     },
34379
34380     /**
34381      * Expand all nodes
34382      */
34383     expandAll : function(){
34384         this.root.expand(true);
34385     },
34386
34387     /**
34388      * Collapse all nodes
34389      */
34390     collapseAll : function(){
34391         this.root.collapse(true);
34392     },
34393
34394     /**
34395      * Returns the selection model used by this TreePanel
34396      */
34397     getSelectionModel : function(){
34398         if(!this.selModel){
34399             this.selModel = new Roo.tree.DefaultSelectionModel();
34400         }
34401         return this.selModel;
34402     },
34403
34404     /**
34405      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34406      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34407      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34408      * @return {Array}
34409      */
34410     getChecked : function(a, startNode){
34411         startNode = startNode || this.root;
34412         var r = [];
34413         var f = function(){
34414             if(this.attributes.checked){
34415                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34416             }
34417         }
34418         startNode.cascade(f);
34419         return r;
34420     },
34421
34422     /**
34423      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34424      * @param {String} path
34425      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34426      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34427      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34428      */
34429     expandPath : function(path, attr, callback){
34430         attr = attr || "id";
34431         var keys = path.split(this.pathSeparator);
34432         var curNode = this.root;
34433         if(curNode.attributes[attr] != keys[1]){ // invalid root
34434             if(callback){
34435                 callback(false, null);
34436             }
34437             return;
34438         }
34439         var index = 1;
34440         var f = function(){
34441             if(++index == keys.length){
34442                 if(callback){
34443                     callback(true, curNode);
34444                 }
34445                 return;
34446             }
34447             var c = curNode.findChild(attr, keys[index]);
34448             if(!c){
34449                 if(callback){
34450                     callback(false, curNode);
34451                 }
34452                 return;
34453             }
34454             curNode = c;
34455             c.expand(false, false, f);
34456         };
34457         curNode.expand(false, false, f);
34458     },
34459
34460     /**
34461      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34462      * @param {String} path
34463      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34464      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34465      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34466      */
34467     selectPath : function(path, attr, callback){
34468         attr = attr || "id";
34469         var keys = path.split(this.pathSeparator);
34470         var v = keys.pop();
34471         if(keys.length > 0){
34472             var f = function(success, node){
34473                 if(success && node){
34474                     var n = node.findChild(attr, v);
34475                     if(n){
34476                         n.select();
34477                         if(callback){
34478                             callback(true, n);
34479                         }
34480                     }else if(callback){
34481                         callback(false, n);
34482                     }
34483                 }else{
34484                     if(callback){
34485                         callback(false, n);
34486                     }
34487                 }
34488             };
34489             this.expandPath(keys.join(this.pathSeparator), attr, f);
34490         }else{
34491             this.root.select();
34492             if(callback){
34493                 callback(true, this.root);
34494             }
34495         }
34496     },
34497
34498     getTreeEl : function(){
34499         return this.el;
34500     },
34501
34502     /**
34503      * Trigger rendering of this TreePanel
34504      */
34505     render : function(){
34506         if (this.innerCt) {
34507             return this; // stop it rendering more than once!!
34508         }
34509         
34510         this.innerCt = this.el.createChild({tag:"ul",
34511                cls:"x-tree-root-ct " +
34512                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34513
34514         if(this.containerScroll){
34515             Roo.dd.ScrollManager.register(this.el);
34516         }
34517         if((this.enableDD || this.enableDrop) && !this.dropZone){
34518            /**
34519             * The dropZone used by this tree if drop is enabled
34520             * @type Roo.tree.TreeDropZone
34521             */
34522              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34523                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34524            });
34525         }
34526         if((this.enableDD || this.enableDrag) && !this.dragZone){
34527            /**
34528             * The dragZone used by this tree if drag is enabled
34529             * @type Roo.tree.TreeDragZone
34530             */
34531             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34532                ddGroup: this.ddGroup || "TreeDD",
34533                scroll: this.ddScroll
34534            });
34535         }
34536         this.getSelectionModel().init(this);
34537         if (!this.root) {
34538             Roo.log("ROOT not set in tree");
34539             return this;
34540         }
34541         this.root.render();
34542         if(!this.rootVisible){
34543             this.root.renderChildren();
34544         }
34545         return this;
34546     }
34547 });/*
34548  * Based on:
34549  * Ext JS Library 1.1.1
34550  * Copyright(c) 2006-2007, Ext JS, LLC.
34551  *
34552  * Originally Released Under LGPL - original licence link has changed is not relivant.
34553  *
34554  * Fork - LGPL
34555  * <script type="text/javascript">
34556  */
34557  
34558
34559 /**
34560  * @class Roo.tree.DefaultSelectionModel
34561  * @extends Roo.util.Observable
34562  * The default single selection for a TreePanel.
34563  * @param {Object} cfg Configuration
34564  */
34565 Roo.tree.DefaultSelectionModel = function(cfg){
34566    this.selNode = null;
34567    
34568    
34569    
34570    this.addEvents({
34571        /**
34572         * @event selectionchange
34573         * Fires when the selected node changes
34574         * @param {DefaultSelectionModel} this
34575         * @param {TreeNode} node the new selection
34576         */
34577        "selectionchange" : true,
34578
34579        /**
34580         * @event beforeselect
34581         * Fires before the selected node changes, return false to cancel the change
34582         * @param {DefaultSelectionModel} this
34583         * @param {TreeNode} node the new selection
34584         * @param {TreeNode} node the old selection
34585         */
34586        "beforeselect" : true
34587    });
34588    
34589     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34590 };
34591
34592 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34593     init : function(tree){
34594         this.tree = tree;
34595         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34596         tree.on("click", this.onNodeClick, this);
34597     },
34598     
34599     onNodeClick : function(node, e){
34600         if (e.ctrlKey && this.selNode == node)  {
34601             this.unselect(node);
34602             return;
34603         }
34604         this.select(node);
34605     },
34606     
34607     /**
34608      * Select a node.
34609      * @param {TreeNode} node The node to select
34610      * @return {TreeNode} The selected node
34611      */
34612     select : function(node){
34613         var last = this.selNode;
34614         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34615             if(last){
34616                 last.ui.onSelectedChange(false);
34617             }
34618             this.selNode = node;
34619             node.ui.onSelectedChange(true);
34620             this.fireEvent("selectionchange", this, node, last);
34621         }
34622         return node;
34623     },
34624     
34625     /**
34626      * Deselect a node.
34627      * @param {TreeNode} node The node to unselect
34628      */
34629     unselect : function(node){
34630         if(this.selNode == node){
34631             this.clearSelections();
34632         }    
34633     },
34634     
34635     /**
34636      * Clear all selections
34637      */
34638     clearSelections : function(){
34639         var n = this.selNode;
34640         if(n){
34641             n.ui.onSelectedChange(false);
34642             this.selNode = null;
34643             this.fireEvent("selectionchange", this, null);
34644         }
34645         return n;
34646     },
34647     
34648     /**
34649      * Get the selected node
34650      * @return {TreeNode} The selected node
34651      */
34652     getSelectedNode : function(){
34653         return this.selNode;    
34654     },
34655     
34656     /**
34657      * Returns true if the node is selected
34658      * @param {TreeNode} node The node to check
34659      * @return {Boolean}
34660      */
34661     isSelected : function(node){
34662         return this.selNode == node;  
34663     },
34664
34665     /**
34666      * Selects the node above the selected node in the tree, intelligently walking the nodes
34667      * @return TreeNode The new selection
34668      */
34669     selectPrevious : function(){
34670         var s = this.selNode || this.lastSelNode;
34671         if(!s){
34672             return null;
34673         }
34674         var ps = s.previousSibling;
34675         if(ps){
34676             if(!ps.isExpanded() || ps.childNodes.length < 1){
34677                 return this.select(ps);
34678             } else{
34679                 var lc = ps.lastChild;
34680                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34681                     lc = lc.lastChild;
34682                 }
34683                 return this.select(lc);
34684             }
34685         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34686             return this.select(s.parentNode);
34687         }
34688         return null;
34689     },
34690
34691     /**
34692      * Selects the node above the selected node in the tree, intelligently walking the nodes
34693      * @return TreeNode The new selection
34694      */
34695     selectNext : function(){
34696         var s = this.selNode || this.lastSelNode;
34697         if(!s){
34698             return null;
34699         }
34700         if(s.firstChild && s.isExpanded()){
34701              return this.select(s.firstChild);
34702          }else if(s.nextSibling){
34703              return this.select(s.nextSibling);
34704          }else if(s.parentNode){
34705             var newS = null;
34706             s.parentNode.bubble(function(){
34707                 if(this.nextSibling){
34708                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34709                     return false;
34710                 }
34711             });
34712             return newS;
34713          }
34714         return null;
34715     },
34716
34717     onKeyDown : function(e){
34718         var s = this.selNode || this.lastSelNode;
34719         // undesirable, but required
34720         var sm = this;
34721         if(!s){
34722             return;
34723         }
34724         var k = e.getKey();
34725         switch(k){
34726              case e.DOWN:
34727                  e.stopEvent();
34728                  this.selectNext();
34729              break;
34730              case e.UP:
34731                  e.stopEvent();
34732                  this.selectPrevious();
34733              break;
34734              case e.RIGHT:
34735                  e.preventDefault();
34736                  if(s.hasChildNodes()){
34737                      if(!s.isExpanded()){
34738                          s.expand();
34739                      }else if(s.firstChild){
34740                          this.select(s.firstChild, e);
34741                      }
34742                  }
34743              break;
34744              case e.LEFT:
34745                  e.preventDefault();
34746                  if(s.hasChildNodes() && s.isExpanded()){
34747                      s.collapse();
34748                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34749                      this.select(s.parentNode, e);
34750                  }
34751              break;
34752         };
34753     }
34754 });
34755
34756 /**
34757  * @class Roo.tree.MultiSelectionModel
34758  * @extends Roo.util.Observable
34759  * Multi selection for a TreePanel.
34760  * @param {Object} cfg Configuration
34761  */
34762 Roo.tree.MultiSelectionModel = function(){
34763    this.selNodes = [];
34764    this.selMap = {};
34765    this.addEvents({
34766        /**
34767         * @event selectionchange
34768         * Fires when the selected nodes change
34769         * @param {MultiSelectionModel} this
34770         * @param {Array} nodes Array of the selected nodes
34771         */
34772        "selectionchange" : true
34773    });
34774    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34775    
34776 };
34777
34778 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34779     init : function(tree){
34780         this.tree = tree;
34781         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34782         tree.on("click", this.onNodeClick, this);
34783     },
34784     
34785     onNodeClick : function(node, e){
34786         this.select(node, e, e.ctrlKey);
34787     },
34788     
34789     /**
34790      * Select a node.
34791      * @param {TreeNode} node The node to select
34792      * @param {EventObject} e (optional) An event associated with the selection
34793      * @param {Boolean} keepExisting True to retain existing selections
34794      * @return {TreeNode} The selected node
34795      */
34796     select : function(node, e, keepExisting){
34797         if(keepExisting !== true){
34798             this.clearSelections(true);
34799         }
34800         if(this.isSelected(node)){
34801             this.lastSelNode = node;
34802             return node;
34803         }
34804         this.selNodes.push(node);
34805         this.selMap[node.id] = node;
34806         this.lastSelNode = node;
34807         node.ui.onSelectedChange(true);
34808         this.fireEvent("selectionchange", this, this.selNodes);
34809         return node;
34810     },
34811     
34812     /**
34813      * Deselect a node.
34814      * @param {TreeNode} node The node to unselect
34815      */
34816     unselect : function(node){
34817         if(this.selMap[node.id]){
34818             node.ui.onSelectedChange(false);
34819             var sn = this.selNodes;
34820             var index = -1;
34821             if(sn.indexOf){
34822                 index = sn.indexOf(node);
34823             }else{
34824                 for(var i = 0, len = sn.length; i < len; i++){
34825                     if(sn[i] == node){
34826                         index = i;
34827                         break;
34828                     }
34829                 }
34830             }
34831             if(index != -1){
34832                 this.selNodes.splice(index, 1);
34833             }
34834             delete this.selMap[node.id];
34835             this.fireEvent("selectionchange", this, this.selNodes);
34836         }
34837     },
34838     
34839     /**
34840      * Clear all selections
34841      */
34842     clearSelections : function(suppressEvent){
34843         var sn = this.selNodes;
34844         if(sn.length > 0){
34845             for(var i = 0, len = sn.length; i < len; i++){
34846                 sn[i].ui.onSelectedChange(false);
34847             }
34848             this.selNodes = [];
34849             this.selMap = {};
34850             if(suppressEvent !== true){
34851                 this.fireEvent("selectionchange", this, this.selNodes);
34852             }
34853         }
34854     },
34855     
34856     /**
34857      * Returns true if the node is selected
34858      * @param {TreeNode} node The node to check
34859      * @return {Boolean}
34860      */
34861     isSelected : function(node){
34862         return this.selMap[node.id] ? true : false;  
34863     },
34864     
34865     /**
34866      * Returns an array of the selected nodes
34867      * @return {Array}
34868      */
34869     getSelectedNodes : function(){
34870         return this.selNodes;    
34871     },
34872
34873     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34874
34875     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34876
34877     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34878 });/*
34879  * Based on:
34880  * Ext JS Library 1.1.1
34881  * Copyright(c) 2006-2007, Ext JS, LLC.
34882  *
34883  * Originally Released Under LGPL - original licence link has changed is not relivant.
34884  *
34885  * Fork - LGPL
34886  * <script type="text/javascript">
34887  */
34888  
34889 /**
34890  * @class Roo.tree.TreeNode
34891  * @extends Roo.data.Node
34892  * @cfg {String} text The text for this node
34893  * @cfg {Boolean} expanded true to start the node expanded
34894  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34895  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34896  * @cfg {Boolean} disabled true to start the node disabled
34897  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34898  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34899  * @cfg {String} cls A css class to be added to the node
34900  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34901  * @cfg {String} href URL of the link used for the node (defaults to #)
34902  * @cfg {String} hrefTarget target frame for the link
34903  * @cfg {String} qtip An Ext QuickTip for the node
34904  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34905  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34906  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34907  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34908  * (defaults to undefined with no checkbox rendered)
34909  * @constructor
34910  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34911  */
34912 Roo.tree.TreeNode = function(attributes){
34913     attributes = attributes || {};
34914     if(typeof attributes == "string"){
34915         attributes = {text: attributes};
34916     }
34917     this.childrenRendered = false;
34918     this.rendered = false;
34919     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34920     this.expanded = attributes.expanded === true;
34921     this.isTarget = attributes.isTarget !== false;
34922     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34923     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34924
34925     /**
34926      * Read-only. The text for this node. To change it use setText().
34927      * @type String
34928      */
34929     this.text = attributes.text;
34930     /**
34931      * True if this node is disabled.
34932      * @type Boolean
34933      */
34934     this.disabled = attributes.disabled === true;
34935
34936     this.addEvents({
34937         /**
34938         * @event textchange
34939         * Fires when the text for this node is changed
34940         * @param {Node} this This node
34941         * @param {String} text The new text
34942         * @param {String} oldText The old text
34943         */
34944         "textchange" : true,
34945         /**
34946         * @event beforeexpand
34947         * Fires before this node is expanded, return false to cancel.
34948         * @param {Node} this This node
34949         * @param {Boolean} deep
34950         * @param {Boolean} anim
34951         */
34952         "beforeexpand" : true,
34953         /**
34954         * @event beforecollapse
34955         * Fires before this node is collapsed, return false to cancel.
34956         * @param {Node} this This node
34957         * @param {Boolean} deep
34958         * @param {Boolean} anim
34959         */
34960         "beforecollapse" : true,
34961         /**
34962         * @event expand
34963         * Fires when this node is expanded
34964         * @param {Node} this This node
34965         */
34966         "expand" : true,
34967         /**
34968         * @event disabledchange
34969         * Fires when the disabled status of this node changes
34970         * @param {Node} this This node
34971         * @param {Boolean} disabled
34972         */
34973         "disabledchange" : true,
34974         /**
34975         * @event collapse
34976         * Fires when this node is collapsed
34977         * @param {Node} this This node
34978         */
34979         "collapse" : true,
34980         /**
34981         * @event beforeclick
34982         * Fires before click processing. Return false to cancel the default action.
34983         * @param {Node} this This node
34984         * @param {Roo.EventObject} e The event object
34985         */
34986         "beforeclick":true,
34987         /**
34988         * @event checkchange
34989         * Fires when a node with a checkbox's checked property changes
34990         * @param {Node} this This node
34991         * @param {Boolean} checked
34992         */
34993         "checkchange":true,
34994         /**
34995         * @event click
34996         * Fires when this node is clicked
34997         * @param {Node} this This node
34998         * @param {Roo.EventObject} e The event object
34999         */
35000         "click":true,
35001         /**
35002         * @event dblclick
35003         * Fires when this node is double clicked
35004         * @param {Node} this This node
35005         * @param {Roo.EventObject} e The event object
35006         */
35007         "dblclick":true,
35008         /**
35009         * @event contextmenu
35010         * Fires when this node is right clicked
35011         * @param {Node} this This node
35012         * @param {Roo.EventObject} e The event object
35013         */
35014         "contextmenu":true,
35015         /**
35016         * @event beforechildrenrendered
35017         * Fires right before the child nodes for this node are rendered
35018         * @param {Node} this This node
35019         */
35020         "beforechildrenrendered":true
35021     });
35022
35023     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35024
35025     /**
35026      * Read-only. The UI for this node
35027      * @type TreeNodeUI
35028      */
35029     this.ui = new uiClass(this);
35030     
35031     // finally support items[]
35032     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35033         return;
35034     }
35035     
35036     
35037     Roo.each(this.attributes.items, function(c) {
35038         this.appendChild(Roo.factory(c,Roo.Tree));
35039     }, this);
35040     delete this.attributes.items;
35041     
35042     
35043     
35044 };
35045 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35046     preventHScroll: true,
35047     /**
35048      * Returns true if this node is expanded
35049      * @return {Boolean}
35050      */
35051     isExpanded : function(){
35052         return this.expanded;
35053     },
35054
35055     /**
35056      * Returns the UI object for this node
35057      * @return {TreeNodeUI}
35058      */
35059     getUI : function(){
35060         return this.ui;
35061     },
35062
35063     // private override
35064     setFirstChild : function(node){
35065         var of = this.firstChild;
35066         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35067         if(this.childrenRendered && of && node != of){
35068             of.renderIndent(true, true);
35069         }
35070         if(this.rendered){
35071             this.renderIndent(true, true);
35072         }
35073     },
35074
35075     // private override
35076     setLastChild : function(node){
35077         var ol = this.lastChild;
35078         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35079         if(this.childrenRendered && ol && node != ol){
35080             ol.renderIndent(true, true);
35081         }
35082         if(this.rendered){
35083             this.renderIndent(true, true);
35084         }
35085     },
35086
35087     // these methods are overridden to provide lazy rendering support
35088     // private override
35089     appendChild : function()
35090     {
35091         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35092         if(node && this.childrenRendered){
35093             node.render();
35094         }
35095         this.ui.updateExpandIcon();
35096         return node;
35097     },
35098
35099     // private override
35100     removeChild : function(node){
35101         this.ownerTree.getSelectionModel().unselect(node);
35102         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35103         // if it's been rendered remove dom node
35104         if(this.childrenRendered){
35105             node.ui.remove();
35106         }
35107         if(this.childNodes.length < 1){
35108             this.collapse(false, false);
35109         }else{
35110             this.ui.updateExpandIcon();
35111         }
35112         if(!this.firstChild) {
35113             this.childrenRendered = false;
35114         }
35115         return node;
35116     },
35117
35118     // private override
35119     insertBefore : function(node, refNode){
35120         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35121         if(newNode && refNode && this.childrenRendered){
35122             node.render();
35123         }
35124         this.ui.updateExpandIcon();
35125         return newNode;
35126     },
35127
35128     /**
35129      * Sets the text for this node
35130      * @param {String} text
35131      */
35132     setText : function(text){
35133         var oldText = this.text;
35134         this.text = text;
35135         this.attributes.text = text;
35136         if(this.rendered){ // event without subscribing
35137             this.ui.onTextChange(this, text, oldText);
35138         }
35139         this.fireEvent("textchange", this, text, oldText);
35140     },
35141
35142     /**
35143      * Triggers selection of this node
35144      */
35145     select : function(){
35146         this.getOwnerTree().getSelectionModel().select(this);
35147     },
35148
35149     /**
35150      * Triggers deselection of this node
35151      */
35152     unselect : function(){
35153         this.getOwnerTree().getSelectionModel().unselect(this);
35154     },
35155
35156     /**
35157      * Returns true if this node is selected
35158      * @return {Boolean}
35159      */
35160     isSelected : function(){
35161         return this.getOwnerTree().getSelectionModel().isSelected(this);
35162     },
35163
35164     /**
35165      * Expand this node.
35166      * @param {Boolean} deep (optional) True to expand all children as well
35167      * @param {Boolean} anim (optional) false to cancel the default animation
35168      * @param {Function} callback (optional) A callback to be called when
35169      * expanding this node completes (does not wait for deep expand to complete).
35170      * Called with 1 parameter, this node.
35171      */
35172     expand : function(deep, anim, callback){
35173         if(!this.expanded){
35174             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35175                 return;
35176             }
35177             if(!this.childrenRendered){
35178                 this.renderChildren();
35179             }
35180             this.expanded = true;
35181             
35182             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35183                 this.ui.animExpand(function(){
35184                     this.fireEvent("expand", this);
35185                     if(typeof callback == "function"){
35186                         callback(this);
35187                     }
35188                     if(deep === true){
35189                         this.expandChildNodes(true);
35190                     }
35191                 }.createDelegate(this));
35192                 return;
35193             }else{
35194                 this.ui.expand();
35195                 this.fireEvent("expand", this);
35196                 if(typeof callback == "function"){
35197                     callback(this);
35198                 }
35199             }
35200         }else{
35201            if(typeof callback == "function"){
35202                callback(this);
35203            }
35204         }
35205         if(deep === true){
35206             this.expandChildNodes(true);
35207         }
35208     },
35209
35210     isHiddenRoot : function(){
35211         return this.isRoot && !this.getOwnerTree().rootVisible;
35212     },
35213
35214     /**
35215      * Collapse this node.
35216      * @param {Boolean} deep (optional) True to collapse all children as well
35217      * @param {Boolean} anim (optional) false to cancel the default animation
35218      */
35219     collapse : function(deep, anim){
35220         if(this.expanded && !this.isHiddenRoot()){
35221             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35222                 return;
35223             }
35224             this.expanded = false;
35225             if((this.getOwnerTree().animate && anim !== false) || anim){
35226                 this.ui.animCollapse(function(){
35227                     this.fireEvent("collapse", this);
35228                     if(deep === true){
35229                         this.collapseChildNodes(true);
35230                     }
35231                 }.createDelegate(this));
35232                 return;
35233             }else{
35234                 this.ui.collapse();
35235                 this.fireEvent("collapse", this);
35236             }
35237         }
35238         if(deep === true){
35239             var cs = this.childNodes;
35240             for(var i = 0, len = cs.length; i < len; i++) {
35241                 cs[i].collapse(true, false);
35242             }
35243         }
35244     },
35245
35246     // private
35247     delayedExpand : function(delay){
35248         if(!this.expandProcId){
35249             this.expandProcId = this.expand.defer(delay, this);
35250         }
35251     },
35252
35253     // private
35254     cancelExpand : function(){
35255         if(this.expandProcId){
35256             clearTimeout(this.expandProcId);
35257         }
35258         this.expandProcId = false;
35259     },
35260
35261     /**
35262      * Toggles expanded/collapsed state of the node
35263      */
35264     toggle : function(){
35265         if(this.expanded){
35266             this.collapse();
35267         }else{
35268             this.expand();
35269         }
35270     },
35271
35272     /**
35273      * Ensures all parent nodes are expanded
35274      */
35275     ensureVisible : function(callback){
35276         var tree = this.getOwnerTree();
35277         tree.expandPath(this.parentNode.getPath(), false, function(){
35278             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35279             Roo.callback(callback);
35280         }.createDelegate(this));
35281     },
35282
35283     /**
35284      * Expand all child nodes
35285      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35286      */
35287     expandChildNodes : function(deep){
35288         var cs = this.childNodes;
35289         for(var i = 0, len = cs.length; i < len; i++) {
35290                 cs[i].expand(deep);
35291         }
35292     },
35293
35294     /**
35295      * Collapse all child nodes
35296      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35297      */
35298     collapseChildNodes : function(deep){
35299         var cs = this.childNodes;
35300         for(var i = 0, len = cs.length; i < len; i++) {
35301                 cs[i].collapse(deep);
35302         }
35303     },
35304
35305     /**
35306      * Disables this node
35307      */
35308     disable : function(){
35309         this.disabled = true;
35310         this.unselect();
35311         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35312             this.ui.onDisableChange(this, true);
35313         }
35314         this.fireEvent("disabledchange", this, true);
35315     },
35316
35317     /**
35318      * Enables this node
35319      */
35320     enable : function(){
35321         this.disabled = false;
35322         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35323             this.ui.onDisableChange(this, false);
35324         }
35325         this.fireEvent("disabledchange", this, false);
35326     },
35327
35328     // private
35329     renderChildren : function(suppressEvent){
35330         if(suppressEvent !== false){
35331             this.fireEvent("beforechildrenrendered", this);
35332         }
35333         var cs = this.childNodes;
35334         for(var i = 0, len = cs.length; i < len; i++){
35335             cs[i].render(true);
35336         }
35337         this.childrenRendered = true;
35338     },
35339
35340     // private
35341     sort : function(fn, scope){
35342         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35343         if(this.childrenRendered){
35344             var cs = this.childNodes;
35345             for(var i = 0, len = cs.length; i < len; i++){
35346                 cs[i].render(true);
35347             }
35348         }
35349     },
35350
35351     // private
35352     render : function(bulkRender){
35353         this.ui.render(bulkRender);
35354         if(!this.rendered){
35355             this.rendered = true;
35356             if(this.expanded){
35357                 this.expanded = false;
35358                 this.expand(false, false);
35359             }
35360         }
35361     },
35362
35363     // private
35364     renderIndent : function(deep, refresh){
35365         if(refresh){
35366             this.ui.childIndent = null;
35367         }
35368         this.ui.renderIndent();
35369         if(deep === true && this.childrenRendered){
35370             var cs = this.childNodes;
35371             for(var i = 0, len = cs.length; i < len; i++){
35372                 cs[i].renderIndent(true, refresh);
35373             }
35374         }
35375     }
35376 });/*
35377  * Based on:
35378  * Ext JS Library 1.1.1
35379  * Copyright(c) 2006-2007, Ext JS, LLC.
35380  *
35381  * Originally Released Under LGPL - original licence link has changed is not relivant.
35382  *
35383  * Fork - LGPL
35384  * <script type="text/javascript">
35385  */
35386  
35387 /**
35388  * @class Roo.tree.AsyncTreeNode
35389  * @extends Roo.tree.TreeNode
35390  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35391  * @constructor
35392  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35393  */
35394  Roo.tree.AsyncTreeNode = function(config){
35395     this.loaded = false;
35396     this.loading = false;
35397     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35398     /**
35399     * @event beforeload
35400     * Fires before this node is loaded, return false to cancel
35401     * @param {Node} this This node
35402     */
35403     this.addEvents({'beforeload':true, 'load': true});
35404     /**
35405     * @event load
35406     * Fires when this node is loaded
35407     * @param {Node} this This node
35408     */
35409     /**
35410      * The loader used by this node (defaults to using the tree's defined loader)
35411      * @type TreeLoader
35412      * @property loader
35413      */
35414 };
35415 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35416     expand : function(deep, anim, callback){
35417         if(this.loading){ // if an async load is already running, waiting til it's done
35418             var timer;
35419             var f = function(){
35420                 if(!this.loading){ // done loading
35421                     clearInterval(timer);
35422                     this.expand(deep, anim, callback);
35423                 }
35424             }.createDelegate(this);
35425             timer = setInterval(f, 200);
35426             return;
35427         }
35428         if(!this.loaded){
35429             if(this.fireEvent("beforeload", this) === false){
35430                 return;
35431             }
35432             this.loading = true;
35433             this.ui.beforeLoad(this);
35434             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35435             if(loader){
35436                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35437                 return;
35438             }
35439         }
35440         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35441     },
35442     
35443     /**
35444      * Returns true if this node is currently loading
35445      * @return {Boolean}
35446      */
35447     isLoading : function(){
35448         return this.loading;  
35449     },
35450     
35451     loadComplete : function(deep, anim, callback){
35452         this.loading = false;
35453         this.loaded = true;
35454         this.ui.afterLoad(this);
35455         this.fireEvent("load", this);
35456         this.expand(deep, anim, callback);
35457     },
35458     
35459     /**
35460      * Returns true if this node has been loaded
35461      * @return {Boolean}
35462      */
35463     isLoaded : function(){
35464         return this.loaded;
35465     },
35466     
35467     hasChildNodes : function(){
35468         if(!this.isLeaf() && !this.loaded){
35469             return true;
35470         }else{
35471             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35472         }
35473     },
35474
35475     /**
35476      * Trigger a reload for this node
35477      * @param {Function} callback
35478      */
35479     reload : function(callback){
35480         this.collapse(false, false);
35481         while(this.firstChild){
35482             this.removeChild(this.firstChild);
35483         }
35484         this.childrenRendered = false;
35485         this.loaded = false;
35486         if(this.isHiddenRoot()){
35487             this.expanded = false;
35488         }
35489         this.expand(false, false, callback);
35490     }
35491 });/*
35492  * Based on:
35493  * Ext JS Library 1.1.1
35494  * Copyright(c) 2006-2007, Ext JS, LLC.
35495  *
35496  * Originally Released Under LGPL - original licence link has changed is not relivant.
35497  *
35498  * Fork - LGPL
35499  * <script type="text/javascript">
35500  */
35501  
35502 /**
35503  * @class Roo.tree.TreeNodeUI
35504  * @constructor
35505  * @param {Object} node The node to render
35506  * The TreeNode UI implementation is separate from the
35507  * tree implementation. Unless you are customizing the tree UI,
35508  * you should never have to use this directly.
35509  */
35510 Roo.tree.TreeNodeUI = function(node){
35511     this.node = node;
35512     this.rendered = false;
35513     this.animating = false;
35514     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35515 };
35516
35517 Roo.tree.TreeNodeUI.prototype = {
35518     removeChild : function(node){
35519         if(this.rendered){
35520             this.ctNode.removeChild(node.ui.getEl());
35521         }
35522     },
35523
35524     beforeLoad : function(){
35525          this.addClass("x-tree-node-loading");
35526     },
35527
35528     afterLoad : function(){
35529          this.removeClass("x-tree-node-loading");
35530     },
35531
35532     onTextChange : function(node, text, oldText){
35533         if(this.rendered){
35534             this.textNode.innerHTML = text;
35535         }
35536     },
35537
35538     onDisableChange : function(node, state){
35539         this.disabled = state;
35540         if(state){
35541             this.addClass("x-tree-node-disabled");
35542         }else{
35543             this.removeClass("x-tree-node-disabled");
35544         }
35545     },
35546
35547     onSelectedChange : function(state){
35548         if(state){
35549             this.focus();
35550             this.addClass("x-tree-selected");
35551         }else{
35552             //this.blur();
35553             this.removeClass("x-tree-selected");
35554         }
35555     },
35556
35557     onMove : function(tree, node, oldParent, newParent, index, refNode){
35558         this.childIndent = null;
35559         if(this.rendered){
35560             var targetNode = newParent.ui.getContainer();
35561             if(!targetNode){//target not rendered
35562                 this.holder = document.createElement("div");
35563                 this.holder.appendChild(this.wrap);
35564                 return;
35565             }
35566             var insertBefore = refNode ? refNode.ui.getEl() : null;
35567             if(insertBefore){
35568                 targetNode.insertBefore(this.wrap, insertBefore);
35569             }else{
35570                 targetNode.appendChild(this.wrap);
35571             }
35572             this.node.renderIndent(true);
35573         }
35574     },
35575
35576     addClass : function(cls){
35577         if(this.elNode){
35578             Roo.fly(this.elNode).addClass(cls);
35579         }
35580     },
35581
35582     removeClass : function(cls){
35583         if(this.elNode){
35584             Roo.fly(this.elNode).removeClass(cls);
35585         }
35586     },
35587
35588     remove : function(){
35589         if(this.rendered){
35590             this.holder = document.createElement("div");
35591             this.holder.appendChild(this.wrap);
35592         }
35593     },
35594
35595     fireEvent : function(){
35596         return this.node.fireEvent.apply(this.node, arguments);
35597     },
35598
35599     initEvents : function(){
35600         this.node.on("move", this.onMove, this);
35601         var E = Roo.EventManager;
35602         var a = this.anchor;
35603
35604         var el = Roo.fly(a, '_treeui');
35605
35606         if(Roo.isOpera){ // opera render bug ignores the CSS
35607             el.setStyle("text-decoration", "none");
35608         }
35609
35610         el.on("click", this.onClick, this);
35611         el.on("dblclick", this.onDblClick, this);
35612
35613         if(this.checkbox){
35614             Roo.EventManager.on(this.checkbox,
35615                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35616         }
35617
35618         el.on("contextmenu", this.onContextMenu, this);
35619
35620         var icon = Roo.fly(this.iconNode);
35621         icon.on("click", this.onClick, this);
35622         icon.on("dblclick", this.onDblClick, this);
35623         icon.on("contextmenu", this.onContextMenu, this);
35624         E.on(this.ecNode, "click", this.ecClick, this, true);
35625
35626         if(this.node.disabled){
35627             this.addClass("x-tree-node-disabled");
35628         }
35629         if(this.node.hidden){
35630             this.addClass("x-tree-node-disabled");
35631         }
35632         var ot = this.node.getOwnerTree();
35633         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35634         if(dd && (!this.node.isRoot || ot.rootVisible)){
35635             Roo.dd.Registry.register(this.elNode, {
35636                 node: this.node,
35637                 handles: this.getDDHandles(),
35638                 isHandle: false
35639             });
35640         }
35641     },
35642
35643     getDDHandles : function(){
35644         return [this.iconNode, this.textNode];
35645     },
35646
35647     hide : function(){
35648         if(this.rendered){
35649             this.wrap.style.display = "none";
35650         }
35651     },
35652
35653     show : function(){
35654         if(this.rendered){
35655             this.wrap.style.display = "";
35656         }
35657     },
35658
35659     onContextMenu : function(e){
35660         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35661             e.preventDefault();
35662             this.focus();
35663             this.fireEvent("contextmenu", this.node, e);
35664         }
35665     },
35666
35667     onClick : function(e){
35668         if(this.dropping){
35669             e.stopEvent();
35670             return;
35671         }
35672         if(this.fireEvent("beforeclick", this.node, e) !== false){
35673             if(!this.disabled && this.node.attributes.href){
35674                 this.fireEvent("click", this.node, e);
35675                 return;
35676             }
35677             e.preventDefault();
35678             if(this.disabled){
35679                 return;
35680             }
35681
35682             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35683                 this.node.toggle();
35684             }
35685
35686             this.fireEvent("click", this.node, e);
35687         }else{
35688             e.stopEvent();
35689         }
35690     },
35691
35692     onDblClick : function(e){
35693         e.preventDefault();
35694         if(this.disabled){
35695             return;
35696         }
35697         if(this.checkbox){
35698             this.toggleCheck();
35699         }
35700         if(!this.animating && this.node.hasChildNodes()){
35701             this.node.toggle();
35702         }
35703         this.fireEvent("dblclick", this.node, e);
35704     },
35705
35706     onCheckChange : function(){
35707         var checked = this.checkbox.checked;
35708         this.node.attributes.checked = checked;
35709         this.fireEvent('checkchange', this.node, checked);
35710     },
35711
35712     ecClick : function(e){
35713         if(!this.animating && this.node.hasChildNodes()){
35714             this.node.toggle();
35715         }
35716     },
35717
35718     startDrop : function(){
35719         this.dropping = true;
35720     },
35721
35722     // delayed drop so the click event doesn't get fired on a drop
35723     endDrop : function(){
35724        setTimeout(function(){
35725            this.dropping = false;
35726        }.createDelegate(this), 50);
35727     },
35728
35729     expand : function(){
35730         this.updateExpandIcon();
35731         this.ctNode.style.display = "";
35732     },
35733
35734     focus : function(){
35735         if(!this.node.preventHScroll){
35736             try{this.anchor.focus();
35737             }catch(e){}
35738         }else if(!Roo.isIE){
35739             try{
35740                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35741                 var l = noscroll.scrollLeft;
35742                 this.anchor.focus();
35743                 noscroll.scrollLeft = l;
35744             }catch(e){}
35745         }
35746     },
35747
35748     toggleCheck : function(value){
35749         var cb = this.checkbox;
35750         if(cb){
35751             cb.checked = (value === undefined ? !cb.checked : value);
35752         }
35753     },
35754
35755     blur : function(){
35756         try{
35757             this.anchor.blur();
35758         }catch(e){}
35759     },
35760
35761     animExpand : function(callback){
35762         var ct = Roo.get(this.ctNode);
35763         ct.stopFx();
35764         if(!this.node.hasChildNodes()){
35765             this.updateExpandIcon();
35766             this.ctNode.style.display = "";
35767             Roo.callback(callback);
35768             return;
35769         }
35770         this.animating = true;
35771         this.updateExpandIcon();
35772
35773         ct.slideIn('t', {
35774            callback : function(){
35775                this.animating = false;
35776                Roo.callback(callback);
35777             },
35778             scope: this,
35779             duration: this.node.ownerTree.duration || .25
35780         });
35781     },
35782
35783     highlight : function(){
35784         var tree = this.node.getOwnerTree();
35785         Roo.fly(this.wrap).highlight(
35786             tree.hlColor || "C3DAF9",
35787             {endColor: tree.hlBaseColor}
35788         );
35789     },
35790
35791     collapse : function(){
35792         this.updateExpandIcon();
35793         this.ctNode.style.display = "none";
35794     },
35795
35796     animCollapse : function(callback){
35797         var ct = Roo.get(this.ctNode);
35798         ct.enableDisplayMode('block');
35799         ct.stopFx();
35800
35801         this.animating = true;
35802         this.updateExpandIcon();
35803
35804         ct.slideOut('t', {
35805             callback : function(){
35806                this.animating = false;
35807                Roo.callback(callback);
35808             },
35809             scope: this,
35810             duration: this.node.ownerTree.duration || .25
35811         });
35812     },
35813
35814     getContainer : function(){
35815         return this.ctNode;
35816     },
35817
35818     getEl : function(){
35819         return this.wrap;
35820     },
35821
35822     appendDDGhost : function(ghostNode){
35823         ghostNode.appendChild(this.elNode.cloneNode(true));
35824     },
35825
35826     getDDRepairXY : function(){
35827         return Roo.lib.Dom.getXY(this.iconNode);
35828     },
35829
35830     onRender : function(){
35831         this.render();
35832     },
35833
35834     render : function(bulkRender){
35835         var n = this.node, a = n.attributes;
35836         var targetNode = n.parentNode ?
35837               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35838
35839         if(!this.rendered){
35840             this.rendered = true;
35841
35842             this.renderElements(n, a, targetNode, bulkRender);
35843
35844             if(a.qtip){
35845                if(this.textNode.setAttributeNS){
35846                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35847                    if(a.qtipTitle){
35848                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35849                    }
35850                }else{
35851                    this.textNode.setAttribute("ext:qtip", a.qtip);
35852                    if(a.qtipTitle){
35853                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35854                    }
35855                }
35856             }else if(a.qtipCfg){
35857                 a.qtipCfg.target = Roo.id(this.textNode);
35858                 Roo.QuickTips.register(a.qtipCfg);
35859             }
35860             this.initEvents();
35861             if(!this.node.expanded){
35862                 this.updateExpandIcon();
35863             }
35864         }else{
35865             if(bulkRender === true) {
35866                 targetNode.appendChild(this.wrap);
35867             }
35868         }
35869     },
35870
35871     renderElements : function(n, a, targetNode, bulkRender)
35872     {
35873         // add some indent caching, this helps performance when rendering a large tree
35874         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35875         var t = n.getOwnerTree();
35876         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35877         if (typeof(n.attributes.html) != 'undefined') {
35878             txt = n.attributes.html;
35879         }
35880         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35881         var cb = typeof a.checked == 'boolean';
35882         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35883         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35884             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35885             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35886             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35887             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35888             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35889              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35890                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35891             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35892             "</li>"];
35893
35894         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35895             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35896                                 n.nextSibling.ui.getEl(), buf.join(""));
35897         }else{
35898             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35899         }
35900
35901         this.elNode = this.wrap.childNodes[0];
35902         this.ctNode = this.wrap.childNodes[1];
35903         var cs = this.elNode.childNodes;
35904         this.indentNode = cs[0];
35905         this.ecNode = cs[1];
35906         this.iconNode = cs[2];
35907         var index = 3;
35908         if(cb){
35909             this.checkbox = cs[3];
35910             index++;
35911         }
35912         this.anchor = cs[index];
35913         this.textNode = cs[index].firstChild;
35914     },
35915
35916     getAnchor : function(){
35917         return this.anchor;
35918     },
35919
35920     getTextEl : function(){
35921         return this.textNode;
35922     },
35923
35924     getIconEl : function(){
35925         return this.iconNode;
35926     },
35927
35928     isChecked : function(){
35929         return this.checkbox ? this.checkbox.checked : false;
35930     },
35931
35932     updateExpandIcon : function(){
35933         if(this.rendered){
35934             var n = this.node, c1, c2;
35935             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35936             var hasChild = n.hasChildNodes();
35937             if(hasChild){
35938                 if(n.expanded){
35939                     cls += "-minus";
35940                     c1 = "x-tree-node-collapsed";
35941                     c2 = "x-tree-node-expanded";
35942                 }else{
35943                     cls += "-plus";
35944                     c1 = "x-tree-node-expanded";
35945                     c2 = "x-tree-node-collapsed";
35946                 }
35947                 if(this.wasLeaf){
35948                     this.removeClass("x-tree-node-leaf");
35949                     this.wasLeaf = false;
35950                 }
35951                 if(this.c1 != c1 || this.c2 != c2){
35952                     Roo.fly(this.elNode).replaceClass(c1, c2);
35953                     this.c1 = c1; this.c2 = c2;
35954                 }
35955             }else{
35956                 // this changes non-leafs into leafs if they have no children.
35957                 // it's not very rational behaviour..
35958                 
35959                 if(!this.wasLeaf && this.node.leaf){
35960                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
35961                     delete this.c1;
35962                     delete this.c2;
35963                     this.wasLeaf = true;
35964                 }
35965             }
35966             var ecc = "x-tree-ec-icon "+cls;
35967             if(this.ecc != ecc){
35968                 this.ecNode.className = ecc;
35969                 this.ecc = ecc;
35970             }
35971         }
35972     },
35973
35974     getChildIndent : function(){
35975         if(!this.childIndent){
35976             var buf = [];
35977             var p = this.node;
35978             while(p){
35979                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
35980                     if(!p.isLast()) {
35981                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
35982                     } else {
35983                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
35984                     }
35985                 }
35986                 p = p.parentNode;
35987             }
35988             this.childIndent = buf.join("");
35989         }
35990         return this.childIndent;
35991     },
35992
35993     renderIndent : function(){
35994         if(this.rendered){
35995             var indent = "";
35996             var p = this.node.parentNode;
35997             if(p){
35998                 indent = p.ui.getChildIndent();
35999             }
36000             if(this.indentMarkup != indent){ // don't rerender if not required
36001                 this.indentNode.innerHTML = indent;
36002                 this.indentMarkup = indent;
36003             }
36004             this.updateExpandIcon();
36005         }
36006     }
36007 };
36008
36009 Roo.tree.RootTreeNodeUI = function(){
36010     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36011 };
36012 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36013     render : function(){
36014         if(!this.rendered){
36015             var targetNode = this.node.ownerTree.innerCt.dom;
36016             this.node.expanded = true;
36017             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36018             this.wrap = this.ctNode = targetNode.firstChild;
36019         }
36020     },
36021     collapse : function(){
36022     },
36023     expand : function(){
36024     }
36025 });/*
36026  * Based on:
36027  * Ext JS Library 1.1.1
36028  * Copyright(c) 2006-2007, Ext JS, LLC.
36029  *
36030  * Originally Released Under LGPL - original licence link has changed is not relivant.
36031  *
36032  * Fork - LGPL
36033  * <script type="text/javascript">
36034  */
36035 /**
36036  * @class Roo.tree.TreeLoader
36037  * @extends Roo.util.Observable
36038  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36039  * nodes from a specified URL. The response must be a javascript Array definition
36040  * who's elements are node definition objects. eg:
36041  * <pre><code>
36042 {  success : true,
36043    data :      [
36044    
36045     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36046     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36047     ]
36048 }
36049
36050
36051 </code></pre>
36052  * <br><br>
36053  * The old style respose with just an array is still supported, but not recommended.
36054  * <br><br>
36055  *
36056  * A server request is sent, and child nodes are loaded only when a node is expanded.
36057  * The loading node's id is passed to the server under the parameter name "node" to
36058  * enable the server to produce the correct child nodes.
36059  * <br><br>
36060  * To pass extra parameters, an event handler may be attached to the "beforeload"
36061  * event, and the parameters specified in the TreeLoader's baseParams property:
36062  * <pre><code>
36063     myTreeLoader.on("beforeload", function(treeLoader, node) {
36064         this.baseParams.category = node.attributes.category;
36065     }, this);
36066     
36067 </code></pre>
36068  *
36069  * This would pass an HTTP parameter called "category" to the server containing
36070  * the value of the Node's "category" attribute.
36071  * @constructor
36072  * Creates a new Treeloader.
36073  * @param {Object} config A config object containing config properties.
36074  */
36075 Roo.tree.TreeLoader = function(config){
36076     this.baseParams = {};
36077     this.requestMethod = "POST";
36078     Roo.apply(this, config);
36079
36080     this.addEvents({
36081     
36082         /**
36083          * @event beforeload
36084          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36085          * @param {Object} This TreeLoader object.
36086          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36087          * @param {Object} callback The callback function specified in the {@link #load} call.
36088          */
36089         beforeload : true,
36090         /**
36091          * @event load
36092          * Fires when the node has been successfuly loaded.
36093          * @param {Object} This TreeLoader object.
36094          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36095          * @param {Object} response The response object containing the data from the server.
36096          */
36097         load : true,
36098         /**
36099          * @event loadexception
36100          * Fires if the network request failed.
36101          * @param {Object} This TreeLoader object.
36102          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36103          * @param {Object} response The response object containing the data from the server.
36104          */
36105         loadexception : true,
36106         /**
36107          * @event create
36108          * Fires before a node is created, enabling you to return custom Node types 
36109          * @param {Object} This TreeLoader object.
36110          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36111          */
36112         create : true
36113     });
36114
36115     Roo.tree.TreeLoader.superclass.constructor.call(this);
36116 };
36117
36118 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36119     /**
36120     * @cfg {String} dataUrl The URL from which to request a Json string which
36121     * specifies an array of node definition object representing the child nodes
36122     * to be loaded.
36123     */
36124     /**
36125     * @cfg {String} requestMethod either GET or POST
36126     * defaults to POST (due to BC)
36127     * to be loaded.
36128     */
36129     /**
36130     * @cfg {Object} baseParams (optional) An object containing properties which
36131     * specify HTTP parameters to be passed to each request for child nodes.
36132     */
36133     /**
36134     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36135     * created by this loader. If the attributes sent by the server have an attribute in this object,
36136     * they take priority.
36137     */
36138     /**
36139     * @cfg {Object} uiProviders (optional) An object containing properties which
36140     * 
36141     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36142     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36143     * <i>uiProvider</i> attribute of a returned child node is a string rather
36144     * than a reference to a TreeNodeUI implementation, this that string value
36145     * is used as a property name in the uiProviders object. You can define the provider named
36146     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36147     */
36148     uiProviders : {},
36149
36150     /**
36151     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36152     * child nodes before loading.
36153     */
36154     clearOnLoad : true,
36155
36156     /**
36157     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36158     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36159     * Grid query { data : [ .....] }
36160     */
36161     
36162     root : false,
36163      /**
36164     * @cfg {String} queryParam (optional) 
36165     * Name of the query as it will be passed on the querystring (defaults to 'node')
36166     * eg. the request will be ?node=[id]
36167     */
36168     
36169     
36170     queryParam: false,
36171     
36172     /**
36173      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36174      * This is called automatically when a node is expanded, but may be used to reload
36175      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36176      * @param {Roo.tree.TreeNode} node
36177      * @param {Function} callback
36178      */
36179     load : function(node, callback){
36180         if(this.clearOnLoad){
36181             while(node.firstChild){
36182                 node.removeChild(node.firstChild);
36183             }
36184         }
36185         if(node.attributes.children){ // preloaded json children
36186             var cs = node.attributes.children;
36187             for(var i = 0, len = cs.length; i < len; i++){
36188                 node.appendChild(this.createNode(cs[i]));
36189             }
36190             if(typeof callback == "function"){
36191                 callback();
36192             }
36193         }else if(this.dataUrl){
36194             this.requestData(node, callback);
36195         }
36196     },
36197
36198     getParams: function(node){
36199         var buf = [], bp = this.baseParams;
36200         for(var key in bp){
36201             if(typeof bp[key] != "function"){
36202                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36203             }
36204         }
36205         var n = this.queryParam === false ? 'node' : this.queryParam;
36206         buf.push(n + "=", encodeURIComponent(node.id));
36207         return buf.join("");
36208     },
36209
36210     requestData : function(node, callback){
36211         if(this.fireEvent("beforeload", this, node, callback) !== false){
36212             this.transId = Roo.Ajax.request({
36213                 method:this.requestMethod,
36214                 url: this.dataUrl||this.url,
36215                 success: this.handleResponse,
36216                 failure: this.handleFailure,
36217                 scope: this,
36218                 argument: {callback: callback, node: node},
36219                 params: this.getParams(node)
36220             });
36221         }else{
36222             // if the load is cancelled, make sure we notify
36223             // the node that we are done
36224             if(typeof callback == "function"){
36225                 callback();
36226             }
36227         }
36228     },
36229
36230     isLoading : function(){
36231         return this.transId ? true : false;
36232     },
36233
36234     abort : function(){
36235         if(this.isLoading()){
36236             Roo.Ajax.abort(this.transId);
36237         }
36238     },
36239
36240     // private
36241     createNode : function(attr)
36242     {
36243         // apply baseAttrs, nice idea Corey!
36244         if(this.baseAttrs){
36245             Roo.applyIf(attr, this.baseAttrs);
36246         }
36247         if(this.applyLoader !== false){
36248             attr.loader = this;
36249         }
36250         // uiProvider = depreciated..
36251         
36252         if(typeof(attr.uiProvider) == 'string'){
36253            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36254                 /**  eval:var:attr */ eval(attr.uiProvider);
36255         }
36256         if(typeof(this.uiProviders['default']) != 'undefined') {
36257             attr.uiProvider = this.uiProviders['default'];
36258         }
36259         
36260         this.fireEvent('create', this, attr);
36261         
36262         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36263         return(attr.leaf ?
36264                         new Roo.tree.TreeNode(attr) :
36265                         new Roo.tree.AsyncTreeNode(attr));
36266     },
36267
36268     processResponse : function(response, node, callback)
36269     {
36270         var json = response.responseText;
36271         try {
36272             
36273             var o = Roo.decode(json);
36274             
36275             if (this.root === false && typeof(o.success) != undefined) {
36276                 this.root = 'data'; // the default behaviour for list like data..
36277                 }
36278                 
36279             if (this.root !== false &&  !o.success) {
36280                 // it's a failure condition.
36281                 var a = response.argument;
36282                 this.fireEvent("loadexception", this, a.node, response);
36283                 Roo.log("Load failed - should have a handler really");
36284                 return;
36285             }
36286             
36287             
36288             
36289             if (this.root !== false) {
36290                  o = o[this.root];
36291             }
36292             
36293             for(var i = 0, len = o.length; i < len; i++){
36294                 var n = this.createNode(o[i]);
36295                 if(n){
36296                     node.appendChild(n);
36297                 }
36298             }
36299             if(typeof callback == "function"){
36300                 callback(this, node);
36301             }
36302         }catch(e){
36303             this.handleFailure(response);
36304         }
36305     },
36306
36307     handleResponse : function(response){
36308         this.transId = false;
36309         var a = response.argument;
36310         this.processResponse(response, a.node, a.callback);
36311         this.fireEvent("load", this, a.node, response);
36312     },
36313
36314     handleFailure : function(response)
36315     {
36316         // should handle failure better..
36317         this.transId = false;
36318         var a = response.argument;
36319         this.fireEvent("loadexception", this, a.node, response);
36320         if(typeof a.callback == "function"){
36321             a.callback(this, a.node);
36322         }
36323     }
36324 });/*
36325  * Based on:
36326  * Ext JS Library 1.1.1
36327  * Copyright(c) 2006-2007, Ext JS, LLC.
36328  *
36329  * Originally Released Under LGPL - original licence link has changed is not relivant.
36330  *
36331  * Fork - LGPL
36332  * <script type="text/javascript">
36333  */
36334
36335 /**
36336 * @class Roo.tree.TreeFilter
36337 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36338 * @param {TreePanel} tree
36339 * @param {Object} config (optional)
36340  */
36341 Roo.tree.TreeFilter = function(tree, config){
36342     this.tree = tree;
36343     this.filtered = {};
36344     Roo.apply(this, config);
36345 };
36346
36347 Roo.tree.TreeFilter.prototype = {
36348     clearBlank:false,
36349     reverse:false,
36350     autoClear:false,
36351     remove:false,
36352
36353      /**
36354      * Filter the data by a specific attribute.
36355      * @param {String/RegExp} value Either string that the attribute value
36356      * should start with or a RegExp to test against the attribute
36357      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36358      * @param {TreeNode} startNode (optional) The node to start the filter at.
36359      */
36360     filter : function(value, attr, startNode){
36361         attr = attr || "text";
36362         var f;
36363         if(typeof value == "string"){
36364             var vlen = value.length;
36365             // auto clear empty filter
36366             if(vlen == 0 && this.clearBlank){
36367                 this.clear();
36368                 return;
36369             }
36370             value = value.toLowerCase();
36371             f = function(n){
36372                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36373             };
36374         }else if(value.exec){ // regex?
36375             f = function(n){
36376                 return value.test(n.attributes[attr]);
36377             };
36378         }else{
36379             throw 'Illegal filter type, must be string or regex';
36380         }
36381         this.filterBy(f, null, startNode);
36382         },
36383
36384     /**
36385      * Filter by a function. The passed function will be called with each
36386      * node in the tree (or from the startNode). If the function returns true, the node is kept
36387      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36388      * @param {Function} fn The filter function
36389      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36390      */
36391     filterBy : function(fn, scope, startNode){
36392         startNode = startNode || this.tree.root;
36393         if(this.autoClear){
36394             this.clear();
36395         }
36396         var af = this.filtered, rv = this.reverse;
36397         var f = function(n){
36398             if(n == startNode){
36399                 return true;
36400             }
36401             if(af[n.id]){
36402                 return false;
36403             }
36404             var m = fn.call(scope || n, n);
36405             if(!m || rv){
36406                 af[n.id] = n;
36407                 n.ui.hide();
36408                 return false;
36409             }
36410             return true;
36411         };
36412         startNode.cascade(f);
36413         if(this.remove){
36414            for(var id in af){
36415                if(typeof id != "function"){
36416                    var n = af[id];
36417                    if(n && n.parentNode){
36418                        n.parentNode.removeChild(n);
36419                    }
36420                }
36421            }
36422         }
36423     },
36424
36425     /**
36426      * Clears the current filter. Note: with the "remove" option
36427      * set a filter cannot be cleared.
36428      */
36429     clear : function(){
36430         var t = this.tree;
36431         var af = this.filtered;
36432         for(var id in af){
36433             if(typeof id != "function"){
36434                 var n = af[id];
36435                 if(n){
36436                     n.ui.show();
36437                 }
36438             }
36439         }
36440         this.filtered = {};
36441     }
36442 };
36443 /*
36444  * Based on:
36445  * Ext JS Library 1.1.1
36446  * Copyright(c) 2006-2007, Ext JS, LLC.
36447  *
36448  * Originally Released Under LGPL - original licence link has changed is not relivant.
36449  *
36450  * Fork - LGPL
36451  * <script type="text/javascript">
36452  */
36453  
36454
36455 /**
36456  * @class Roo.tree.TreeSorter
36457  * Provides sorting of nodes in a TreePanel
36458  * 
36459  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36460  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36461  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36462  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36463  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36464  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36465  * @constructor
36466  * @param {TreePanel} tree
36467  * @param {Object} config
36468  */
36469 Roo.tree.TreeSorter = function(tree, config){
36470     Roo.apply(this, config);
36471     tree.on("beforechildrenrendered", this.doSort, this);
36472     tree.on("append", this.updateSort, this);
36473     tree.on("insert", this.updateSort, this);
36474     
36475     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36476     var p = this.property || "text";
36477     var sortType = this.sortType;
36478     var fs = this.folderSort;
36479     var cs = this.caseSensitive === true;
36480     var leafAttr = this.leafAttr || 'leaf';
36481
36482     this.sortFn = function(n1, n2){
36483         if(fs){
36484             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36485                 return 1;
36486             }
36487             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36488                 return -1;
36489             }
36490         }
36491         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36492         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36493         if(v1 < v2){
36494                         return dsc ? +1 : -1;
36495                 }else if(v1 > v2){
36496                         return dsc ? -1 : +1;
36497         }else{
36498                 return 0;
36499         }
36500     };
36501 };
36502
36503 Roo.tree.TreeSorter.prototype = {
36504     doSort : function(node){
36505         node.sort(this.sortFn);
36506     },
36507     
36508     compareNodes : function(n1, n2){
36509         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36510     },
36511     
36512     updateSort : function(tree, node){
36513         if(node.childrenRendered){
36514             this.doSort.defer(1, this, [node]);
36515         }
36516     }
36517 };/*
36518  * Based on:
36519  * Ext JS Library 1.1.1
36520  * Copyright(c) 2006-2007, Ext JS, LLC.
36521  *
36522  * Originally Released Under LGPL - original licence link has changed is not relivant.
36523  *
36524  * Fork - LGPL
36525  * <script type="text/javascript">
36526  */
36527
36528 if(Roo.dd.DropZone){
36529     
36530 Roo.tree.TreeDropZone = function(tree, config){
36531     this.allowParentInsert = false;
36532     this.allowContainerDrop = false;
36533     this.appendOnly = false;
36534     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36535     this.tree = tree;
36536     this.lastInsertClass = "x-tree-no-status";
36537     this.dragOverData = {};
36538 };
36539
36540 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36541     ddGroup : "TreeDD",
36542     scroll:  true,
36543     
36544     expandDelay : 1000,
36545     
36546     expandNode : function(node){
36547         if(node.hasChildNodes() && !node.isExpanded()){
36548             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36549         }
36550     },
36551     
36552     queueExpand : function(node){
36553         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36554     },
36555     
36556     cancelExpand : function(){
36557         if(this.expandProcId){
36558             clearTimeout(this.expandProcId);
36559             this.expandProcId = false;
36560         }
36561     },
36562     
36563     isValidDropPoint : function(n, pt, dd, e, data){
36564         if(!n || !data){ return false; }
36565         var targetNode = n.node;
36566         var dropNode = data.node;
36567         // default drop rules
36568         if(!(targetNode && targetNode.isTarget && pt)){
36569             return false;
36570         }
36571         if(pt == "append" && targetNode.allowChildren === false){
36572             return false;
36573         }
36574         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36575             return false;
36576         }
36577         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36578             return false;
36579         }
36580         // reuse the object
36581         var overEvent = this.dragOverData;
36582         overEvent.tree = this.tree;
36583         overEvent.target = targetNode;
36584         overEvent.data = data;
36585         overEvent.point = pt;
36586         overEvent.source = dd;
36587         overEvent.rawEvent = e;
36588         overEvent.dropNode = dropNode;
36589         overEvent.cancel = false;  
36590         var result = this.tree.fireEvent("nodedragover", overEvent);
36591         return overEvent.cancel === false && result !== false;
36592     },
36593     
36594     getDropPoint : function(e, n, dd)
36595     {
36596         var tn = n.node;
36597         if(tn.isRoot){
36598             return tn.allowChildren !== false ? "append" : false; // always append for root
36599         }
36600         var dragEl = n.ddel;
36601         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36602         var y = Roo.lib.Event.getPageY(e);
36603         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36604         
36605         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36606         var noAppend = tn.allowChildren === false;
36607         if(this.appendOnly || tn.parentNode.allowChildren === false){
36608             return noAppend ? false : "append";
36609         }
36610         var noBelow = false;
36611         if(!this.allowParentInsert){
36612             noBelow = tn.hasChildNodes() && tn.isExpanded();
36613         }
36614         var q = (b - t) / (noAppend ? 2 : 3);
36615         if(y >= t && y < (t + q)){
36616             return "above";
36617         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36618             return "below";
36619         }else{
36620             return "append";
36621         }
36622     },
36623     
36624     onNodeEnter : function(n, dd, e, data)
36625     {
36626         this.cancelExpand();
36627     },
36628     
36629     onNodeOver : function(n, dd, e, data)
36630     {
36631        
36632         var pt = this.getDropPoint(e, n, dd);
36633         var node = n.node;
36634         
36635         // auto node expand check
36636         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36637             this.queueExpand(node);
36638         }else if(pt != "append"){
36639             this.cancelExpand();
36640         }
36641         
36642         // set the insert point style on the target node
36643         var returnCls = this.dropNotAllowed;
36644         if(this.isValidDropPoint(n, pt, dd, e, data)){
36645            if(pt){
36646                var el = n.ddel;
36647                var cls;
36648                if(pt == "above"){
36649                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36650                    cls = "x-tree-drag-insert-above";
36651                }else if(pt == "below"){
36652                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36653                    cls = "x-tree-drag-insert-below";
36654                }else{
36655                    returnCls = "x-tree-drop-ok-append";
36656                    cls = "x-tree-drag-append";
36657                }
36658                if(this.lastInsertClass != cls){
36659                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36660                    this.lastInsertClass = cls;
36661                }
36662            }
36663        }
36664        return returnCls;
36665     },
36666     
36667     onNodeOut : function(n, dd, e, data){
36668         
36669         this.cancelExpand();
36670         this.removeDropIndicators(n);
36671     },
36672     
36673     onNodeDrop : function(n, dd, e, data){
36674         var point = this.getDropPoint(e, n, dd);
36675         var targetNode = n.node;
36676         targetNode.ui.startDrop();
36677         if(!this.isValidDropPoint(n, point, dd, e, data)){
36678             targetNode.ui.endDrop();
36679             return false;
36680         }
36681         // first try to find the drop node
36682         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36683         var dropEvent = {
36684             tree : this.tree,
36685             target: targetNode,
36686             data: data,
36687             point: point,
36688             source: dd,
36689             rawEvent: e,
36690             dropNode: dropNode,
36691             cancel: !dropNode   
36692         };
36693         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36694         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36695             targetNode.ui.endDrop();
36696             return false;
36697         }
36698         // allow target changing
36699         targetNode = dropEvent.target;
36700         if(point == "append" && !targetNode.isExpanded()){
36701             targetNode.expand(false, null, function(){
36702                 this.completeDrop(dropEvent);
36703             }.createDelegate(this));
36704         }else{
36705             this.completeDrop(dropEvent);
36706         }
36707         return true;
36708     },
36709     
36710     completeDrop : function(de){
36711         var ns = de.dropNode, p = de.point, t = de.target;
36712         if(!(ns instanceof Array)){
36713             ns = [ns];
36714         }
36715         var n;
36716         for(var i = 0, len = ns.length; i < len; i++){
36717             n = ns[i];
36718             if(p == "above"){
36719                 t.parentNode.insertBefore(n, t);
36720             }else if(p == "below"){
36721                 t.parentNode.insertBefore(n, t.nextSibling);
36722             }else{
36723                 t.appendChild(n);
36724             }
36725         }
36726         n.ui.focus();
36727         if(this.tree.hlDrop){
36728             n.ui.highlight();
36729         }
36730         t.ui.endDrop();
36731         this.tree.fireEvent("nodedrop", de);
36732     },
36733     
36734     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36735         if(this.tree.hlDrop){
36736             dropNode.ui.focus();
36737             dropNode.ui.highlight();
36738         }
36739         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36740     },
36741     
36742     getTree : function(){
36743         return this.tree;
36744     },
36745     
36746     removeDropIndicators : function(n){
36747         if(n && n.ddel){
36748             var el = n.ddel;
36749             Roo.fly(el).removeClass([
36750                     "x-tree-drag-insert-above",
36751                     "x-tree-drag-insert-below",
36752                     "x-tree-drag-append"]);
36753             this.lastInsertClass = "_noclass";
36754         }
36755     },
36756     
36757     beforeDragDrop : function(target, e, id){
36758         this.cancelExpand();
36759         return true;
36760     },
36761     
36762     afterRepair : function(data){
36763         if(data && Roo.enableFx){
36764             data.node.ui.highlight();
36765         }
36766         this.hideProxy();
36767     } 
36768     
36769 });
36770
36771 }
36772 /*
36773  * Based on:
36774  * Ext JS Library 1.1.1
36775  * Copyright(c) 2006-2007, Ext JS, LLC.
36776  *
36777  * Originally Released Under LGPL - original licence link has changed is not relivant.
36778  *
36779  * Fork - LGPL
36780  * <script type="text/javascript">
36781  */
36782  
36783
36784 if(Roo.dd.DragZone){
36785 Roo.tree.TreeDragZone = function(tree, config){
36786     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36787     this.tree = tree;
36788 };
36789
36790 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36791     ddGroup : "TreeDD",
36792    
36793     onBeforeDrag : function(data, e){
36794         var n = data.node;
36795         return n && n.draggable && !n.disabled;
36796     },
36797      
36798     
36799     onInitDrag : function(e){
36800         var data = this.dragData;
36801         this.tree.getSelectionModel().select(data.node);
36802         this.proxy.update("");
36803         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36804         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36805     },
36806     
36807     getRepairXY : function(e, data){
36808         return data.node.ui.getDDRepairXY();
36809     },
36810     
36811     onEndDrag : function(data, e){
36812         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36813         
36814         
36815     },
36816     
36817     onValidDrop : function(dd, e, id){
36818         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36819         this.hideProxy();
36820     },
36821     
36822     beforeInvalidDrop : function(e, id){
36823         // this scrolls the original position back into view
36824         var sm = this.tree.getSelectionModel();
36825         sm.clearSelections();
36826         sm.select(this.dragData.node);
36827     }
36828 });
36829 }/*
36830  * Based on:
36831  * Ext JS Library 1.1.1
36832  * Copyright(c) 2006-2007, Ext JS, LLC.
36833  *
36834  * Originally Released Under LGPL - original licence link has changed is not relivant.
36835  *
36836  * Fork - LGPL
36837  * <script type="text/javascript">
36838  */
36839 /**
36840  * @class Roo.tree.TreeEditor
36841  * @extends Roo.Editor
36842  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36843  * as the editor field.
36844  * @constructor
36845  * @param {Object} config (used to be the tree panel.)
36846  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36847  * 
36848  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36849  * @cfg {Roo.form.TextField|Object} field The field configuration
36850  *
36851  * 
36852  */
36853 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36854     var tree = config;
36855     var field;
36856     if (oldconfig) { // old style..
36857         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36858     } else {
36859         // new style..
36860         tree = config.tree;
36861         config.field = config.field  || {};
36862         config.field.xtype = 'TextField';
36863         field = Roo.factory(config.field, Roo.form);
36864     }
36865     config = config || {};
36866     
36867     
36868     this.addEvents({
36869         /**
36870          * @event beforenodeedit
36871          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36872          * false from the handler of this event.
36873          * @param {Editor} this
36874          * @param {Roo.tree.Node} node 
36875          */
36876         "beforenodeedit" : true
36877     });
36878     
36879     //Roo.log(config);
36880     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36881
36882     this.tree = tree;
36883
36884     tree.on('beforeclick', this.beforeNodeClick, this);
36885     tree.getTreeEl().on('mousedown', this.hide, this);
36886     this.on('complete', this.updateNode, this);
36887     this.on('beforestartedit', this.fitToTree, this);
36888     this.on('startedit', this.bindScroll, this, {delay:10});
36889     this.on('specialkey', this.onSpecialKey, this);
36890 };
36891
36892 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36893     /**
36894      * @cfg {String} alignment
36895      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36896      */
36897     alignment: "l-l",
36898     // inherit
36899     autoSize: false,
36900     /**
36901      * @cfg {Boolean} hideEl
36902      * True to hide the bound element while the editor is displayed (defaults to false)
36903      */
36904     hideEl : false,
36905     /**
36906      * @cfg {String} cls
36907      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36908      */
36909     cls: "x-small-editor x-tree-editor",
36910     /**
36911      * @cfg {Boolean} shim
36912      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36913      */
36914     shim:false,
36915     // inherit
36916     shadow:"frame",
36917     /**
36918      * @cfg {Number} maxWidth
36919      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36920      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36921      * scroll and client offsets into account prior to each edit.
36922      */
36923     maxWidth: 250,
36924
36925     editDelay : 350,
36926
36927     // private
36928     fitToTree : function(ed, el){
36929         var td = this.tree.getTreeEl().dom, nd = el.dom;
36930         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36931             td.scrollLeft = nd.offsetLeft;
36932         }
36933         var w = Math.min(
36934                 this.maxWidth,
36935                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36936         this.setSize(w, '');
36937         
36938         return this.fireEvent('beforenodeedit', this, this.editNode);
36939         
36940     },
36941
36942     // private
36943     triggerEdit : function(node){
36944         this.completeEdit();
36945         this.editNode = node;
36946         this.startEdit(node.ui.textNode, node.text);
36947     },
36948
36949     // private
36950     bindScroll : function(){
36951         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36952     },
36953
36954     // private
36955     beforeNodeClick : function(node, e){
36956         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
36957         this.lastClick = new Date();
36958         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
36959             e.stopEvent();
36960             this.triggerEdit(node);
36961             return false;
36962         }
36963         return true;
36964     },
36965
36966     // private
36967     updateNode : function(ed, value){
36968         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
36969         this.editNode.setText(value);
36970     },
36971
36972     // private
36973     onHide : function(){
36974         Roo.tree.TreeEditor.superclass.onHide.call(this);
36975         if(this.editNode){
36976             this.editNode.ui.focus();
36977         }
36978     },
36979
36980     // private
36981     onSpecialKey : function(field, e){
36982         var k = e.getKey();
36983         if(k == e.ESC){
36984             e.stopEvent();
36985             this.cancelEdit();
36986         }else if(k == e.ENTER && !e.hasModifier()){
36987             e.stopEvent();
36988             this.completeEdit();
36989         }
36990     }
36991 });//<Script type="text/javascript">
36992 /*
36993  * Based on:
36994  * Ext JS Library 1.1.1
36995  * Copyright(c) 2006-2007, Ext JS, LLC.
36996  *
36997  * Originally Released Under LGPL - original licence link has changed is not relivant.
36998  *
36999  * Fork - LGPL
37000  * <script type="text/javascript">
37001  */
37002  
37003 /**
37004  * Not documented??? - probably should be...
37005  */
37006
37007 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37008     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37009     
37010     renderElements : function(n, a, targetNode, bulkRender){
37011         //consel.log("renderElements?");
37012         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37013
37014         var t = n.getOwnerTree();
37015         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37016         
37017         var cols = t.columns;
37018         var bw = t.borderWidth;
37019         var c = cols[0];
37020         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37021          var cb = typeof a.checked == "boolean";
37022         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37023         var colcls = 'x-t-' + tid + '-c0';
37024         var buf = [
37025             '<li class="x-tree-node">',
37026             
37027                 
37028                 '<div class="x-tree-node-el ', a.cls,'">',
37029                     // extran...
37030                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37031                 
37032                 
37033                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37034                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37035                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37036                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37037                            (a.iconCls ? ' '+a.iconCls : ''),
37038                            '" unselectable="on" />',
37039                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37040                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37041                              
37042                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37043                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37044                             '<span unselectable="on" qtip="' + tx + '">',
37045                              tx,
37046                              '</span></a>' ,
37047                     '</div>',
37048                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37049                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37050                  ];
37051         for(var i = 1, len = cols.length; i < len; i++){
37052             c = cols[i];
37053             colcls = 'x-t-' + tid + '-c' +i;
37054             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37055             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37056                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37057                       "</div>");
37058          }
37059          
37060          buf.push(
37061             '</a>',
37062             '<div class="x-clear"></div></div>',
37063             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37064             "</li>");
37065         
37066         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37067             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37068                                 n.nextSibling.ui.getEl(), buf.join(""));
37069         }else{
37070             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37071         }
37072         var el = this.wrap.firstChild;
37073         this.elRow = el;
37074         this.elNode = el.firstChild;
37075         this.ranchor = el.childNodes[1];
37076         this.ctNode = this.wrap.childNodes[1];
37077         var cs = el.firstChild.childNodes;
37078         this.indentNode = cs[0];
37079         this.ecNode = cs[1];
37080         this.iconNode = cs[2];
37081         var index = 3;
37082         if(cb){
37083             this.checkbox = cs[3];
37084             index++;
37085         }
37086         this.anchor = cs[index];
37087         
37088         this.textNode = cs[index].firstChild;
37089         
37090         //el.on("click", this.onClick, this);
37091         //el.on("dblclick", this.onDblClick, this);
37092         
37093         
37094        // console.log(this);
37095     },
37096     initEvents : function(){
37097         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37098         
37099             
37100         var a = this.ranchor;
37101
37102         var el = Roo.get(a);
37103
37104         if(Roo.isOpera){ // opera render bug ignores the CSS
37105             el.setStyle("text-decoration", "none");
37106         }
37107
37108         el.on("click", this.onClick, this);
37109         el.on("dblclick", this.onDblClick, this);
37110         el.on("contextmenu", this.onContextMenu, this);
37111         
37112     },
37113     
37114     /*onSelectedChange : function(state){
37115         if(state){
37116             this.focus();
37117             this.addClass("x-tree-selected");
37118         }else{
37119             //this.blur();
37120             this.removeClass("x-tree-selected");
37121         }
37122     },*/
37123     addClass : function(cls){
37124         if(this.elRow){
37125             Roo.fly(this.elRow).addClass(cls);
37126         }
37127         
37128     },
37129     
37130     
37131     removeClass : function(cls){
37132         if(this.elRow){
37133             Roo.fly(this.elRow).removeClass(cls);
37134         }
37135     }
37136
37137     
37138     
37139 });//<Script type="text/javascript">
37140
37141 /*
37142  * Based on:
37143  * Ext JS Library 1.1.1
37144  * Copyright(c) 2006-2007, Ext JS, LLC.
37145  *
37146  * Originally Released Under LGPL - original licence link has changed is not relivant.
37147  *
37148  * Fork - LGPL
37149  * <script type="text/javascript">
37150  */
37151  
37152
37153 /**
37154  * @class Roo.tree.ColumnTree
37155  * @extends Roo.data.TreePanel
37156  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37157  * @cfg {int} borderWidth  compined right/left border allowance
37158  * @constructor
37159  * @param {String/HTMLElement/Element} el The container element
37160  * @param {Object} config
37161  */
37162 Roo.tree.ColumnTree =  function(el, config)
37163 {
37164    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37165    this.addEvents({
37166         /**
37167         * @event resize
37168         * Fire this event on a container when it resizes
37169         * @param {int} w Width
37170         * @param {int} h Height
37171         */
37172        "resize" : true
37173     });
37174     this.on('resize', this.onResize, this);
37175 };
37176
37177 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37178     //lines:false,
37179     
37180     
37181     borderWidth: Roo.isBorderBox ? 0 : 2, 
37182     headEls : false,
37183     
37184     render : function(){
37185         // add the header.....
37186        
37187         Roo.tree.ColumnTree.superclass.render.apply(this);
37188         
37189         this.el.addClass('x-column-tree');
37190         
37191         this.headers = this.el.createChild(
37192             {cls:'x-tree-headers'},this.innerCt.dom);
37193    
37194         var cols = this.columns, c;
37195         var totalWidth = 0;
37196         this.headEls = [];
37197         var  len = cols.length;
37198         for(var i = 0; i < len; i++){
37199              c = cols[i];
37200              totalWidth += c.width;
37201             this.headEls.push(this.headers.createChild({
37202                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37203                  cn: {
37204                      cls:'x-tree-hd-text',
37205                      html: c.header
37206                  },
37207                  style:'width:'+(c.width-this.borderWidth)+'px;'
37208              }));
37209         }
37210         this.headers.createChild({cls:'x-clear'});
37211         // prevent floats from wrapping when clipped
37212         this.headers.setWidth(totalWidth);
37213         //this.innerCt.setWidth(totalWidth);
37214         this.innerCt.setStyle({ overflow: 'auto' });
37215         this.onResize(this.width, this.height);
37216              
37217         
37218     },
37219     onResize : function(w,h)
37220     {
37221         this.height = h;
37222         this.width = w;
37223         // resize cols..
37224         this.innerCt.setWidth(this.width);
37225         this.innerCt.setHeight(this.height-20);
37226         
37227         // headers...
37228         var cols = this.columns, c;
37229         var totalWidth = 0;
37230         var expEl = false;
37231         var len = cols.length;
37232         for(var i = 0; i < len; i++){
37233             c = cols[i];
37234             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37235                 // it's the expander..
37236                 expEl  = this.headEls[i];
37237                 continue;
37238             }
37239             totalWidth += c.width;
37240             
37241         }
37242         if (expEl) {
37243             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37244         }
37245         this.headers.setWidth(w-20);
37246
37247         
37248         
37249         
37250     }
37251 });
37252 /*
37253  * Based on:
37254  * Ext JS Library 1.1.1
37255  * Copyright(c) 2006-2007, Ext JS, LLC.
37256  *
37257  * Originally Released Under LGPL - original licence link has changed is not relivant.
37258  *
37259  * Fork - LGPL
37260  * <script type="text/javascript">
37261  */
37262  
37263 /**
37264  * @class Roo.menu.Menu
37265  * @extends Roo.util.Observable
37266  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37267  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37268  * @constructor
37269  * Creates a new Menu
37270  * @param {Object} config Configuration options
37271  */
37272 Roo.menu.Menu = function(config){
37273     
37274     Roo.menu.Menu.superclass.constructor.call(this, config);
37275     
37276     this.id = this.id || Roo.id();
37277     this.addEvents({
37278         /**
37279          * @event beforeshow
37280          * Fires before this menu is displayed
37281          * @param {Roo.menu.Menu} this
37282          */
37283         beforeshow : true,
37284         /**
37285          * @event beforehide
37286          * Fires before this menu is hidden
37287          * @param {Roo.menu.Menu} this
37288          */
37289         beforehide : true,
37290         /**
37291          * @event show
37292          * Fires after this menu is displayed
37293          * @param {Roo.menu.Menu} this
37294          */
37295         show : true,
37296         /**
37297          * @event hide
37298          * Fires after this menu is hidden
37299          * @param {Roo.menu.Menu} this
37300          */
37301         hide : true,
37302         /**
37303          * @event click
37304          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37305          * @param {Roo.menu.Menu} this
37306          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37307          * @param {Roo.EventObject} e
37308          */
37309         click : true,
37310         /**
37311          * @event mouseover
37312          * Fires when the mouse is hovering over this menu
37313          * @param {Roo.menu.Menu} this
37314          * @param {Roo.EventObject} e
37315          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37316          */
37317         mouseover : true,
37318         /**
37319          * @event mouseout
37320          * Fires when the mouse exits this menu
37321          * @param {Roo.menu.Menu} this
37322          * @param {Roo.EventObject} e
37323          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37324          */
37325         mouseout : true,
37326         /**
37327          * @event itemclick
37328          * Fires when a menu item contained in this menu is clicked
37329          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37330          * @param {Roo.EventObject} e
37331          */
37332         itemclick: true
37333     });
37334     if (this.registerMenu) {
37335         Roo.menu.MenuMgr.register(this);
37336     }
37337     
37338     var mis = this.items;
37339     this.items = new Roo.util.MixedCollection();
37340     if(mis){
37341         this.add.apply(this, mis);
37342     }
37343 };
37344
37345 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37346     /**
37347      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37348      */
37349     minWidth : 120,
37350     /**
37351      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37352      * for bottom-right shadow (defaults to "sides")
37353      */
37354     shadow : "sides",
37355     /**
37356      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37357      * this menu (defaults to "tl-tr?")
37358      */
37359     subMenuAlign : "tl-tr?",
37360     /**
37361      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37362      * relative to its element of origin (defaults to "tl-bl?")
37363      */
37364     defaultAlign : "tl-bl?",
37365     /**
37366      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37367      */
37368     allowOtherMenus : false,
37369     /**
37370      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37371      */
37372     registerMenu : true,
37373
37374     hidden:true,
37375
37376     // private
37377     render : function(){
37378         if(this.el){
37379             return;
37380         }
37381         var el = this.el = new Roo.Layer({
37382             cls: "x-menu",
37383             shadow:this.shadow,
37384             constrain: false,
37385             parentEl: this.parentEl || document.body,
37386             zindex:15000
37387         });
37388
37389         this.keyNav = new Roo.menu.MenuNav(this);
37390
37391         if(this.plain){
37392             el.addClass("x-menu-plain");
37393         }
37394         if(this.cls){
37395             el.addClass(this.cls);
37396         }
37397         // generic focus element
37398         this.focusEl = el.createChild({
37399             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37400         });
37401         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37402         //disabling touch- as it's causing issues ..
37403         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37404         ul.on('click'   , this.onClick, this);
37405         
37406         
37407         ul.on("mouseover", this.onMouseOver, this);
37408         ul.on("mouseout", this.onMouseOut, this);
37409         this.items.each(function(item){
37410             if (item.hidden) {
37411                 return;
37412             }
37413             
37414             var li = document.createElement("li");
37415             li.className = "x-menu-list-item";
37416             ul.dom.appendChild(li);
37417             item.render(li, this);
37418         }, this);
37419         this.ul = ul;
37420         this.autoWidth();
37421     },
37422
37423     // private
37424     autoWidth : function(){
37425         var el = this.el, ul = this.ul;
37426         if(!el){
37427             return;
37428         }
37429         var w = this.width;
37430         if(w){
37431             el.setWidth(w);
37432         }else if(Roo.isIE){
37433             el.setWidth(this.minWidth);
37434             var t = el.dom.offsetWidth; // force recalc
37435             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37436         }
37437     },
37438
37439     // private
37440     delayAutoWidth : function(){
37441         if(this.rendered){
37442             if(!this.awTask){
37443                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37444             }
37445             this.awTask.delay(20);
37446         }
37447     },
37448
37449     // private
37450     findTargetItem : function(e){
37451         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37452         if(t && t.menuItemId){
37453             return this.items.get(t.menuItemId);
37454         }
37455     },
37456
37457     // private
37458     onClick : function(e){
37459         Roo.log("menu.onClick");
37460         var t = this.findTargetItem(e);
37461         if(!t){
37462             return;
37463         }
37464         Roo.log(e);
37465         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37466             if(t == this.activeItem && t.shouldDeactivate(e)){
37467                 this.activeItem.deactivate();
37468                 delete this.activeItem;
37469                 return;
37470             }
37471             if(t.canActivate){
37472                 this.setActiveItem(t, true);
37473             }
37474             return;
37475             
37476             
37477         }
37478         
37479         t.onClick(e);
37480         this.fireEvent("click", this, t, e);
37481     },
37482
37483     // private
37484     setActiveItem : function(item, autoExpand){
37485         if(item != this.activeItem){
37486             if(this.activeItem){
37487                 this.activeItem.deactivate();
37488             }
37489             this.activeItem = item;
37490             item.activate(autoExpand);
37491         }else if(autoExpand){
37492             item.expandMenu();
37493         }
37494     },
37495
37496     // private
37497     tryActivate : function(start, step){
37498         var items = this.items;
37499         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37500             var item = items.get(i);
37501             if(!item.disabled && item.canActivate){
37502                 this.setActiveItem(item, false);
37503                 return item;
37504             }
37505         }
37506         return false;
37507     },
37508
37509     // private
37510     onMouseOver : function(e){
37511         var t;
37512         if(t = this.findTargetItem(e)){
37513             if(t.canActivate && !t.disabled){
37514                 this.setActiveItem(t, true);
37515             }
37516         }
37517         this.fireEvent("mouseover", this, e, t);
37518     },
37519
37520     // private
37521     onMouseOut : function(e){
37522         var t;
37523         if(t = this.findTargetItem(e)){
37524             if(t == this.activeItem && t.shouldDeactivate(e)){
37525                 this.activeItem.deactivate();
37526                 delete this.activeItem;
37527             }
37528         }
37529         this.fireEvent("mouseout", this, e, t);
37530     },
37531
37532     /**
37533      * Read-only.  Returns true if the menu is currently displayed, else false.
37534      * @type Boolean
37535      */
37536     isVisible : function(){
37537         return this.el && !this.hidden;
37538     },
37539
37540     /**
37541      * Displays this menu relative to another element
37542      * @param {String/HTMLElement/Roo.Element} element The element to align to
37543      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37544      * the element (defaults to this.defaultAlign)
37545      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37546      */
37547     show : function(el, pos, parentMenu){
37548         this.parentMenu = parentMenu;
37549         if(!this.el){
37550             this.render();
37551         }
37552         this.fireEvent("beforeshow", this);
37553         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37554     },
37555
37556     /**
37557      * Displays this menu at a specific xy position
37558      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37559      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37560      */
37561     showAt : function(xy, parentMenu, /* private: */_e){
37562         this.parentMenu = parentMenu;
37563         if(!this.el){
37564             this.render();
37565         }
37566         if(_e !== false){
37567             this.fireEvent("beforeshow", this);
37568             xy = this.el.adjustForConstraints(xy);
37569         }
37570         this.el.setXY(xy);
37571         this.el.show();
37572         this.hidden = false;
37573         this.focus();
37574         this.fireEvent("show", this);
37575     },
37576
37577     focus : function(){
37578         if(!this.hidden){
37579             this.doFocus.defer(50, this);
37580         }
37581     },
37582
37583     doFocus : function(){
37584         if(!this.hidden){
37585             this.focusEl.focus();
37586         }
37587     },
37588
37589     /**
37590      * Hides this menu and optionally all parent menus
37591      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37592      */
37593     hide : function(deep){
37594         if(this.el && this.isVisible()){
37595             this.fireEvent("beforehide", this);
37596             if(this.activeItem){
37597                 this.activeItem.deactivate();
37598                 this.activeItem = null;
37599             }
37600             this.el.hide();
37601             this.hidden = true;
37602             this.fireEvent("hide", this);
37603         }
37604         if(deep === true && this.parentMenu){
37605             this.parentMenu.hide(true);
37606         }
37607     },
37608
37609     /**
37610      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37611      * Any of the following are valid:
37612      * <ul>
37613      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37614      * <li>An HTMLElement object which will be converted to a menu item</li>
37615      * <li>A menu item config object that will be created as a new menu item</li>
37616      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37617      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37618      * </ul>
37619      * Usage:
37620      * <pre><code>
37621 // Create the menu
37622 var menu = new Roo.menu.Menu();
37623
37624 // Create a menu item to add by reference
37625 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37626
37627 // Add a bunch of items at once using different methods.
37628 // Only the last item added will be returned.
37629 var item = menu.add(
37630     menuItem,                // add existing item by ref
37631     'Dynamic Item',          // new TextItem
37632     '-',                     // new separator
37633     { text: 'Config Item' }  // new item by config
37634 );
37635 </code></pre>
37636      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37637      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37638      */
37639     add : function(){
37640         var a = arguments, l = a.length, item;
37641         for(var i = 0; i < l; i++){
37642             var el = a[i];
37643             if ((typeof(el) == "object") && el.xtype && el.xns) {
37644                 el = Roo.factory(el, Roo.menu);
37645             }
37646             
37647             if(el.render){ // some kind of Item
37648                 item = this.addItem(el);
37649             }else if(typeof el == "string"){ // string
37650                 if(el == "separator" || el == "-"){
37651                     item = this.addSeparator();
37652                 }else{
37653                     item = this.addText(el);
37654                 }
37655             }else if(el.tagName || el.el){ // element
37656                 item = this.addElement(el);
37657             }else if(typeof el == "object"){ // must be menu item config?
37658                 item = this.addMenuItem(el);
37659             }
37660         }
37661         return item;
37662     },
37663
37664     /**
37665      * Returns this menu's underlying {@link Roo.Element} object
37666      * @return {Roo.Element} The element
37667      */
37668     getEl : function(){
37669         if(!this.el){
37670             this.render();
37671         }
37672         return this.el;
37673     },
37674
37675     /**
37676      * Adds a separator bar to the menu
37677      * @return {Roo.menu.Item} The menu item that was added
37678      */
37679     addSeparator : function(){
37680         return this.addItem(new Roo.menu.Separator());
37681     },
37682
37683     /**
37684      * Adds an {@link Roo.Element} object to the menu
37685      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37686      * @return {Roo.menu.Item} The menu item that was added
37687      */
37688     addElement : function(el){
37689         return this.addItem(new Roo.menu.BaseItem(el));
37690     },
37691
37692     /**
37693      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37694      * @param {Roo.menu.Item} item The menu item to add
37695      * @return {Roo.menu.Item} The menu item that was added
37696      */
37697     addItem : function(item){
37698         this.items.add(item);
37699         if(this.ul){
37700             var li = document.createElement("li");
37701             li.className = "x-menu-list-item";
37702             this.ul.dom.appendChild(li);
37703             item.render(li, this);
37704             this.delayAutoWidth();
37705         }
37706         return item;
37707     },
37708
37709     /**
37710      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37711      * @param {Object} config A MenuItem config object
37712      * @return {Roo.menu.Item} The menu item that was added
37713      */
37714     addMenuItem : function(config){
37715         if(!(config instanceof Roo.menu.Item)){
37716             if(typeof config.checked == "boolean"){ // must be check menu item config?
37717                 config = new Roo.menu.CheckItem(config);
37718             }else{
37719                 config = new Roo.menu.Item(config);
37720             }
37721         }
37722         return this.addItem(config);
37723     },
37724
37725     /**
37726      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37727      * @param {String} text The text to display in the menu item
37728      * @return {Roo.menu.Item} The menu item that was added
37729      */
37730     addText : function(text){
37731         return this.addItem(new Roo.menu.TextItem({ text : text }));
37732     },
37733
37734     /**
37735      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37736      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37737      * @param {Roo.menu.Item} item The menu item to add
37738      * @return {Roo.menu.Item} The menu item that was added
37739      */
37740     insert : function(index, item){
37741         this.items.insert(index, item);
37742         if(this.ul){
37743             var li = document.createElement("li");
37744             li.className = "x-menu-list-item";
37745             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37746             item.render(li, this);
37747             this.delayAutoWidth();
37748         }
37749         return item;
37750     },
37751
37752     /**
37753      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37754      * @param {Roo.menu.Item} item The menu item to remove
37755      */
37756     remove : function(item){
37757         this.items.removeKey(item.id);
37758         item.destroy();
37759     },
37760
37761     /**
37762      * Removes and destroys all items in the menu
37763      */
37764     removeAll : function(){
37765         var f;
37766         while(f = this.items.first()){
37767             this.remove(f);
37768         }
37769     }
37770 });
37771
37772 // MenuNav is a private utility class used internally by the Menu
37773 Roo.menu.MenuNav = function(menu){
37774     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37775     this.scope = this.menu = menu;
37776 };
37777
37778 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37779     doRelay : function(e, h){
37780         var k = e.getKey();
37781         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37782             this.menu.tryActivate(0, 1);
37783             return false;
37784         }
37785         return h.call(this.scope || this, e, this.menu);
37786     },
37787
37788     up : function(e, m){
37789         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37790             m.tryActivate(m.items.length-1, -1);
37791         }
37792     },
37793
37794     down : function(e, m){
37795         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37796             m.tryActivate(0, 1);
37797         }
37798     },
37799
37800     right : function(e, m){
37801         if(m.activeItem){
37802             m.activeItem.expandMenu(true);
37803         }
37804     },
37805
37806     left : function(e, m){
37807         m.hide();
37808         if(m.parentMenu && m.parentMenu.activeItem){
37809             m.parentMenu.activeItem.activate();
37810         }
37811     },
37812
37813     enter : function(e, m){
37814         if(m.activeItem){
37815             e.stopPropagation();
37816             m.activeItem.onClick(e);
37817             m.fireEvent("click", this, m.activeItem);
37818             return true;
37819         }
37820     }
37821 });/*
37822  * Based on:
37823  * Ext JS Library 1.1.1
37824  * Copyright(c) 2006-2007, Ext JS, LLC.
37825  *
37826  * Originally Released Under LGPL - original licence link has changed is not relivant.
37827  *
37828  * Fork - LGPL
37829  * <script type="text/javascript">
37830  */
37831  
37832 /**
37833  * @class Roo.menu.MenuMgr
37834  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37835  * @singleton
37836  */
37837 Roo.menu.MenuMgr = function(){
37838    var menus, active, groups = {}, attached = false, lastShow = new Date();
37839
37840    // private - called when first menu is created
37841    function init(){
37842        menus = {};
37843        active = new Roo.util.MixedCollection();
37844        Roo.get(document).addKeyListener(27, function(){
37845            if(active.length > 0){
37846                hideAll();
37847            }
37848        });
37849    }
37850
37851    // private
37852    function hideAll(){
37853        if(active && active.length > 0){
37854            var c = active.clone();
37855            c.each(function(m){
37856                m.hide();
37857            });
37858        }
37859    }
37860
37861    // private
37862    function onHide(m){
37863        active.remove(m);
37864        if(active.length < 1){
37865            Roo.get(document).un("mousedown", onMouseDown);
37866            attached = false;
37867        }
37868    }
37869
37870    // private
37871    function onShow(m){
37872        var last = active.last();
37873        lastShow = new Date();
37874        active.add(m);
37875        if(!attached){
37876            Roo.get(document).on("mousedown", onMouseDown);
37877            attached = true;
37878        }
37879        if(m.parentMenu){
37880           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37881           m.parentMenu.activeChild = m;
37882        }else if(last && last.isVisible()){
37883           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37884        }
37885    }
37886
37887    // private
37888    function onBeforeHide(m){
37889        if(m.activeChild){
37890            m.activeChild.hide();
37891        }
37892        if(m.autoHideTimer){
37893            clearTimeout(m.autoHideTimer);
37894            delete m.autoHideTimer;
37895        }
37896    }
37897
37898    // private
37899    function onBeforeShow(m){
37900        var pm = m.parentMenu;
37901        if(!pm && !m.allowOtherMenus){
37902            hideAll();
37903        }else if(pm && pm.activeChild && active != m){
37904            pm.activeChild.hide();
37905        }
37906    }
37907
37908    // private
37909    function onMouseDown(e){
37910        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37911            hideAll();
37912        }
37913    }
37914
37915    // private
37916    function onBeforeCheck(mi, state){
37917        if(state){
37918            var g = groups[mi.group];
37919            for(var i = 0, l = g.length; i < l; i++){
37920                if(g[i] != mi){
37921                    g[i].setChecked(false);
37922                }
37923            }
37924        }
37925    }
37926
37927    return {
37928
37929        /**
37930         * Hides all menus that are currently visible
37931         */
37932        hideAll : function(){
37933             hideAll();  
37934        },
37935
37936        // private
37937        register : function(menu){
37938            if(!menus){
37939                init();
37940            }
37941            menus[menu.id] = menu;
37942            menu.on("beforehide", onBeforeHide);
37943            menu.on("hide", onHide);
37944            menu.on("beforeshow", onBeforeShow);
37945            menu.on("show", onShow);
37946            var g = menu.group;
37947            if(g && menu.events["checkchange"]){
37948                if(!groups[g]){
37949                    groups[g] = [];
37950                }
37951                groups[g].push(menu);
37952                menu.on("checkchange", onCheck);
37953            }
37954        },
37955
37956         /**
37957          * Returns a {@link Roo.menu.Menu} object
37958          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
37959          * be used to generate and return a new Menu instance.
37960          */
37961        get : function(menu){
37962            if(typeof menu == "string"){ // menu id
37963                return menus[menu];
37964            }else if(menu.events){  // menu instance
37965                return menu;
37966            }else if(typeof menu.length == 'number'){ // array of menu items?
37967                return new Roo.menu.Menu({items:menu});
37968            }else{ // otherwise, must be a config
37969                return new Roo.menu.Menu(menu);
37970            }
37971        },
37972
37973        // private
37974        unregister : function(menu){
37975            delete menus[menu.id];
37976            menu.un("beforehide", onBeforeHide);
37977            menu.un("hide", onHide);
37978            menu.un("beforeshow", onBeforeShow);
37979            menu.un("show", onShow);
37980            var g = menu.group;
37981            if(g && menu.events["checkchange"]){
37982                groups[g].remove(menu);
37983                menu.un("checkchange", onCheck);
37984            }
37985        },
37986
37987        // private
37988        registerCheckable : function(menuItem){
37989            var g = menuItem.group;
37990            if(g){
37991                if(!groups[g]){
37992                    groups[g] = [];
37993                }
37994                groups[g].push(menuItem);
37995                menuItem.on("beforecheckchange", onBeforeCheck);
37996            }
37997        },
37998
37999        // private
38000        unregisterCheckable : function(menuItem){
38001            var g = menuItem.group;
38002            if(g){
38003                groups[g].remove(menuItem);
38004                menuItem.un("beforecheckchange", onBeforeCheck);
38005            }
38006        }
38007    };
38008 }();/*
38009  * Based on:
38010  * Ext JS Library 1.1.1
38011  * Copyright(c) 2006-2007, Ext JS, LLC.
38012  *
38013  * Originally Released Under LGPL - original licence link has changed is not relivant.
38014  *
38015  * Fork - LGPL
38016  * <script type="text/javascript">
38017  */
38018  
38019
38020 /**
38021  * @class Roo.menu.BaseItem
38022  * @extends Roo.Component
38023  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38024  * management and base configuration options shared by all menu components.
38025  * @constructor
38026  * Creates a new BaseItem
38027  * @param {Object} config Configuration options
38028  */
38029 Roo.menu.BaseItem = function(config){
38030     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38031
38032     this.addEvents({
38033         /**
38034          * @event click
38035          * Fires when this item is clicked
38036          * @param {Roo.menu.BaseItem} this
38037          * @param {Roo.EventObject} e
38038          */
38039         click: true,
38040         /**
38041          * @event activate
38042          * Fires when this item is activated
38043          * @param {Roo.menu.BaseItem} this
38044          */
38045         activate : true,
38046         /**
38047          * @event deactivate
38048          * Fires when this item is deactivated
38049          * @param {Roo.menu.BaseItem} this
38050          */
38051         deactivate : true
38052     });
38053
38054     if(this.handler){
38055         this.on("click", this.handler, this.scope, true);
38056     }
38057 };
38058
38059 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38060     /**
38061      * @cfg {Function} handler
38062      * A function that will handle the click event of this menu item (defaults to undefined)
38063      */
38064     /**
38065      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38066      */
38067     canActivate : false,
38068     
38069      /**
38070      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38071      */
38072     hidden: false,
38073     
38074     /**
38075      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38076      */
38077     activeClass : "x-menu-item-active",
38078     /**
38079      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38080      */
38081     hideOnClick : true,
38082     /**
38083      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38084      */
38085     hideDelay : 100,
38086
38087     // private
38088     ctype: "Roo.menu.BaseItem",
38089
38090     // private
38091     actionMode : "container",
38092
38093     // private
38094     render : function(container, parentMenu){
38095         this.parentMenu = parentMenu;
38096         Roo.menu.BaseItem.superclass.render.call(this, container);
38097         this.container.menuItemId = this.id;
38098     },
38099
38100     // private
38101     onRender : function(container, position){
38102         this.el = Roo.get(this.el);
38103         container.dom.appendChild(this.el.dom);
38104     },
38105
38106     // private
38107     onClick : function(e){
38108         if(!this.disabled && this.fireEvent("click", this, e) !== false
38109                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38110             this.handleClick(e);
38111         }else{
38112             e.stopEvent();
38113         }
38114     },
38115
38116     // private
38117     activate : function(){
38118         if(this.disabled){
38119             return false;
38120         }
38121         var li = this.container;
38122         li.addClass(this.activeClass);
38123         this.region = li.getRegion().adjust(2, 2, -2, -2);
38124         this.fireEvent("activate", this);
38125         return true;
38126     },
38127
38128     // private
38129     deactivate : function(){
38130         this.container.removeClass(this.activeClass);
38131         this.fireEvent("deactivate", this);
38132     },
38133
38134     // private
38135     shouldDeactivate : function(e){
38136         return !this.region || !this.region.contains(e.getPoint());
38137     },
38138
38139     // private
38140     handleClick : function(e){
38141         if(this.hideOnClick){
38142             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38143         }
38144     },
38145
38146     // private
38147     expandMenu : function(autoActivate){
38148         // do nothing
38149     },
38150
38151     // private
38152     hideMenu : function(){
38153         // do nothing
38154     }
38155 });/*
38156  * Based on:
38157  * Ext JS Library 1.1.1
38158  * Copyright(c) 2006-2007, Ext JS, LLC.
38159  *
38160  * Originally Released Under LGPL - original licence link has changed is not relivant.
38161  *
38162  * Fork - LGPL
38163  * <script type="text/javascript">
38164  */
38165  
38166 /**
38167  * @class Roo.menu.Adapter
38168  * @extends Roo.menu.BaseItem
38169  * 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.
38170  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38171  * @constructor
38172  * Creates a new Adapter
38173  * @param {Object} config Configuration options
38174  */
38175 Roo.menu.Adapter = function(component, config){
38176     Roo.menu.Adapter.superclass.constructor.call(this, config);
38177     this.component = component;
38178 };
38179 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38180     // private
38181     canActivate : true,
38182
38183     // private
38184     onRender : function(container, position){
38185         this.component.render(container);
38186         this.el = this.component.getEl();
38187     },
38188
38189     // private
38190     activate : function(){
38191         if(this.disabled){
38192             return false;
38193         }
38194         this.component.focus();
38195         this.fireEvent("activate", this);
38196         return true;
38197     },
38198
38199     // private
38200     deactivate : function(){
38201         this.fireEvent("deactivate", this);
38202     },
38203
38204     // private
38205     disable : function(){
38206         this.component.disable();
38207         Roo.menu.Adapter.superclass.disable.call(this);
38208     },
38209
38210     // private
38211     enable : function(){
38212         this.component.enable();
38213         Roo.menu.Adapter.superclass.enable.call(this);
38214     }
38215 });/*
38216  * Based on:
38217  * Ext JS Library 1.1.1
38218  * Copyright(c) 2006-2007, Ext JS, LLC.
38219  *
38220  * Originally Released Under LGPL - original licence link has changed is not relivant.
38221  *
38222  * Fork - LGPL
38223  * <script type="text/javascript">
38224  */
38225
38226 /**
38227  * @class Roo.menu.TextItem
38228  * @extends Roo.menu.BaseItem
38229  * Adds a static text string to a menu, usually used as either a heading or group separator.
38230  * Note: old style constructor with text is still supported.
38231  * 
38232  * @constructor
38233  * Creates a new TextItem
38234  * @param {Object} cfg Configuration
38235  */
38236 Roo.menu.TextItem = function(cfg){
38237     if (typeof(cfg) == 'string') {
38238         this.text = cfg;
38239     } else {
38240         Roo.apply(this,cfg);
38241     }
38242     
38243     Roo.menu.TextItem.superclass.constructor.call(this);
38244 };
38245
38246 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38247     /**
38248      * @cfg {Boolean} text Text to show on item.
38249      */
38250     text : '',
38251     
38252     /**
38253      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38254      */
38255     hideOnClick : false,
38256     /**
38257      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38258      */
38259     itemCls : "x-menu-text",
38260
38261     // private
38262     onRender : function(){
38263         var s = document.createElement("span");
38264         s.className = this.itemCls;
38265         s.innerHTML = this.text;
38266         this.el = s;
38267         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38268     }
38269 });/*
38270  * Based on:
38271  * Ext JS Library 1.1.1
38272  * Copyright(c) 2006-2007, Ext JS, LLC.
38273  *
38274  * Originally Released Under LGPL - original licence link has changed is not relivant.
38275  *
38276  * Fork - LGPL
38277  * <script type="text/javascript">
38278  */
38279
38280 /**
38281  * @class Roo.menu.Separator
38282  * @extends Roo.menu.BaseItem
38283  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38284  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38285  * @constructor
38286  * @param {Object} config Configuration options
38287  */
38288 Roo.menu.Separator = function(config){
38289     Roo.menu.Separator.superclass.constructor.call(this, config);
38290 };
38291
38292 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38293     /**
38294      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38295      */
38296     itemCls : "x-menu-sep",
38297     /**
38298      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38299      */
38300     hideOnClick : false,
38301
38302     // private
38303     onRender : function(li){
38304         var s = document.createElement("span");
38305         s.className = this.itemCls;
38306         s.innerHTML = "&#160;";
38307         this.el = s;
38308         li.addClass("x-menu-sep-li");
38309         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38310     }
38311 });/*
38312  * Based on:
38313  * Ext JS Library 1.1.1
38314  * Copyright(c) 2006-2007, Ext JS, LLC.
38315  *
38316  * Originally Released Under LGPL - original licence link has changed is not relivant.
38317  *
38318  * Fork - LGPL
38319  * <script type="text/javascript">
38320  */
38321 /**
38322  * @class Roo.menu.Item
38323  * @extends Roo.menu.BaseItem
38324  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38325  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38326  * activation and click handling.
38327  * @constructor
38328  * Creates a new Item
38329  * @param {Object} config Configuration options
38330  */
38331 Roo.menu.Item = function(config){
38332     Roo.menu.Item.superclass.constructor.call(this, config);
38333     if(this.menu){
38334         this.menu = Roo.menu.MenuMgr.get(this.menu);
38335     }
38336 };
38337 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38338     
38339     /**
38340      * @cfg {String} text
38341      * The text to show on the menu item.
38342      */
38343     text: '',
38344      /**
38345      * @cfg {String} HTML to render in menu
38346      * The text to show on the menu item (HTML version).
38347      */
38348     html: '',
38349     /**
38350      * @cfg {String} icon
38351      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38352      */
38353     icon: undefined,
38354     /**
38355      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38356      */
38357     itemCls : "x-menu-item",
38358     /**
38359      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38360      */
38361     canActivate : true,
38362     /**
38363      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38364      */
38365     showDelay: 200,
38366     // doc'd in BaseItem
38367     hideDelay: 200,
38368
38369     // private
38370     ctype: "Roo.menu.Item",
38371     
38372     // private
38373     onRender : function(container, position){
38374         var el = document.createElement("a");
38375         el.hideFocus = true;
38376         el.unselectable = "on";
38377         el.href = this.href || "#";
38378         if(this.hrefTarget){
38379             el.target = this.hrefTarget;
38380         }
38381         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38382         
38383         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38384         
38385         el.innerHTML = String.format(
38386                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38387                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38388         this.el = el;
38389         Roo.menu.Item.superclass.onRender.call(this, container, position);
38390     },
38391
38392     /**
38393      * Sets the text to display in this menu item
38394      * @param {String} text The text to display
38395      * @param {Boolean} isHTML true to indicate text is pure html.
38396      */
38397     setText : function(text, isHTML){
38398         if (isHTML) {
38399             this.html = text;
38400         } else {
38401             this.text = text;
38402             this.html = '';
38403         }
38404         if(this.rendered){
38405             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38406      
38407             this.el.update(String.format(
38408                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38409                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38410             this.parentMenu.autoWidth();
38411         }
38412     },
38413
38414     // private
38415     handleClick : function(e){
38416         if(!this.href){ // if no link defined, stop the event automatically
38417             e.stopEvent();
38418         }
38419         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38420     },
38421
38422     // private
38423     activate : function(autoExpand){
38424         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38425             this.focus();
38426             if(autoExpand){
38427                 this.expandMenu();
38428             }
38429         }
38430         return true;
38431     },
38432
38433     // private
38434     shouldDeactivate : function(e){
38435         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38436             if(this.menu && this.menu.isVisible()){
38437                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38438             }
38439             return true;
38440         }
38441         return false;
38442     },
38443
38444     // private
38445     deactivate : function(){
38446         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38447         this.hideMenu();
38448     },
38449
38450     // private
38451     expandMenu : function(autoActivate){
38452         if(!this.disabled && this.menu){
38453             clearTimeout(this.hideTimer);
38454             delete this.hideTimer;
38455             if(!this.menu.isVisible() && !this.showTimer){
38456                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38457             }else if (this.menu.isVisible() && autoActivate){
38458                 this.menu.tryActivate(0, 1);
38459             }
38460         }
38461     },
38462
38463     // private
38464     deferExpand : function(autoActivate){
38465         delete this.showTimer;
38466         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38467         if(autoActivate){
38468             this.menu.tryActivate(0, 1);
38469         }
38470     },
38471
38472     // private
38473     hideMenu : function(){
38474         clearTimeout(this.showTimer);
38475         delete this.showTimer;
38476         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38477             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38478         }
38479     },
38480
38481     // private
38482     deferHide : function(){
38483         delete this.hideTimer;
38484         this.menu.hide();
38485     }
38486 });/*
38487  * Based on:
38488  * Ext JS Library 1.1.1
38489  * Copyright(c) 2006-2007, Ext JS, LLC.
38490  *
38491  * Originally Released Under LGPL - original licence link has changed is not relivant.
38492  *
38493  * Fork - LGPL
38494  * <script type="text/javascript">
38495  */
38496  
38497 /**
38498  * @class Roo.menu.CheckItem
38499  * @extends Roo.menu.Item
38500  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38501  * @constructor
38502  * Creates a new CheckItem
38503  * @param {Object} config Configuration options
38504  */
38505 Roo.menu.CheckItem = function(config){
38506     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38507     this.addEvents({
38508         /**
38509          * @event beforecheckchange
38510          * Fires before the checked value is set, providing an opportunity to cancel if needed
38511          * @param {Roo.menu.CheckItem} this
38512          * @param {Boolean} checked The new checked value that will be set
38513          */
38514         "beforecheckchange" : true,
38515         /**
38516          * @event checkchange
38517          * Fires after the checked value has been set
38518          * @param {Roo.menu.CheckItem} this
38519          * @param {Boolean} checked The checked value that was set
38520          */
38521         "checkchange" : true
38522     });
38523     if(this.checkHandler){
38524         this.on('checkchange', this.checkHandler, this.scope);
38525     }
38526 };
38527 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38528     /**
38529      * @cfg {String} group
38530      * All check items with the same group name will automatically be grouped into a single-select
38531      * radio button group (defaults to '')
38532      */
38533     /**
38534      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38535      */
38536     itemCls : "x-menu-item x-menu-check-item",
38537     /**
38538      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38539      */
38540     groupClass : "x-menu-group-item",
38541
38542     /**
38543      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38544      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38545      * initialized with checked = true will be rendered as checked.
38546      */
38547     checked: false,
38548
38549     // private
38550     ctype: "Roo.menu.CheckItem",
38551
38552     // private
38553     onRender : function(c){
38554         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38555         if(this.group){
38556             this.el.addClass(this.groupClass);
38557         }
38558         Roo.menu.MenuMgr.registerCheckable(this);
38559         if(this.checked){
38560             this.checked = false;
38561             this.setChecked(true, true);
38562         }
38563     },
38564
38565     // private
38566     destroy : function(){
38567         if(this.rendered){
38568             Roo.menu.MenuMgr.unregisterCheckable(this);
38569         }
38570         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38571     },
38572
38573     /**
38574      * Set the checked state of this item
38575      * @param {Boolean} checked The new checked value
38576      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38577      */
38578     setChecked : function(state, suppressEvent){
38579         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38580             if(this.container){
38581                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38582             }
38583             this.checked = state;
38584             if(suppressEvent !== true){
38585                 this.fireEvent("checkchange", this, state);
38586             }
38587         }
38588     },
38589
38590     // private
38591     handleClick : function(e){
38592        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38593            this.setChecked(!this.checked);
38594        }
38595        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38596     }
38597 });/*
38598  * Based on:
38599  * Ext JS Library 1.1.1
38600  * Copyright(c) 2006-2007, Ext JS, LLC.
38601  *
38602  * Originally Released Under LGPL - original licence link has changed is not relivant.
38603  *
38604  * Fork - LGPL
38605  * <script type="text/javascript">
38606  */
38607  
38608 /**
38609  * @class Roo.menu.DateItem
38610  * @extends Roo.menu.Adapter
38611  * A menu item that wraps the {@link Roo.DatPicker} component.
38612  * @constructor
38613  * Creates a new DateItem
38614  * @param {Object} config Configuration options
38615  */
38616 Roo.menu.DateItem = function(config){
38617     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38618     /** The Roo.DatePicker object @type Roo.DatePicker */
38619     this.picker = this.component;
38620     this.addEvents({select: true});
38621     
38622     this.picker.on("render", function(picker){
38623         picker.getEl().swallowEvent("click");
38624         picker.container.addClass("x-menu-date-item");
38625     });
38626
38627     this.picker.on("select", this.onSelect, this);
38628 };
38629
38630 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38631     // private
38632     onSelect : function(picker, date){
38633         this.fireEvent("select", this, date, picker);
38634         Roo.menu.DateItem.superclass.handleClick.call(this);
38635     }
38636 });/*
38637  * Based on:
38638  * Ext JS Library 1.1.1
38639  * Copyright(c) 2006-2007, Ext JS, LLC.
38640  *
38641  * Originally Released Under LGPL - original licence link has changed is not relivant.
38642  *
38643  * Fork - LGPL
38644  * <script type="text/javascript">
38645  */
38646  
38647 /**
38648  * @class Roo.menu.ColorItem
38649  * @extends Roo.menu.Adapter
38650  * A menu item that wraps the {@link Roo.ColorPalette} component.
38651  * @constructor
38652  * Creates a new ColorItem
38653  * @param {Object} config Configuration options
38654  */
38655 Roo.menu.ColorItem = function(config){
38656     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38657     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38658     this.palette = this.component;
38659     this.relayEvents(this.palette, ["select"]);
38660     if(this.selectHandler){
38661         this.on('select', this.selectHandler, this.scope);
38662     }
38663 };
38664 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38665  * Based on:
38666  * Ext JS Library 1.1.1
38667  * Copyright(c) 2006-2007, Ext JS, LLC.
38668  *
38669  * Originally Released Under LGPL - original licence link has changed is not relivant.
38670  *
38671  * Fork - LGPL
38672  * <script type="text/javascript">
38673  */
38674  
38675
38676 /**
38677  * @class Roo.menu.DateMenu
38678  * @extends Roo.menu.Menu
38679  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38680  * @constructor
38681  * Creates a new DateMenu
38682  * @param {Object} config Configuration options
38683  */
38684 Roo.menu.DateMenu = function(config){
38685     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38686     this.plain = true;
38687     var di = new Roo.menu.DateItem(config);
38688     this.add(di);
38689     /**
38690      * The {@link Roo.DatePicker} instance for this DateMenu
38691      * @type DatePicker
38692      */
38693     this.picker = di.picker;
38694     /**
38695      * @event select
38696      * @param {DatePicker} picker
38697      * @param {Date} date
38698      */
38699     this.relayEvents(di, ["select"]);
38700     this.on('beforeshow', function(){
38701         if(this.picker){
38702             this.picker.hideMonthPicker(false);
38703         }
38704     }, this);
38705 };
38706 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38707     cls:'x-date-menu'
38708 });/*
38709  * Based on:
38710  * Ext JS Library 1.1.1
38711  * Copyright(c) 2006-2007, Ext JS, LLC.
38712  *
38713  * Originally Released Under LGPL - original licence link has changed is not relivant.
38714  *
38715  * Fork - LGPL
38716  * <script type="text/javascript">
38717  */
38718  
38719
38720 /**
38721  * @class Roo.menu.ColorMenu
38722  * @extends Roo.menu.Menu
38723  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38724  * @constructor
38725  * Creates a new ColorMenu
38726  * @param {Object} config Configuration options
38727  */
38728 Roo.menu.ColorMenu = function(config){
38729     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38730     this.plain = true;
38731     var ci = new Roo.menu.ColorItem(config);
38732     this.add(ci);
38733     /**
38734      * The {@link Roo.ColorPalette} instance for this ColorMenu
38735      * @type ColorPalette
38736      */
38737     this.palette = ci.palette;
38738     /**
38739      * @event select
38740      * @param {ColorPalette} palette
38741      * @param {String} color
38742      */
38743     this.relayEvents(ci, ["select"]);
38744 };
38745 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38746  * Based on:
38747  * Ext JS Library 1.1.1
38748  * Copyright(c) 2006-2007, Ext JS, LLC.
38749  *
38750  * Originally Released Under LGPL - original licence link has changed is not relivant.
38751  *
38752  * Fork - LGPL
38753  * <script type="text/javascript">
38754  */
38755  
38756 /**
38757  * @class Roo.form.TextItem
38758  * @extends Roo.BoxComponent
38759  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38760  * @constructor
38761  * Creates a new TextItem
38762  * @param {Object} config Configuration options
38763  */
38764 Roo.form.TextItem = function(config){
38765     Roo.form.TextItem.superclass.constructor.call(this, config);
38766 };
38767
38768 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38769     
38770     /**
38771      * @cfg {String} tag the tag for this item (default div)
38772      */
38773     tag : 'div',
38774     /**
38775      * @cfg {String} html the content for this item
38776      */
38777     html : '',
38778     
38779     getAutoCreate : function()
38780     {
38781         var cfg = {
38782             id: this.id,
38783             tag: this.tag,
38784             html: this.html,
38785             cls: 'x-form-item'
38786         };
38787         
38788         return cfg;
38789         
38790     },
38791     
38792     onRender : function(ct, position)
38793     {
38794         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38795         
38796         if(!this.el){
38797             var cfg = this.getAutoCreate();
38798             if(!cfg.name){
38799                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38800             }
38801             if (!cfg.name.length) {
38802                 delete cfg.name;
38803             }
38804             this.el = ct.createChild(cfg, position);
38805         }
38806     }
38807     
38808 });/*
38809  * Based on:
38810  * Ext JS Library 1.1.1
38811  * Copyright(c) 2006-2007, Ext JS, LLC.
38812  *
38813  * Originally Released Under LGPL - original licence link has changed is not relivant.
38814  *
38815  * Fork - LGPL
38816  * <script type="text/javascript">
38817  */
38818  
38819 /**
38820  * @class Roo.form.Field
38821  * @extends Roo.BoxComponent
38822  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38823  * @constructor
38824  * Creates a new Field
38825  * @param {Object} config Configuration options
38826  */
38827 Roo.form.Field = function(config){
38828     Roo.form.Field.superclass.constructor.call(this, config);
38829 };
38830
38831 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38832     /**
38833      * @cfg {String} fieldLabel Label to use when rendering a form.
38834      */
38835        /**
38836      * @cfg {String} qtip Mouse over tip
38837      */
38838      
38839     /**
38840      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38841      */
38842     invalidClass : "x-form-invalid",
38843     /**
38844      * @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")
38845      */
38846     invalidText : "The value in this field is invalid",
38847     /**
38848      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38849      */
38850     focusClass : "x-form-focus",
38851     /**
38852      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38853       automatic validation (defaults to "keyup").
38854      */
38855     validationEvent : "keyup",
38856     /**
38857      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38858      */
38859     validateOnBlur : true,
38860     /**
38861      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38862      */
38863     validationDelay : 250,
38864     /**
38865      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38866      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38867      */
38868     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38869     /**
38870      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38871      */
38872     fieldClass : "x-form-field",
38873     /**
38874      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38875      *<pre>
38876 Value         Description
38877 -----------   ----------------------------------------------------------------------
38878 qtip          Display a quick tip when the user hovers over the field
38879 title         Display a default browser title attribute popup
38880 under         Add a block div beneath the field containing the error text
38881 side          Add an error icon to the right of the field with a popup on hover
38882 [element id]  Add the error text directly to the innerHTML of the specified element
38883 </pre>
38884      */
38885     msgTarget : 'qtip',
38886     /**
38887      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38888      */
38889     msgFx : 'normal',
38890
38891     /**
38892      * @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.
38893      */
38894     readOnly : false,
38895
38896     /**
38897      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38898      */
38899     disabled : false,
38900
38901     /**
38902      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38903      */
38904     inputType : undefined,
38905     
38906     /**
38907      * @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).
38908          */
38909         tabIndex : undefined,
38910         
38911     // private
38912     isFormField : true,
38913
38914     // private
38915     hasFocus : false,
38916     /**
38917      * @property {Roo.Element} fieldEl
38918      * Element Containing the rendered Field (with label etc.)
38919      */
38920     /**
38921      * @cfg {Mixed} value A value to initialize this field with.
38922      */
38923     value : undefined,
38924
38925     /**
38926      * @cfg {String} name The field's HTML name attribute.
38927      */
38928     /**
38929      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38930      */
38931     // private
38932     loadedValue : false,
38933      
38934      
38935         // private ??
38936         initComponent : function(){
38937         Roo.form.Field.superclass.initComponent.call(this);
38938         this.addEvents({
38939             /**
38940              * @event focus
38941              * Fires when this field receives input focus.
38942              * @param {Roo.form.Field} this
38943              */
38944             focus : true,
38945             /**
38946              * @event blur
38947              * Fires when this field loses input focus.
38948              * @param {Roo.form.Field} this
38949              */
38950             blur : true,
38951             /**
38952              * @event specialkey
38953              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38954              * {@link Roo.EventObject#getKey} to determine which key was pressed.
38955              * @param {Roo.form.Field} this
38956              * @param {Roo.EventObject} e The event object
38957              */
38958             specialkey : true,
38959             /**
38960              * @event change
38961              * Fires just before the field blurs if the field value has changed.
38962              * @param {Roo.form.Field} this
38963              * @param {Mixed} newValue The new value
38964              * @param {Mixed} oldValue The original value
38965              */
38966             change : true,
38967             /**
38968              * @event invalid
38969              * Fires after the field has been marked as invalid.
38970              * @param {Roo.form.Field} this
38971              * @param {String} msg The validation message
38972              */
38973             invalid : true,
38974             /**
38975              * @event valid
38976              * Fires after the field has been validated with no errors.
38977              * @param {Roo.form.Field} this
38978              */
38979             valid : true,
38980              /**
38981              * @event keyup
38982              * Fires after the key up
38983              * @param {Roo.form.Field} this
38984              * @param {Roo.EventObject}  e The event Object
38985              */
38986             keyup : true
38987         });
38988     },
38989
38990     /**
38991      * Returns the name attribute of the field if available
38992      * @return {String} name The field name
38993      */
38994     getName: function(){
38995          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
38996     },
38997
38998     // private
38999     onRender : function(ct, position){
39000         Roo.form.Field.superclass.onRender.call(this, ct, position);
39001         if(!this.el){
39002             var cfg = this.getAutoCreate();
39003             if(!cfg.name){
39004                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39005             }
39006             if (!cfg.name.length) {
39007                 delete cfg.name;
39008             }
39009             if(this.inputType){
39010                 cfg.type = this.inputType;
39011             }
39012             this.el = ct.createChild(cfg, position);
39013         }
39014         var type = this.el.dom.type;
39015         if(type){
39016             if(type == 'password'){
39017                 type = 'text';
39018             }
39019             this.el.addClass('x-form-'+type);
39020         }
39021         if(this.readOnly){
39022             this.el.dom.readOnly = true;
39023         }
39024         if(this.tabIndex !== undefined){
39025             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39026         }
39027
39028         this.el.addClass([this.fieldClass, this.cls]);
39029         this.initValue();
39030     },
39031
39032     /**
39033      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39034      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39035      * @return {Roo.form.Field} this
39036      */
39037     applyTo : function(target){
39038         this.allowDomMove = false;
39039         this.el = Roo.get(target);
39040         this.render(this.el.dom.parentNode);
39041         return this;
39042     },
39043
39044     // private
39045     initValue : function(){
39046         if(this.value !== undefined){
39047             this.setValue(this.value);
39048         }else if(this.el.dom.value.length > 0){
39049             this.setValue(this.el.dom.value);
39050         }
39051     },
39052
39053     /**
39054      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39055      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39056      */
39057     isDirty : function() {
39058         if(this.disabled) {
39059             return false;
39060         }
39061         return String(this.getValue()) !== String(this.originalValue);
39062     },
39063
39064     /**
39065      * stores the current value in loadedValue
39066      */
39067     resetHasChanged : function()
39068     {
39069         this.loadedValue = String(this.getValue());
39070     },
39071     /**
39072      * checks the current value against the 'loaded' value.
39073      * Note - will return false if 'resetHasChanged' has not been called first.
39074      */
39075     hasChanged : function()
39076     {
39077         if(this.disabled || this.readOnly) {
39078             return false;
39079         }
39080         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39081     },
39082     
39083     
39084     
39085     // private
39086     afterRender : function(){
39087         Roo.form.Field.superclass.afterRender.call(this);
39088         this.initEvents();
39089     },
39090
39091     // private
39092     fireKey : function(e){
39093         //Roo.log('field ' + e.getKey());
39094         if(e.isNavKeyPress()){
39095             this.fireEvent("specialkey", this, e);
39096         }
39097     },
39098
39099     /**
39100      * Resets the current field value to the originally loaded value and clears any validation messages
39101      */
39102     reset : function(){
39103         this.setValue(this.resetValue);
39104         this.originalValue = this.getValue();
39105         this.clearInvalid();
39106     },
39107
39108     // private
39109     initEvents : function(){
39110         // safari killled keypress - so keydown is now used..
39111         this.el.on("keydown" , this.fireKey,  this);
39112         this.el.on("focus", this.onFocus,  this);
39113         this.el.on("blur", this.onBlur,  this);
39114         this.el.relayEvent('keyup', this);
39115
39116         // reference to original value for reset
39117         this.originalValue = this.getValue();
39118         this.resetValue =  this.getValue();
39119     },
39120
39121     // private
39122     onFocus : function(){
39123         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39124             this.el.addClass(this.focusClass);
39125         }
39126         if(!this.hasFocus){
39127             this.hasFocus = true;
39128             this.startValue = this.getValue();
39129             this.fireEvent("focus", this);
39130         }
39131     },
39132
39133     beforeBlur : Roo.emptyFn,
39134
39135     // private
39136     onBlur : function(){
39137         this.beforeBlur();
39138         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39139             this.el.removeClass(this.focusClass);
39140         }
39141         this.hasFocus = false;
39142         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39143             this.validate();
39144         }
39145         var v = this.getValue();
39146         if(String(v) !== String(this.startValue)){
39147             this.fireEvent('change', this, v, this.startValue);
39148         }
39149         this.fireEvent("blur", this);
39150     },
39151
39152     /**
39153      * Returns whether or not the field value is currently valid
39154      * @param {Boolean} preventMark True to disable marking the field invalid
39155      * @return {Boolean} True if the value is valid, else false
39156      */
39157     isValid : function(preventMark){
39158         if(this.disabled){
39159             return true;
39160         }
39161         var restore = this.preventMark;
39162         this.preventMark = preventMark === true;
39163         var v = this.validateValue(this.processValue(this.getRawValue()));
39164         this.preventMark = restore;
39165         return v;
39166     },
39167
39168     /**
39169      * Validates the field value
39170      * @return {Boolean} True if the value is valid, else false
39171      */
39172     validate : function(){
39173         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39174             this.clearInvalid();
39175             return true;
39176         }
39177         return false;
39178     },
39179
39180     processValue : function(value){
39181         return value;
39182     },
39183
39184     // private
39185     // Subclasses should provide the validation implementation by overriding this
39186     validateValue : function(value){
39187         return true;
39188     },
39189
39190     /**
39191      * Mark this field as invalid
39192      * @param {String} msg The validation message
39193      */
39194     markInvalid : function(msg){
39195         if(!this.rendered || this.preventMark){ // not rendered
39196             return;
39197         }
39198         
39199         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39200         
39201         obj.el.addClass(this.invalidClass);
39202         msg = msg || this.invalidText;
39203         switch(this.msgTarget){
39204             case 'qtip':
39205                 obj.el.dom.qtip = msg;
39206                 obj.el.dom.qclass = 'x-form-invalid-tip';
39207                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39208                     Roo.QuickTips.enable();
39209                 }
39210                 break;
39211             case 'title':
39212                 this.el.dom.title = msg;
39213                 break;
39214             case 'under':
39215                 if(!this.errorEl){
39216                     var elp = this.el.findParent('.x-form-element', 5, true);
39217                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39218                     this.errorEl.setWidth(elp.getWidth(true)-20);
39219                 }
39220                 this.errorEl.update(msg);
39221                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39222                 break;
39223             case 'side':
39224                 if(!this.errorIcon){
39225                     var elp = this.el.findParent('.x-form-element', 5, true);
39226                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39227                 }
39228                 this.alignErrorIcon();
39229                 this.errorIcon.dom.qtip = msg;
39230                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39231                 this.errorIcon.show();
39232                 this.on('resize', this.alignErrorIcon, this);
39233                 break;
39234             default:
39235                 var t = Roo.getDom(this.msgTarget);
39236                 t.innerHTML = msg;
39237                 t.style.display = this.msgDisplay;
39238                 break;
39239         }
39240         this.fireEvent('invalid', this, msg);
39241     },
39242
39243     // private
39244     alignErrorIcon : function(){
39245         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39246     },
39247
39248     /**
39249      * Clear any invalid styles/messages for this field
39250      */
39251     clearInvalid : function(){
39252         if(!this.rendered || this.preventMark){ // not rendered
39253             return;
39254         }
39255         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39256         
39257         obj.el.removeClass(this.invalidClass);
39258         switch(this.msgTarget){
39259             case 'qtip':
39260                 obj.el.dom.qtip = '';
39261                 break;
39262             case 'title':
39263                 this.el.dom.title = '';
39264                 break;
39265             case 'under':
39266                 if(this.errorEl){
39267                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39268                 }
39269                 break;
39270             case 'side':
39271                 if(this.errorIcon){
39272                     this.errorIcon.dom.qtip = '';
39273                     this.errorIcon.hide();
39274                     this.un('resize', this.alignErrorIcon, this);
39275                 }
39276                 break;
39277             default:
39278                 var t = Roo.getDom(this.msgTarget);
39279                 t.innerHTML = '';
39280                 t.style.display = 'none';
39281                 break;
39282         }
39283         this.fireEvent('valid', this);
39284     },
39285
39286     /**
39287      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39288      * @return {Mixed} value The field value
39289      */
39290     getRawValue : function(){
39291         var v = this.el.getValue();
39292         
39293         return v;
39294     },
39295
39296     /**
39297      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39298      * @return {Mixed} value The field value
39299      */
39300     getValue : function(){
39301         var v = this.el.getValue();
39302          
39303         return v;
39304     },
39305
39306     /**
39307      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39308      * @param {Mixed} value The value to set
39309      */
39310     setRawValue : function(v){
39311         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39312     },
39313
39314     /**
39315      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39316      * @param {Mixed} value The value to set
39317      */
39318     setValue : function(v){
39319         this.value = v;
39320         if(this.rendered){
39321             this.el.dom.value = (v === null || v === undefined ? '' : v);
39322              this.validate();
39323         }
39324     },
39325
39326     adjustSize : function(w, h){
39327         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39328         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39329         return s;
39330     },
39331
39332     adjustWidth : function(tag, w){
39333         tag = tag.toLowerCase();
39334         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39335             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39336                 if(tag == 'input'){
39337                     return w + 2;
39338                 }
39339                 if(tag == 'textarea'){
39340                     return w-2;
39341                 }
39342             }else if(Roo.isOpera){
39343                 if(tag == 'input'){
39344                     return w + 2;
39345                 }
39346                 if(tag == 'textarea'){
39347                     return w-2;
39348                 }
39349             }
39350         }
39351         return w;
39352     }
39353 });
39354
39355
39356 // anything other than normal should be considered experimental
39357 Roo.form.Field.msgFx = {
39358     normal : {
39359         show: function(msgEl, f){
39360             msgEl.setDisplayed('block');
39361         },
39362
39363         hide : function(msgEl, f){
39364             msgEl.setDisplayed(false).update('');
39365         }
39366     },
39367
39368     slide : {
39369         show: function(msgEl, f){
39370             msgEl.slideIn('t', {stopFx:true});
39371         },
39372
39373         hide : function(msgEl, f){
39374             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39375         }
39376     },
39377
39378     slideRight : {
39379         show: function(msgEl, f){
39380             msgEl.fixDisplay();
39381             msgEl.alignTo(f.el, 'tl-tr');
39382             msgEl.slideIn('l', {stopFx:true});
39383         },
39384
39385         hide : function(msgEl, f){
39386             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39387         }
39388     }
39389 };/*
39390  * Based on:
39391  * Ext JS Library 1.1.1
39392  * Copyright(c) 2006-2007, Ext JS, LLC.
39393  *
39394  * Originally Released Under LGPL - original licence link has changed is not relivant.
39395  *
39396  * Fork - LGPL
39397  * <script type="text/javascript">
39398  */
39399  
39400
39401 /**
39402  * @class Roo.form.TextField
39403  * @extends Roo.form.Field
39404  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39405  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39406  * @constructor
39407  * Creates a new TextField
39408  * @param {Object} config Configuration options
39409  */
39410 Roo.form.TextField = function(config){
39411     Roo.form.TextField.superclass.constructor.call(this, config);
39412     this.addEvents({
39413         /**
39414          * @event autosize
39415          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39416          * according to the default logic, but this event provides a hook for the developer to apply additional
39417          * logic at runtime to resize the field if needed.
39418              * @param {Roo.form.Field} this This text field
39419              * @param {Number} width The new field width
39420              */
39421         autosize : true
39422     });
39423 };
39424
39425 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39426     /**
39427      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39428      */
39429     grow : false,
39430     /**
39431      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39432      */
39433     growMin : 30,
39434     /**
39435      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39436      */
39437     growMax : 800,
39438     /**
39439      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39440      */
39441     vtype : null,
39442     /**
39443      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39444      */
39445     maskRe : null,
39446     /**
39447      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39448      */
39449     disableKeyFilter : false,
39450     /**
39451      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39452      */
39453     allowBlank : true,
39454     /**
39455      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39456      */
39457     minLength : 0,
39458     /**
39459      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39460      */
39461     maxLength : Number.MAX_VALUE,
39462     /**
39463      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39464      */
39465     minLengthText : "The minimum length for this field is {0}",
39466     /**
39467      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39468      */
39469     maxLengthText : "The maximum length for this field is {0}",
39470     /**
39471      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39472      */
39473     selectOnFocus : false,
39474     /**
39475      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39476      */    
39477     allowLeadingSpace : false,
39478     /**
39479      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39480      */
39481     blankText : "This field is required",
39482     /**
39483      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39484      * If available, this function will be called only after the basic validators all return true, and will be passed the
39485      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39486      */
39487     validator : null,
39488     /**
39489      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39490      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39491      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39492      */
39493     regex : null,
39494     /**
39495      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39496      */
39497     regexText : "",
39498     /**
39499      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39500      */
39501     emptyText : null,
39502    
39503
39504     // private
39505     initEvents : function()
39506     {
39507         if (this.emptyText) {
39508             this.el.attr('placeholder', this.emptyText);
39509         }
39510         
39511         Roo.form.TextField.superclass.initEvents.call(this);
39512         if(this.validationEvent == 'keyup'){
39513             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39514             this.el.on('keyup', this.filterValidation, this);
39515         }
39516         else if(this.validationEvent !== false){
39517             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39518         }
39519         
39520         if(this.selectOnFocus){
39521             this.on("focus", this.preFocus, this);
39522         }
39523         if (!this.allowLeadingSpace) {
39524             this.on('blur', this.cleanLeadingSpace, this);
39525         }
39526         
39527         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39528             this.el.on("keypress", this.filterKeys, this);
39529         }
39530         if(this.grow){
39531             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39532             this.el.on("click", this.autoSize,  this);
39533         }
39534         if(this.el.is('input[type=password]') && Roo.isSafari){
39535             this.el.on('keydown', this.SafariOnKeyDown, this);
39536         }
39537     },
39538
39539     processValue : function(value){
39540         if(this.stripCharsRe){
39541             var newValue = value.replace(this.stripCharsRe, '');
39542             if(newValue !== value){
39543                 this.setRawValue(newValue);
39544                 return newValue;
39545             }
39546         }
39547         return value;
39548     },
39549
39550     filterValidation : function(e){
39551         if(!e.isNavKeyPress()){
39552             this.validationTask.delay(this.validationDelay);
39553         }
39554     },
39555
39556     // private
39557     onKeyUp : function(e){
39558         if(!e.isNavKeyPress()){
39559             this.autoSize();
39560         }
39561     },
39562     // private - clean the leading white space
39563     cleanLeadingSpace : function(e)
39564     {
39565         if ( this.inputType == 'file') {
39566             return;
39567         }
39568         
39569         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39570     },
39571     /**
39572      * Resets the current field value to the originally-loaded value and clears any validation messages.
39573      *  
39574      */
39575     reset : function(){
39576         Roo.form.TextField.superclass.reset.call(this);
39577        
39578     }, 
39579     // private
39580     preFocus : function(){
39581         
39582         if(this.selectOnFocus){
39583             this.el.dom.select();
39584         }
39585     },
39586
39587     
39588     // private
39589     filterKeys : function(e){
39590         var k = e.getKey();
39591         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39592             return;
39593         }
39594         var c = e.getCharCode(), cc = String.fromCharCode(c);
39595         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39596             return;
39597         }
39598         if(!this.maskRe.test(cc)){
39599             e.stopEvent();
39600         }
39601     },
39602
39603     setValue : function(v){
39604         
39605         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39606         
39607         this.autoSize();
39608     },
39609
39610     /**
39611      * Validates a value according to the field's validation rules and marks the field as invalid
39612      * if the validation fails
39613      * @param {Mixed} value The value to validate
39614      * @return {Boolean} True if the value is valid, else false
39615      */
39616     validateValue : function(value){
39617         if(value.length < 1)  { // if it's blank
39618              if(this.allowBlank){
39619                 this.clearInvalid();
39620                 return true;
39621              }else{
39622                 this.markInvalid(this.blankText);
39623                 return false;
39624              }
39625         }
39626         if(value.length < this.minLength){
39627             this.markInvalid(String.format(this.minLengthText, this.minLength));
39628             return false;
39629         }
39630         if(value.length > this.maxLength){
39631             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39632             return false;
39633         }
39634         if(this.vtype){
39635             var vt = Roo.form.VTypes;
39636             if(!vt[this.vtype](value, this)){
39637                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39638                 return false;
39639             }
39640         }
39641         if(typeof this.validator == "function"){
39642             var msg = this.validator(value);
39643             if(msg !== true){
39644                 this.markInvalid(msg);
39645                 return false;
39646             }
39647         }
39648         if(this.regex && !this.regex.test(value)){
39649             this.markInvalid(this.regexText);
39650             return false;
39651         }
39652         return true;
39653     },
39654
39655     /**
39656      * Selects text in this field
39657      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39658      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39659      */
39660     selectText : function(start, end){
39661         var v = this.getRawValue();
39662         if(v.length > 0){
39663             start = start === undefined ? 0 : start;
39664             end = end === undefined ? v.length : end;
39665             var d = this.el.dom;
39666             if(d.setSelectionRange){
39667                 d.setSelectionRange(start, end);
39668             }else if(d.createTextRange){
39669                 var range = d.createTextRange();
39670                 range.moveStart("character", start);
39671                 range.moveEnd("character", v.length-end);
39672                 range.select();
39673             }
39674         }
39675     },
39676
39677     /**
39678      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39679      * This only takes effect if grow = true, and fires the autosize event.
39680      */
39681     autoSize : function(){
39682         if(!this.grow || !this.rendered){
39683             return;
39684         }
39685         if(!this.metrics){
39686             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39687         }
39688         var el = this.el;
39689         var v = el.dom.value;
39690         var d = document.createElement('div');
39691         d.appendChild(document.createTextNode(v));
39692         v = d.innerHTML;
39693         d = null;
39694         v += "&#160;";
39695         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39696         this.el.setWidth(w);
39697         this.fireEvent("autosize", this, w);
39698     },
39699     
39700     // private
39701     SafariOnKeyDown : function(event)
39702     {
39703         // this is a workaround for a password hang bug on chrome/ webkit.
39704         
39705         var isSelectAll = false;
39706         
39707         if(this.el.dom.selectionEnd > 0){
39708             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39709         }
39710         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39711             event.preventDefault();
39712             this.setValue('');
39713             return;
39714         }
39715         
39716         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39717             
39718             event.preventDefault();
39719             // this is very hacky as keydown always get's upper case.
39720             
39721             var cc = String.fromCharCode(event.getCharCode());
39722             
39723             
39724             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39725             
39726         }
39727         
39728         
39729     }
39730 });/*
39731  * Based on:
39732  * Ext JS Library 1.1.1
39733  * Copyright(c) 2006-2007, Ext JS, LLC.
39734  *
39735  * Originally Released Under LGPL - original licence link has changed is not relivant.
39736  *
39737  * Fork - LGPL
39738  * <script type="text/javascript">
39739  */
39740  
39741 /**
39742  * @class Roo.form.Hidden
39743  * @extends Roo.form.TextField
39744  * Simple Hidden element used on forms 
39745  * 
39746  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39747  * 
39748  * @constructor
39749  * Creates a new Hidden form element.
39750  * @param {Object} config Configuration options
39751  */
39752
39753
39754
39755 // easy hidden field...
39756 Roo.form.Hidden = function(config){
39757     Roo.form.Hidden.superclass.constructor.call(this, config);
39758 };
39759   
39760 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39761     fieldLabel:      '',
39762     inputType:      'hidden',
39763     width:          50,
39764     allowBlank:     true,
39765     labelSeparator: '',
39766     hidden:         true,
39767     itemCls :       'x-form-item-display-none'
39768
39769
39770 });
39771
39772
39773 /*
39774  * Based on:
39775  * Ext JS Library 1.1.1
39776  * Copyright(c) 2006-2007, Ext JS, LLC.
39777  *
39778  * Originally Released Under LGPL - original licence link has changed is not relivant.
39779  *
39780  * Fork - LGPL
39781  * <script type="text/javascript">
39782  */
39783  
39784 /**
39785  * @class Roo.form.TriggerField
39786  * @extends Roo.form.TextField
39787  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39788  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39789  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39790  * for which you can provide a custom implementation.  For example:
39791  * <pre><code>
39792 var trigger = new Roo.form.TriggerField();
39793 trigger.onTriggerClick = myTriggerFn;
39794 trigger.applyTo('my-field');
39795 </code></pre>
39796  *
39797  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39798  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39799  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39800  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39801  * @constructor
39802  * Create a new TriggerField.
39803  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39804  * to the base TextField)
39805  */
39806 Roo.form.TriggerField = function(config){
39807     this.mimicing = false;
39808     Roo.form.TriggerField.superclass.constructor.call(this, config);
39809 };
39810
39811 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39812     /**
39813      * @cfg {String} triggerClass A CSS class to apply to the trigger
39814      */
39815     /**
39816      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39817      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39818      */
39819     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39820     /**
39821      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39822      */
39823     hideTrigger:false,
39824
39825     /** @cfg {Boolean} grow @hide */
39826     /** @cfg {Number} growMin @hide */
39827     /** @cfg {Number} growMax @hide */
39828
39829     /**
39830      * @hide 
39831      * @method
39832      */
39833     autoSize: Roo.emptyFn,
39834     // private
39835     monitorTab : true,
39836     // private
39837     deferHeight : true,
39838
39839     
39840     actionMode : 'wrap',
39841     // private
39842     onResize : function(w, h){
39843         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39844         if(typeof w == 'number'){
39845             var x = w - this.trigger.getWidth();
39846             this.el.setWidth(this.adjustWidth('input', x));
39847             this.trigger.setStyle('left', x+'px');
39848         }
39849     },
39850
39851     // private
39852     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39853
39854     // private
39855     getResizeEl : function(){
39856         return this.wrap;
39857     },
39858
39859     // private
39860     getPositionEl : function(){
39861         return this.wrap;
39862     },
39863
39864     // private
39865     alignErrorIcon : function(){
39866         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39867     },
39868
39869     // private
39870     onRender : function(ct, position){
39871         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39872         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39873         this.trigger = this.wrap.createChild(this.triggerConfig ||
39874                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39875         if(this.hideTrigger){
39876             this.trigger.setDisplayed(false);
39877         }
39878         this.initTrigger();
39879         if(!this.width){
39880             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39881         }
39882     },
39883
39884     // private
39885     initTrigger : function(){
39886         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39887         this.trigger.addClassOnOver('x-form-trigger-over');
39888         this.trigger.addClassOnClick('x-form-trigger-click');
39889     },
39890
39891     // private
39892     onDestroy : function(){
39893         if(this.trigger){
39894             this.trigger.removeAllListeners();
39895             this.trigger.remove();
39896         }
39897         if(this.wrap){
39898             this.wrap.remove();
39899         }
39900         Roo.form.TriggerField.superclass.onDestroy.call(this);
39901     },
39902
39903     // private
39904     onFocus : function(){
39905         Roo.form.TriggerField.superclass.onFocus.call(this);
39906         if(!this.mimicing){
39907             this.wrap.addClass('x-trigger-wrap-focus');
39908             this.mimicing = true;
39909             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39910             if(this.monitorTab){
39911                 this.el.on("keydown", this.checkTab, this);
39912             }
39913         }
39914     },
39915
39916     // private
39917     checkTab : function(e){
39918         if(e.getKey() == e.TAB){
39919             this.triggerBlur();
39920         }
39921     },
39922
39923     // private
39924     onBlur : function(){
39925         // do nothing
39926     },
39927
39928     // private
39929     mimicBlur : function(e, t){
39930         if(!this.wrap.contains(t) && this.validateBlur()){
39931             this.triggerBlur();
39932         }
39933     },
39934
39935     // private
39936     triggerBlur : function(){
39937         this.mimicing = false;
39938         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39939         if(this.monitorTab){
39940             this.el.un("keydown", this.checkTab, this);
39941         }
39942         this.wrap.removeClass('x-trigger-wrap-focus');
39943         Roo.form.TriggerField.superclass.onBlur.call(this);
39944     },
39945
39946     // private
39947     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39948     validateBlur : function(e, t){
39949         return true;
39950     },
39951
39952     // private
39953     onDisable : function(){
39954         Roo.form.TriggerField.superclass.onDisable.call(this);
39955         if(this.wrap){
39956             this.wrap.addClass('x-item-disabled');
39957         }
39958     },
39959
39960     // private
39961     onEnable : function(){
39962         Roo.form.TriggerField.superclass.onEnable.call(this);
39963         if(this.wrap){
39964             this.wrap.removeClass('x-item-disabled');
39965         }
39966     },
39967
39968     // private
39969     onShow : function(){
39970         var ae = this.getActionEl();
39971         
39972         if(ae){
39973             ae.dom.style.display = '';
39974             ae.dom.style.visibility = 'visible';
39975         }
39976     },
39977
39978     // private
39979     
39980     onHide : function(){
39981         var ae = this.getActionEl();
39982         ae.dom.style.display = 'none';
39983     },
39984
39985     /**
39986      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
39987      * by an implementing function.
39988      * @method
39989      * @param {EventObject} e
39990      */
39991     onTriggerClick : Roo.emptyFn
39992 });
39993
39994 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
39995 // to be extended by an implementing class.  For an example of implementing this class, see the custom
39996 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
39997 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
39998     initComponent : function(){
39999         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40000
40001         this.triggerConfig = {
40002             tag:'span', cls:'x-form-twin-triggers', cn:[
40003             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40004             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40005         ]};
40006     },
40007
40008     getTrigger : function(index){
40009         return this.triggers[index];
40010     },
40011
40012     initTrigger : function(){
40013         var ts = this.trigger.select('.x-form-trigger', true);
40014         this.wrap.setStyle('overflow', 'hidden');
40015         var triggerField = this;
40016         ts.each(function(t, all, index){
40017             t.hide = function(){
40018                 var w = triggerField.wrap.getWidth();
40019                 this.dom.style.display = 'none';
40020                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40021             };
40022             t.show = function(){
40023                 var w = triggerField.wrap.getWidth();
40024                 this.dom.style.display = '';
40025                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40026             };
40027             var triggerIndex = 'Trigger'+(index+1);
40028
40029             if(this['hide'+triggerIndex]){
40030                 t.dom.style.display = 'none';
40031             }
40032             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40033             t.addClassOnOver('x-form-trigger-over');
40034             t.addClassOnClick('x-form-trigger-click');
40035         }, this);
40036         this.triggers = ts.elements;
40037     },
40038
40039     onTrigger1Click : Roo.emptyFn,
40040     onTrigger2Click : Roo.emptyFn
40041 });/*
40042  * Based on:
40043  * Ext JS Library 1.1.1
40044  * Copyright(c) 2006-2007, Ext JS, LLC.
40045  *
40046  * Originally Released Under LGPL - original licence link has changed is not relivant.
40047  *
40048  * Fork - LGPL
40049  * <script type="text/javascript">
40050  */
40051  
40052 /**
40053  * @class Roo.form.TextArea
40054  * @extends Roo.form.TextField
40055  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40056  * support for auto-sizing.
40057  * @constructor
40058  * Creates a new TextArea
40059  * @param {Object} config Configuration options
40060  */
40061 Roo.form.TextArea = function(config){
40062     Roo.form.TextArea.superclass.constructor.call(this, config);
40063     // these are provided exchanges for backwards compat
40064     // minHeight/maxHeight were replaced by growMin/growMax to be
40065     // compatible with TextField growing config values
40066     if(this.minHeight !== undefined){
40067         this.growMin = this.minHeight;
40068     }
40069     if(this.maxHeight !== undefined){
40070         this.growMax = this.maxHeight;
40071     }
40072 };
40073
40074 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40075     /**
40076      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40077      */
40078     growMin : 60,
40079     /**
40080      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40081      */
40082     growMax: 1000,
40083     /**
40084      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40085      * in the field (equivalent to setting overflow: hidden, defaults to false)
40086      */
40087     preventScrollbars: false,
40088     /**
40089      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40090      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40091      */
40092
40093     // private
40094     onRender : function(ct, position){
40095         if(!this.el){
40096             this.defaultAutoCreate = {
40097                 tag: "textarea",
40098                 style:"width:300px;height:60px;",
40099                 autocomplete: "new-password"
40100             };
40101         }
40102         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40103         if(this.grow){
40104             this.textSizeEl = Roo.DomHelper.append(document.body, {
40105                 tag: "pre", cls: "x-form-grow-sizer"
40106             });
40107             if(this.preventScrollbars){
40108                 this.el.setStyle("overflow", "hidden");
40109             }
40110             this.el.setHeight(this.growMin);
40111         }
40112     },
40113
40114     onDestroy : function(){
40115         if(this.textSizeEl){
40116             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40117         }
40118         Roo.form.TextArea.superclass.onDestroy.call(this);
40119     },
40120
40121     // private
40122     onKeyUp : function(e){
40123         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40124             this.autoSize();
40125         }
40126     },
40127
40128     /**
40129      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40130      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40131      */
40132     autoSize : function(){
40133         if(!this.grow || !this.textSizeEl){
40134             return;
40135         }
40136         var el = this.el;
40137         var v = el.dom.value;
40138         var ts = this.textSizeEl;
40139
40140         ts.innerHTML = '';
40141         ts.appendChild(document.createTextNode(v));
40142         v = ts.innerHTML;
40143
40144         Roo.fly(ts).setWidth(this.el.getWidth());
40145         if(v.length < 1){
40146             v = "&#160;&#160;";
40147         }else{
40148             if(Roo.isIE){
40149                 v = v.replace(/\n/g, '<p>&#160;</p>');
40150             }
40151             v += "&#160;\n&#160;";
40152         }
40153         ts.innerHTML = v;
40154         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40155         if(h != this.lastHeight){
40156             this.lastHeight = h;
40157             this.el.setHeight(h);
40158             this.fireEvent("autosize", this, h);
40159         }
40160     }
40161 });/*
40162  * Based on:
40163  * Ext JS Library 1.1.1
40164  * Copyright(c) 2006-2007, Ext JS, LLC.
40165  *
40166  * Originally Released Under LGPL - original licence link has changed is not relivant.
40167  *
40168  * Fork - LGPL
40169  * <script type="text/javascript">
40170  */
40171  
40172
40173 /**
40174  * @class Roo.form.NumberField
40175  * @extends Roo.form.TextField
40176  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40177  * @constructor
40178  * Creates a new NumberField
40179  * @param {Object} config Configuration options
40180  */
40181 Roo.form.NumberField = function(config){
40182     Roo.form.NumberField.superclass.constructor.call(this, config);
40183 };
40184
40185 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40186     /**
40187      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40188      */
40189     fieldClass: "x-form-field x-form-num-field",
40190     /**
40191      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40192      */
40193     allowDecimals : true,
40194     /**
40195      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40196      */
40197     decimalSeparator : ".",
40198     /**
40199      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40200      */
40201     decimalPrecision : 2,
40202     /**
40203      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40204      */
40205     allowNegative : true,
40206     /**
40207      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40208      */
40209     minValue : Number.NEGATIVE_INFINITY,
40210     /**
40211      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40212      */
40213     maxValue : Number.MAX_VALUE,
40214     /**
40215      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40216      */
40217     minText : "The minimum value for this field is {0}",
40218     /**
40219      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40220      */
40221     maxText : "The maximum value for this field is {0}",
40222     /**
40223      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40224      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40225      */
40226     nanText : "{0} is not a valid number",
40227
40228     // private
40229     initEvents : function(){
40230         Roo.form.NumberField.superclass.initEvents.call(this);
40231         var allowed = "0123456789";
40232         if(this.allowDecimals){
40233             allowed += this.decimalSeparator;
40234         }
40235         if(this.allowNegative){
40236             allowed += "-";
40237         }
40238         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40239         var keyPress = function(e){
40240             var k = e.getKey();
40241             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40242                 return;
40243             }
40244             var c = e.getCharCode();
40245             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40246                 e.stopEvent();
40247             }
40248         };
40249         this.el.on("keypress", keyPress, this);
40250     },
40251
40252     // private
40253     validateValue : function(value){
40254         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40255             return false;
40256         }
40257         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40258              return true;
40259         }
40260         var num = this.parseValue(value);
40261         if(isNaN(num)){
40262             this.markInvalid(String.format(this.nanText, value));
40263             return false;
40264         }
40265         if(num < this.minValue){
40266             this.markInvalid(String.format(this.minText, this.minValue));
40267             return false;
40268         }
40269         if(num > this.maxValue){
40270             this.markInvalid(String.format(this.maxText, this.maxValue));
40271             return false;
40272         }
40273         return true;
40274     },
40275
40276     getValue : function(){
40277         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40278     },
40279
40280     // private
40281     parseValue : function(value){
40282         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40283         return isNaN(value) ? '' : value;
40284     },
40285
40286     // private
40287     fixPrecision : function(value){
40288         var nan = isNaN(value);
40289         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40290             return nan ? '' : value;
40291         }
40292         return parseFloat(value).toFixed(this.decimalPrecision);
40293     },
40294
40295     setValue : function(v){
40296         v = this.fixPrecision(v);
40297         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40298     },
40299
40300     // private
40301     decimalPrecisionFcn : function(v){
40302         return Math.floor(v);
40303     },
40304
40305     beforeBlur : function(){
40306         var v = this.parseValue(this.getRawValue());
40307         if(v){
40308             this.setValue(v);
40309         }
40310     }
40311 });/*
40312  * Based on:
40313  * Ext JS Library 1.1.1
40314  * Copyright(c) 2006-2007, Ext JS, LLC.
40315  *
40316  * Originally Released Under LGPL - original licence link has changed is not relivant.
40317  *
40318  * Fork - LGPL
40319  * <script type="text/javascript">
40320  */
40321  
40322 /**
40323  * @class Roo.form.DateField
40324  * @extends Roo.form.TriggerField
40325  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40326 * @constructor
40327 * Create a new DateField
40328 * @param {Object} config
40329  */
40330 Roo.form.DateField = function(config)
40331 {
40332     Roo.form.DateField.superclass.constructor.call(this, config);
40333     
40334       this.addEvents({
40335          
40336         /**
40337          * @event select
40338          * Fires when a date is selected
40339              * @param {Roo.form.DateField} combo This combo box
40340              * @param {Date} date The date selected
40341              */
40342         'select' : true
40343          
40344     });
40345     
40346     
40347     if(typeof this.minValue == "string") {
40348         this.minValue = this.parseDate(this.minValue);
40349     }
40350     if(typeof this.maxValue == "string") {
40351         this.maxValue = this.parseDate(this.maxValue);
40352     }
40353     this.ddMatch = null;
40354     if(this.disabledDates){
40355         var dd = this.disabledDates;
40356         var re = "(?:";
40357         for(var i = 0; i < dd.length; i++){
40358             re += dd[i];
40359             if(i != dd.length-1) {
40360                 re += "|";
40361             }
40362         }
40363         this.ddMatch = new RegExp(re + ")");
40364     }
40365 };
40366
40367 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40368     /**
40369      * @cfg {String} format
40370      * The default date format string which can be overriden for localization support.  The format must be
40371      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40372      */
40373     format : "m/d/y",
40374     /**
40375      * @cfg {String} altFormats
40376      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40377      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40378      */
40379     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40380     /**
40381      * @cfg {Array} disabledDays
40382      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40383      */
40384     disabledDays : null,
40385     /**
40386      * @cfg {String} disabledDaysText
40387      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40388      */
40389     disabledDaysText : "Disabled",
40390     /**
40391      * @cfg {Array} disabledDates
40392      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40393      * expression so they are very powerful. Some examples:
40394      * <ul>
40395      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40396      * <li>["03/08", "09/16"] would disable those days for every year</li>
40397      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40398      * <li>["03/../2006"] would disable every day in March 2006</li>
40399      * <li>["^03"] would disable every day in every March</li>
40400      * </ul>
40401      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40402      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40403      */
40404     disabledDates : null,
40405     /**
40406      * @cfg {String} disabledDatesText
40407      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40408      */
40409     disabledDatesText : "Disabled",
40410     /**
40411      * @cfg {Date/String} minValue
40412      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40413      * valid format (defaults to null).
40414      */
40415     minValue : null,
40416     /**
40417      * @cfg {Date/String} maxValue
40418      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40419      * valid format (defaults to null).
40420      */
40421     maxValue : null,
40422     /**
40423      * @cfg {String} minText
40424      * The error text to display when the date in the cell is before minValue (defaults to
40425      * 'The date in this field must be after {minValue}').
40426      */
40427     minText : "The date in this field must be equal to or after {0}",
40428     /**
40429      * @cfg {String} maxText
40430      * The error text to display when the date in the cell is after maxValue (defaults to
40431      * 'The date in this field must be before {maxValue}').
40432      */
40433     maxText : "The date in this field must be equal to or before {0}",
40434     /**
40435      * @cfg {String} invalidText
40436      * The error text to display when the date in the field is invalid (defaults to
40437      * '{value} is not a valid date - it must be in the format {format}').
40438      */
40439     invalidText : "{0} is not a valid date - it must be in the format {1}",
40440     /**
40441      * @cfg {String} triggerClass
40442      * An additional CSS class used to style the trigger button.  The trigger will always get the
40443      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40444      * which displays a calendar icon).
40445      */
40446     triggerClass : 'x-form-date-trigger',
40447     
40448
40449     /**
40450      * @cfg {Boolean} useIso
40451      * if enabled, then the date field will use a hidden field to store the 
40452      * real value as iso formated date. default (false)
40453      */ 
40454     useIso : false,
40455     /**
40456      * @cfg {String/Object} autoCreate
40457      * A DomHelper element spec, or true for a default element spec (defaults to
40458      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40459      */ 
40460     // private
40461     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40462     
40463     // private
40464     hiddenField: false,
40465     
40466     onRender : function(ct, position)
40467     {
40468         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40469         if (this.useIso) {
40470             //this.el.dom.removeAttribute('name'); 
40471             Roo.log("Changing name?");
40472             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40473             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40474                     'before', true);
40475             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40476             // prevent input submission
40477             this.hiddenName = this.name;
40478         }
40479             
40480             
40481     },
40482     
40483     // private
40484     validateValue : function(value)
40485     {
40486         value = this.formatDate(value);
40487         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40488             Roo.log('super failed');
40489             return false;
40490         }
40491         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40492              return true;
40493         }
40494         var svalue = value;
40495         value = this.parseDate(value);
40496         if(!value){
40497             Roo.log('parse date failed' + svalue);
40498             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40499             return false;
40500         }
40501         var time = value.getTime();
40502         if(this.minValue && time < this.minValue.getTime()){
40503             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40504             return false;
40505         }
40506         if(this.maxValue && time > this.maxValue.getTime()){
40507             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40508             return false;
40509         }
40510         if(this.disabledDays){
40511             var day = value.getDay();
40512             for(var i = 0; i < this.disabledDays.length; i++) {
40513                 if(day === this.disabledDays[i]){
40514                     this.markInvalid(this.disabledDaysText);
40515                     return false;
40516                 }
40517             }
40518         }
40519         var fvalue = this.formatDate(value);
40520         if(this.ddMatch && this.ddMatch.test(fvalue)){
40521             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40522             return false;
40523         }
40524         return true;
40525     },
40526
40527     // private
40528     // Provides logic to override the default TriggerField.validateBlur which just returns true
40529     validateBlur : function(){
40530         return !this.menu || !this.menu.isVisible();
40531     },
40532     
40533     getName: function()
40534     {
40535         // returns hidden if it's set..
40536         if (!this.rendered) {return ''};
40537         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40538         
40539     },
40540
40541     /**
40542      * Returns the current date value of the date field.
40543      * @return {Date} The date value
40544      */
40545     getValue : function(){
40546         
40547         return  this.hiddenField ?
40548                 this.hiddenField.value :
40549                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40550     },
40551
40552     /**
40553      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40554      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40555      * (the default format used is "m/d/y").
40556      * <br />Usage:
40557      * <pre><code>
40558 //All of these calls set the same date value (May 4, 2006)
40559
40560 //Pass a date object:
40561 var dt = new Date('5/4/06');
40562 dateField.setValue(dt);
40563
40564 //Pass a date string (default format):
40565 dateField.setValue('5/4/06');
40566
40567 //Pass a date string (custom format):
40568 dateField.format = 'Y-m-d';
40569 dateField.setValue('2006-5-4');
40570 </code></pre>
40571      * @param {String/Date} date The date or valid date string
40572      */
40573     setValue : function(date){
40574         if (this.hiddenField) {
40575             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40576         }
40577         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40578         // make sure the value field is always stored as a date..
40579         this.value = this.parseDate(date);
40580         
40581         
40582     },
40583
40584     // private
40585     parseDate : function(value){
40586         if(!value || value instanceof Date){
40587             return value;
40588         }
40589         var v = Date.parseDate(value, this.format);
40590          if (!v && this.useIso) {
40591             v = Date.parseDate(value, 'Y-m-d');
40592         }
40593         if(!v && this.altFormats){
40594             if(!this.altFormatsArray){
40595                 this.altFormatsArray = this.altFormats.split("|");
40596             }
40597             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40598                 v = Date.parseDate(value, this.altFormatsArray[i]);
40599             }
40600         }
40601         return v;
40602     },
40603
40604     // private
40605     formatDate : function(date, fmt){
40606         return (!date || !(date instanceof Date)) ?
40607                date : date.dateFormat(fmt || this.format);
40608     },
40609
40610     // private
40611     menuListeners : {
40612         select: function(m, d){
40613             
40614             this.setValue(d);
40615             this.fireEvent('select', this, d);
40616         },
40617         show : function(){ // retain focus styling
40618             this.onFocus();
40619         },
40620         hide : function(){
40621             this.focus.defer(10, this);
40622             var ml = this.menuListeners;
40623             this.menu.un("select", ml.select,  this);
40624             this.menu.un("show", ml.show,  this);
40625             this.menu.un("hide", ml.hide,  this);
40626         }
40627     },
40628
40629     // private
40630     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40631     onTriggerClick : function(){
40632         if(this.disabled){
40633             return;
40634         }
40635         if(this.menu == null){
40636             this.menu = new Roo.menu.DateMenu();
40637         }
40638         Roo.apply(this.menu.picker,  {
40639             showClear: this.allowBlank,
40640             minDate : this.minValue,
40641             maxDate : this.maxValue,
40642             disabledDatesRE : this.ddMatch,
40643             disabledDatesText : this.disabledDatesText,
40644             disabledDays : this.disabledDays,
40645             disabledDaysText : this.disabledDaysText,
40646             format : this.useIso ? 'Y-m-d' : this.format,
40647             minText : String.format(this.minText, this.formatDate(this.minValue)),
40648             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40649         });
40650         this.menu.on(Roo.apply({}, this.menuListeners, {
40651             scope:this
40652         }));
40653         this.menu.picker.setValue(this.getValue() || new Date());
40654         this.menu.show(this.el, "tl-bl?");
40655     },
40656
40657     beforeBlur : function(){
40658         var v = this.parseDate(this.getRawValue());
40659         if(v){
40660             this.setValue(v);
40661         }
40662     },
40663
40664     /*@
40665      * overide
40666      * 
40667      */
40668     isDirty : function() {
40669         if(this.disabled) {
40670             return false;
40671         }
40672         
40673         if(typeof(this.startValue) === 'undefined'){
40674             return false;
40675         }
40676         
40677         return String(this.getValue()) !== String(this.startValue);
40678         
40679     },
40680     // @overide
40681     cleanLeadingSpace : function(e)
40682     {
40683        return;
40684     }
40685     
40686 });/*
40687  * Based on:
40688  * Ext JS Library 1.1.1
40689  * Copyright(c) 2006-2007, Ext JS, LLC.
40690  *
40691  * Originally Released Under LGPL - original licence link has changed is not relivant.
40692  *
40693  * Fork - LGPL
40694  * <script type="text/javascript">
40695  */
40696  
40697 /**
40698  * @class Roo.form.MonthField
40699  * @extends Roo.form.TriggerField
40700  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40701 * @constructor
40702 * Create a new MonthField
40703 * @param {Object} config
40704  */
40705 Roo.form.MonthField = function(config){
40706     
40707     Roo.form.MonthField.superclass.constructor.call(this, config);
40708     
40709       this.addEvents({
40710          
40711         /**
40712          * @event select
40713          * Fires when a date is selected
40714              * @param {Roo.form.MonthFieeld} combo This combo box
40715              * @param {Date} date The date selected
40716              */
40717         'select' : true
40718          
40719     });
40720     
40721     
40722     if(typeof this.minValue == "string") {
40723         this.minValue = this.parseDate(this.minValue);
40724     }
40725     if(typeof this.maxValue == "string") {
40726         this.maxValue = this.parseDate(this.maxValue);
40727     }
40728     this.ddMatch = null;
40729     if(this.disabledDates){
40730         var dd = this.disabledDates;
40731         var re = "(?:";
40732         for(var i = 0; i < dd.length; i++){
40733             re += dd[i];
40734             if(i != dd.length-1) {
40735                 re += "|";
40736             }
40737         }
40738         this.ddMatch = new RegExp(re + ")");
40739     }
40740 };
40741
40742 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40743     /**
40744      * @cfg {String} format
40745      * The default date format string which can be overriden for localization support.  The format must be
40746      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40747      */
40748     format : "M Y",
40749     /**
40750      * @cfg {String} altFormats
40751      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40752      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40753      */
40754     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40755     /**
40756      * @cfg {Array} disabledDays
40757      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40758      */
40759     disabledDays : [0,1,2,3,4,5,6],
40760     /**
40761      * @cfg {String} disabledDaysText
40762      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40763      */
40764     disabledDaysText : "Disabled",
40765     /**
40766      * @cfg {Array} disabledDates
40767      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40768      * expression so they are very powerful. Some examples:
40769      * <ul>
40770      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40771      * <li>["03/08", "09/16"] would disable those days for every year</li>
40772      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40773      * <li>["03/../2006"] would disable every day in March 2006</li>
40774      * <li>["^03"] would disable every day in every March</li>
40775      * </ul>
40776      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40777      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40778      */
40779     disabledDates : null,
40780     /**
40781      * @cfg {String} disabledDatesText
40782      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40783      */
40784     disabledDatesText : "Disabled",
40785     /**
40786      * @cfg {Date/String} minValue
40787      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40788      * valid format (defaults to null).
40789      */
40790     minValue : null,
40791     /**
40792      * @cfg {Date/String} maxValue
40793      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40794      * valid format (defaults to null).
40795      */
40796     maxValue : null,
40797     /**
40798      * @cfg {String} minText
40799      * The error text to display when the date in the cell is before minValue (defaults to
40800      * 'The date in this field must be after {minValue}').
40801      */
40802     minText : "The date in this field must be equal to or after {0}",
40803     /**
40804      * @cfg {String} maxTextf
40805      * The error text to display when the date in the cell is after maxValue (defaults to
40806      * 'The date in this field must be before {maxValue}').
40807      */
40808     maxText : "The date in this field must be equal to or before {0}",
40809     /**
40810      * @cfg {String} invalidText
40811      * The error text to display when the date in the field is invalid (defaults to
40812      * '{value} is not a valid date - it must be in the format {format}').
40813      */
40814     invalidText : "{0} is not a valid date - it must be in the format {1}",
40815     /**
40816      * @cfg {String} triggerClass
40817      * An additional CSS class used to style the trigger button.  The trigger will always get the
40818      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40819      * which displays a calendar icon).
40820      */
40821     triggerClass : 'x-form-date-trigger',
40822     
40823
40824     /**
40825      * @cfg {Boolean} useIso
40826      * if enabled, then the date field will use a hidden field to store the 
40827      * real value as iso formated date. default (true)
40828      */ 
40829     useIso : true,
40830     /**
40831      * @cfg {String/Object} autoCreate
40832      * A DomHelper element spec, or true for a default element spec (defaults to
40833      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40834      */ 
40835     // private
40836     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40837     
40838     // private
40839     hiddenField: false,
40840     
40841     hideMonthPicker : false,
40842     
40843     onRender : function(ct, position)
40844     {
40845         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40846         if (this.useIso) {
40847             this.el.dom.removeAttribute('name'); 
40848             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40849                     'before', true);
40850             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40851             // prevent input submission
40852             this.hiddenName = this.name;
40853         }
40854             
40855             
40856     },
40857     
40858     // private
40859     validateValue : function(value)
40860     {
40861         value = this.formatDate(value);
40862         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40863             return false;
40864         }
40865         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40866              return true;
40867         }
40868         var svalue = value;
40869         value = this.parseDate(value);
40870         if(!value){
40871             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40872             return false;
40873         }
40874         var time = value.getTime();
40875         if(this.minValue && time < this.minValue.getTime()){
40876             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40877             return false;
40878         }
40879         if(this.maxValue && time > this.maxValue.getTime()){
40880             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40881             return false;
40882         }
40883         /*if(this.disabledDays){
40884             var day = value.getDay();
40885             for(var i = 0; i < this.disabledDays.length; i++) {
40886                 if(day === this.disabledDays[i]){
40887                     this.markInvalid(this.disabledDaysText);
40888                     return false;
40889                 }
40890             }
40891         }
40892         */
40893         var fvalue = this.formatDate(value);
40894         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40895             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40896             return false;
40897         }
40898         */
40899         return true;
40900     },
40901
40902     // private
40903     // Provides logic to override the default TriggerField.validateBlur which just returns true
40904     validateBlur : function(){
40905         return !this.menu || !this.menu.isVisible();
40906     },
40907
40908     /**
40909      * Returns the current date value of the date field.
40910      * @return {Date} The date value
40911      */
40912     getValue : function(){
40913         
40914         
40915         
40916         return  this.hiddenField ?
40917                 this.hiddenField.value :
40918                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40919     },
40920
40921     /**
40922      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40923      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40924      * (the default format used is "m/d/y").
40925      * <br />Usage:
40926      * <pre><code>
40927 //All of these calls set the same date value (May 4, 2006)
40928
40929 //Pass a date object:
40930 var dt = new Date('5/4/06');
40931 monthField.setValue(dt);
40932
40933 //Pass a date string (default format):
40934 monthField.setValue('5/4/06');
40935
40936 //Pass a date string (custom format):
40937 monthField.format = 'Y-m-d';
40938 monthField.setValue('2006-5-4');
40939 </code></pre>
40940      * @param {String/Date} date The date or valid date string
40941      */
40942     setValue : function(date){
40943         Roo.log('month setValue' + date);
40944         // can only be first of month..
40945         
40946         var val = this.parseDate(date);
40947         
40948         if (this.hiddenField) {
40949             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40950         }
40951         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40952         this.value = this.parseDate(date);
40953     },
40954
40955     // private
40956     parseDate : function(value){
40957         if(!value || value instanceof Date){
40958             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
40959             return value;
40960         }
40961         var v = Date.parseDate(value, this.format);
40962         if (!v && this.useIso) {
40963             v = Date.parseDate(value, 'Y-m-d');
40964         }
40965         if (v) {
40966             // 
40967             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
40968         }
40969         
40970         
40971         if(!v && this.altFormats){
40972             if(!this.altFormatsArray){
40973                 this.altFormatsArray = this.altFormats.split("|");
40974             }
40975             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40976                 v = Date.parseDate(value, this.altFormatsArray[i]);
40977             }
40978         }
40979         return v;
40980     },
40981
40982     // private
40983     formatDate : function(date, fmt){
40984         return (!date || !(date instanceof Date)) ?
40985                date : date.dateFormat(fmt || this.format);
40986     },
40987
40988     // private
40989     menuListeners : {
40990         select: function(m, d){
40991             this.setValue(d);
40992             this.fireEvent('select', this, d);
40993         },
40994         show : function(){ // retain focus styling
40995             this.onFocus();
40996         },
40997         hide : function(){
40998             this.focus.defer(10, this);
40999             var ml = this.menuListeners;
41000             this.menu.un("select", ml.select,  this);
41001             this.menu.un("show", ml.show,  this);
41002             this.menu.un("hide", ml.hide,  this);
41003         }
41004     },
41005     // private
41006     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41007     onTriggerClick : function(){
41008         if(this.disabled){
41009             return;
41010         }
41011         if(this.menu == null){
41012             this.menu = new Roo.menu.DateMenu();
41013            
41014         }
41015         
41016         Roo.apply(this.menu.picker,  {
41017             
41018             showClear: this.allowBlank,
41019             minDate : this.minValue,
41020             maxDate : this.maxValue,
41021             disabledDatesRE : this.ddMatch,
41022             disabledDatesText : this.disabledDatesText,
41023             
41024             format : this.useIso ? 'Y-m-d' : this.format,
41025             minText : String.format(this.minText, this.formatDate(this.minValue)),
41026             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41027             
41028         });
41029          this.menu.on(Roo.apply({}, this.menuListeners, {
41030             scope:this
41031         }));
41032        
41033         
41034         var m = this.menu;
41035         var p = m.picker;
41036         
41037         // hide month picker get's called when we called by 'before hide';
41038         
41039         var ignorehide = true;
41040         p.hideMonthPicker  = function(disableAnim){
41041             if (ignorehide) {
41042                 return;
41043             }
41044              if(this.monthPicker){
41045                 Roo.log("hideMonthPicker called");
41046                 if(disableAnim === true){
41047                     this.monthPicker.hide();
41048                 }else{
41049                     this.monthPicker.slideOut('t', {duration:.2});
41050                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41051                     p.fireEvent("select", this, this.value);
41052                     m.hide();
41053                 }
41054             }
41055         }
41056         
41057         Roo.log('picker set value');
41058         Roo.log(this.getValue());
41059         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41060         m.show(this.el, 'tl-bl?');
41061         ignorehide  = false;
41062         // this will trigger hideMonthPicker..
41063         
41064         
41065         // hidden the day picker
41066         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41067         
41068         
41069         
41070       
41071         
41072         p.showMonthPicker.defer(100, p);
41073     
41074         
41075        
41076     },
41077
41078     beforeBlur : function(){
41079         var v = this.parseDate(this.getRawValue());
41080         if(v){
41081             this.setValue(v);
41082         }
41083     }
41084
41085     /** @cfg {Boolean} grow @hide */
41086     /** @cfg {Number} growMin @hide */
41087     /** @cfg {Number} growMax @hide */
41088     /**
41089      * @hide
41090      * @method autoSize
41091      */
41092 });/*
41093  * Based on:
41094  * Ext JS Library 1.1.1
41095  * Copyright(c) 2006-2007, Ext JS, LLC.
41096  *
41097  * Originally Released Under LGPL - original licence link has changed is not relivant.
41098  *
41099  * Fork - LGPL
41100  * <script type="text/javascript">
41101  */
41102  
41103
41104 /**
41105  * @class Roo.form.ComboBox
41106  * @extends Roo.form.TriggerField
41107  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41108  * @constructor
41109  * Create a new ComboBox.
41110  * @param {Object} config Configuration options
41111  */
41112 Roo.form.ComboBox = function(config){
41113     Roo.form.ComboBox.superclass.constructor.call(this, config);
41114     this.addEvents({
41115         /**
41116          * @event expand
41117          * Fires when the dropdown list is expanded
41118              * @param {Roo.form.ComboBox} combo This combo box
41119              */
41120         'expand' : true,
41121         /**
41122          * @event collapse
41123          * Fires when the dropdown list is collapsed
41124              * @param {Roo.form.ComboBox} combo This combo box
41125              */
41126         'collapse' : true,
41127         /**
41128          * @event beforeselect
41129          * Fires before a list item is selected. Return false to cancel the selection.
41130              * @param {Roo.form.ComboBox} combo This combo box
41131              * @param {Roo.data.Record} record The data record returned from the underlying store
41132              * @param {Number} index The index of the selected item in the dropdown list
41133              */
41134         'beforeselect' : true,
41135         /**
41136          * @event select
41137          * Fires when a list item is selected
41138              * @param {Roo.form.ComboBox} combo This combo box
41139              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41140              * @param {Number} index The index of the selected item in the dropdown list
41141              */
41142         'select' : true,
41143         /**
41144          * @event beforequery
41145          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41146          * The event object passed has these properties:
41147              * @param {Roo.form.ComboBox} combo This combo box
41148              * @param {String} query The query
41149              * @param {Boolean} forceAll true to force "all" query
41150              * @param {Boolean} cancel true to cancel the query
41151              * @param {Object} e The query event object
41152              */
41153         'beforequery': true,
41154          /**
41155          * @event add
41156          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41157              * @param {Roo.form.ComboBox} combo This combo box
41158              */
41159         'add' : true,
41160         /**
41161          * @event edit
41162          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41163              * @param {Roo.form.ComboBox} combo This combo box
41164              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41165              */
41166         'edit' : true
41167         
41168         
41169     });
41170     if(this.transform){
41171         this.allowDomMove = false;
41172         var s = Roo.getDom(this.transform);
41173         if(!this.hiddenName){
41174             this.hiddenName = s.name;
41175         }
41176         if(!this.store){
41177             this.mode = 'local';
41178             var d = [], opts = s.options;
41179             for(var i = 0, len = opts.length;i < len; i++){
41180                 var o = opts[i];
41181                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41182                 if(o.selected) {
41183                     this.value = value;
41184                 }
41185                 d.push([value, o.text]);
41186             }
41187             this.store = new Roo.data.SimpleStore({
41188                 'id': 0,
41189                 fields: ['value', 'text'],
41190                 data : d
41191             });
41192             this.valueField = 'value';
41193             this.displayField = 'text';
41194         }
41195         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41196         if(!this.lazyRender){
41197             this.target = true;
41198             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41199             s.parentNode.removeChild(s); // remove it
41200             this.render(this.el.parentNode);
41201         }else{
41202             s.parentNode.removeChild(s); // remove it
41203         }
41204
41205     }
41206     if (this.store) {
41207         this.store = Roo.factory(this.store, Roo.data);
41208     }
41209     
41210     this.selectedIndex = -1;
41211     if(this.mode == 'local'){
41212         if(config.queryDelay === undefined){
41213             this.queryDelay = 10;
41214         }
41215         if(config.minChars === undefined){
41216             this.minChars = 0;
41217         }
41218     }
41219 };
41220
41221 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41222     /**
41223      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41224      */
41225     /**
41226      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41227      * rendering into an Roo.Editor, defaults to false)
41228      */
41229     /**
41230      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41231      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41232      */
41233     /**
41234      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41235      */
41236     /**
41237      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41238      * the dropdown list (defaults to undefined, with no header element)
41239      */
41240
41241      /**
41242      * @cfg {String/Roo.Template} tpl The template to use to render the output
41243      */
41244      
41245     // private
41246     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41247     /**
41248      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41249      */
41250     listWidth: undefined,
41251     /**
41252      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41253      * mode = 'remote' or 'text' if mode = 'local')
41254      */
41255     displayField: undefined,
41256     /**
41257      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41258      * mode = 'remote' or 'value' if mode = 'local'). 
41259      * Note: use of a valueField requires the user make a selection
41260      * in order for a value to be mapped.
41261      */
41262     valueField: undefined,
41263     
41264     
41265     /**
41266      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41267      * field's data value (defaults to the underlying DOM element's name)
41268      */
41269     hiddenName: undefined,
41270     /**
41271      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41272      */
41273     listClass: '',
41274     /**
41275      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41276      */
41277     selectedClass: 'x-combo-selected',
41278     /**
41279      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41280      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41281      * which displays a downward arrow icon).
41282      */
41283     triggerClass : 'x-form-arrow-trigger',
41284     /**
41285      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41286      */
41287     shadow:'sides',
41288     /**
41289      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41290      * anchor positions (defaults to 'tl-bl')
41291      */
41292     listAlign: 'tl-bl?',
41293     /**
41294      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41295      */
41296     maxHeight: 300,
41297     /**
41298      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41299      * query specified by the allQuery config option (defaults to 'query')
41300      */
41301     triggerAction: 'query',
41302     /**
41303      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41304      * (defaults to 4, does not apply if editable = false)
41305      */
41306     minChars : 4,
41307     /**
41308      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41309      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41310      */
41311     typeAhead: false,
41312     /**
41313      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41314      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41315      */
41316     queryDelay: 500,
41317     /**
41318      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41319      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41320      */
41321     pageSize: 0,
41322     /**
41323      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41324      * when editable = true (defaults to false)
41325      */
41326     selectOnFocus:false,
41327     /**
41328      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41329      */
41330     queryParam: 'query',
41331     /**
41332      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41333      * when mode = 'remote' (defaults to 'Loading...')
41334      */
41335     loadingText: 'Loading...',
41336     /**
41337      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41338      */
41339     resizable: false,
41340     /**
41341      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41342      */
41343     handleHeight : 8,
41344     /**
41345      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41346      * traditional select (defaults to true)
41347      */
41348     editable: true,
41349     /**
41350      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41351      */
41352     allQuery: '',
41353     /**
41354      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41355      */
41356     mode: 'remote',
41357     /**
41358      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41359      * listWidth has a higher value)
41360      */
41361     minListWidth : 70,
41362     /**
41363      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41364      * allow the user to set arbitrary text into the field (defaults to false)
41365      */
41366     forceSelection:false,
41367     /**
41368      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41369      * if typeAhead = true (defaults to 250)
41370      */
41371     typeAheadDelay : 250,
41372     /**
41373      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41374      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41375      */
41376     valueNotFoundText : undefined,
41377     /**
41378      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41379      */
41380     blockFocus : false,
41381     
41382     /**
41383      * @cfg {Boolean} disableClear Disable showing of clear button.
41384      */
41385     disableClear : false,
41386     /**
41387      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41388      */
41389     alwaysQuery : false,
41390     
41391     //private
41392     addicon : false,
41393     editicon: false,
41394     
41395     // element that contains real text value.. (when hidden is used..)
41396      
41397     // private
41398     onRender : function(ct, position)
41399     {
41400         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41401         
41402         if(this.hiddenName){
41403             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41404                     'before', true);
41405             this.hiddenField.value =
41406                 this.hiddenValue !== undefined ? this.hiddenValue :
41407                 this.value !== undefined ? this.value : '';
41408
41409             // prevent input submission
41410             this.el.dom.removeAttribute('name');
41411              
41412              
41413         }
41414         
41415         if(Roo.isGecko){
41416             this.el.dom.setAttribute('autocomplete', 'off');
41417         }
41418
41419         var cls = 'x-combo-list';
41420
41421         this.list = new Roo.Layer({
41422             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41423         });
41424
41425         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41426         this.list.setWidth(lw);
41427         this.list.swallowEvent('mousewheel');
41428         this.assetHeight = 0;
41429
41430         if(this.title){
41431             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41432             this.assetHeight += this.header.getHeight();
41433         }
41434
41435         this.innerList = this.list.createChild({cls:cls+'-inner'});
41436         this.innerList.on('mouseover', this.onViewOver, this);
41437         this.innerList.on('mousemove', this.onViewMove, this);
41438         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41439         
41440         if(this.allowBlank && !this.pageSize && !this.disableClear){
41441             this.footer = this.list.createChild({cls:cls+'-ft'});
41442             this.pageTb = new Roo.Toolbar(this.footer);
41443            
41444         }
41445         if(this.pageSize){
41446             this.footer = this.list.createChild({cls:cls+'-ft'});
41447             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41448                     {pageSize: this.pageSize});
41449             
41450         }
41451         
41452         if (this.pageTb && this.allowBlank && !this.disableClear) {
41453             var _this = this;
41454             this.pageTb.add(new Roo.Toolbar.Fill(), {
41455                 cls: 'x-btn-icon x-btn-clear',
41456                 text: '&#160;',
41457                 handler: function()
41458                 {
41459                     _this.collapse();
41460                     _this.clearValue();
41461                     _this.onSelect(false, -1);
41462                 }
41463             });
41464         }
41465         if (this.footer) {
41466             this.assetHeight += this.footer.getHeight();
41467         }
41468         
41469
41470         if(!this.tpl){
41471             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41472         }
41473
41474         this.view = new Roo.View(this.innerList, this.tpl, {
41475             singleSelect:true,
41476             store: this.store,
41477             selectedClass: this.selectedClass
41478         });
41479
41480         this.view.on('click', this.onViewClick, this);
41481
41482         this.store.on('beforeload', this.onBeforeLoad, this);
41483         this.store.on('load', this.onLoad, this);
41484         this.store.on('loadexception', this.onLoadException, this);
41485
41486         if(this.resizable){
41487             this.resizer = new Roo.Resizable(this.list,  {
41488                pinned:true, handles:'se'
41489             });
41490             this.resizer.on('resize', function(r, w, h){
41491                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41492                 this.listWidth = w;
41493                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41494                 this.restrictHeight();
41495             }, this);
41496             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41497         }
41498         if(!this.editable){
41499             this.editable = true;
41500             this.setEditable(false);
41501         }  
41502         
41503         
41504         if (typeof(this.events.add.listeners) != 'undefined') {
41505             
41506             this.addicon = this.wrap.createChild(
41507                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41508        
41509             this.addicon.on('click', function(e) {
41510                 this.fireEvent('add', this);
41511             }, this);
41512         }
41513         if (typeof(this.events.edit.listeners) != 'undefined') {
41514             
41515             this.editicon = this.wrap.createChild(
41516                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41517             if (this.addicon) {
41518                 this.editicon.setStyle('margin-left', '40px');
41519             }
41520             this.editicon.on('click', function(e) {
41521                 
41522                 // we fire even  if inothing is selected..
41523                 this.fireEvent('edit', this, this.lastData );
41524                 
41525             }, this);
41526         }
41527         
41528         
41529         
41530     },
41531
41532     // private
41533     initEvents : function(){
41534         Roo.form.ComboBox.superclass.initEvents.call(this);
41535
41536         this.keyNav = new Roo.KeyNav(this.el, {
41537             "up" : function(e){
41538                 this.inKeyMode = true;
41539                 this.selectPrev();
41540             },
41541
41542             "down" : function(e){
41543                 if(!this.isExpanded()){
41544                     this.onTriggerClick();
41545                 }else{
41546                     this.inKeyMode = true;
41547                     this.selectNext();
41548                 }
41549             },
41550
41551             "enter" : function(e){
41552                 this.onViewClick();
41553                 //return true;
41554             },
41555
41556             "esc" : function(e){
41557                 this.collapse();
41558             },
41559
41560             "tab" : function(e){
41561                 this.onViewClick(false);
41562                 this.fireEvent("specialkey", this, e);
41563                 return true;
41564             },
41565
41566             scope : this,
41567
41568             doRelay : function(foo, bar, hname){
41569                 if(hname == 'down' || this.scope.isExpanded()){
41570                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41571                 }
41572                 return true;
41573             },
41574
41575             forceKeyDown: true
41576         });
41577         this.queryDelay = Math.max(this.queryDelay || 10,
41578                 this.mode == 'local' ? 10 : 250);
41579         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41580         if(this.typeAhead){
41581             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41582         }
41583         if(this.editable !== false){
41584             this.el.on("keyup", this.onKeyUp, this);
41585         }
41586         if(this.forceSelection){
41587             this.on('blur', this.doForce, this);
41588         }
41589     },
41590
41591     onDestroy : function(){
41592         if(this.view){
41593             this.view.setStore(null);
41594             this.view.el.removeAllListeners();
41595             this.view.el.remove();
41596             this.view.purgeListeners();
41597         }
41598         if(this.list){
41599             this.list.destroy();
41600         }
41601         if(this.store){
41602             this.store.un('beforeload', this.onBeforeLoad, this);
41603             this.store.un('load', this.onLoad, this);
41604             this.store.un('loadexception', this.onLoadException, this);
41605         }
41606         Roo.form.ComboBox.superclass.onDestroy.call(this);
41607     },
41608
41609     // private
41610     fireKey : function(e){
41611         if(e.isNavKeyPress() && !this.list.isVisible()){
41612             this.fireEvent("specialkey", this, e);
41613         }
41614     },
41615
41616     // private
41617     onResize: function(w, h){
41618         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41619         
41620         if(typeof w != 'number'){
41621             // we do not handle it!?!?
41622             return;
41623         }
41624         var tw = this.trigger.getWidth();
41625         tw += this.addicon ? this.addicon.getWidth() : 0;
41626         tw += this.editicon ? this.editicon.getWidth() : 0;
41627         var x = w - tw;
41628         this.el.setWidth( this.adjustWidth('input', x));
41629             
41630         this.trigger.setStyle('left', x+'px');
41631         
41632         if(this.list && this.listWidth === undefined){
41633             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41634             this.list.setWidth(lw);
41635             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41636         }
41637         
41638     
41639         
41640     },
41641
41642     /**
41643      * Allow or prevent the user from directly editing the field text.  If false is passed,
41644      * the user will only be able to select from the items defined in the dropdown list.  This method
41645      * is the runtime equivalent of setting the 'editable' config option at config time.
41646      * @param {Boolean} value True to allow the user to directly edit the field text
41647      */
41648     setEditable : function(value){
41649         if(value == this.editable){
41650             return;
41651         }
41652         this.editable = value;
41653         if(!value){
41654             this.el.dom.setAttribute('readOnly', true);
41655             this.el.on('mousedown', this.onTriggerClick,  this);
41656             this.el.addClass('x-combo-noedit');
41657         }else{
41658             this.el.dom.setAttribute('readOnly', false);
41659             this.el.un('mousedown', this.onTriggerClick,  this);
41660             this.el.removeClass('x-combo-noedit');
41661         }
41662     },
41663
41664     // private
41665     onBeforeLoad : function(){
41666         if(!this.hasFocus){
41667             return;
41668         }
41669         this.innerList.update(this.loadingText ?
41670                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41671         this.restrictHeight();
41672         this.selectedIndex = -1;
41673     },
41674
41675     // private
41676     onLoad : function(){
41677         if(!this.hasFocus){
41678             return;
41679         }
41680         if(this.store.getCount() > 0){
41681             this.expand();
41682             this.restrictHeight();
41683             if(this.lastQuery == this.allQuery){
41684                 if(this.editable){
41685                     this.el.dom.select();
41686                 }
41687                 if(!this.selectByValue(this.value, true)){
41688                     this.select(0, true);
41689                 }
41690             }else{
41691                 this.selectNext();
41692                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41693                     this.taTask.delay(this.typeAheadDelay);
41694                 }
41695             }
41696         }else{
41697             this.onEmptyResults();
41698         }
41699         //this.el.focus();
41700     },
41701     // private
41702     onLoadException : function()
41703     {
41704         this.collapse();
41705         Roo.log(this.store.reader.jsonData);
41706         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41707             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41708         }
41709         
41710         
41711     },
41712     // private
41713     onTypeAhead : function(){
41714         if(this.store.getCount() > 0){
41715             var r = this.store.getAt(0);
41716             var newValue = r.data[this.displayField];
41717             var len = newValue.length;
41718             var selStart = this.getRawValue().length;
41719             if(selStart != len){
41720                 this.setRawValue(newValue);
41721                 this.selectText(selStart, newValue.length);
41722             }
41723         }
41724     },
41725
41726     // private
41727     onSelect : function(record, index){
41728         if(this.fireEvent('beforeselect', this, record, index) !== false){
41729             this.setFromData(index > -1 ? record.data : false);
41730             this.collapse();
41731             this.fireEvent('select', this, record, index);
41732         }
41733     },
41734
41735     /**
41736      * Returns the currently selected field value or empty string if no value is set.
41737      * @return {String} value The selected value
41738      */
41739     getValue : function(){
41740         if(this.valueField){
41741             return typeof this.value != 'undefined' ? this.value : '';
41742         }
41743         return Roo.form.ComboBox.superclass.getValue.call(this);
41744     },
41745
41746     /**
41747      * Clears any text/value currently set in the field
41748      */
41749     clearValue : function(){
41750         if(this.hiddenField){
41751             this.hiddenField.value = '';
41752         }
41753         this.value = '';
41754         this.setRawValue('');
41755         this.lastSelectionText = '';
41756         
41757     },
41758
41759     /**
41760      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41761      * will be displayed in the field.  If the value does not match the data value of an existing item,
41762      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41763      * Otherwise the field will be blank (although the value will still be set).
41764      * @param {String} value The value to match
41765      */
41766     setValue : function(v){
41767         var text = v;
41768         if(this.valueField){
41769             var r = this.findRecord(this.valueField, v);
41770             if(r){
41771                 text = r.data[this.displayField];
41772             }else if(this.valueNotFoundText !== undefined){
41773                 text = this.valueNotFoundText;
41774             }
41775         }
41776         this.lastSelectionText = text;
41777         if(this.hiddenField){
41778             this.hiddenField.value = v;
41779         }
41780         Roo.form.ComboBox.superclass.setValue.call(this, text);
41781         this.value = v;
41782     },
41783     /**
41784      * @property {Object} the last set data for the element
41785      */
41786     
41787     lastData : false,
41788     /**
41789      * Sets the value of the field based on a object which is related to the record format for the store.
41790      * @param {Object} value the value to set as. or false on reset?
41791      */
41792     setFromData : function(o){
41793         var dv = ''; // display value
41794         var vv = ''; // value value..
41795         this.lastData = o;
41796         if (this.displayField) {
41797             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41798         } else {
41799             // this is an error condition!!!
41800             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41801         }
41802         
41803         if(this.valueField){
41804             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41805         }
41806         if(this.hiddenField){
41807             this.hiddenField.value = vv;
41808             
41809             this.lastSelectionText = dv;
41810             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41811             this.value = vv;
41812             return;
41813         }
41814         // no hidden field.. - we store the value in 'value', but still display
41815         // display field!!!!
41816         this.lastSelectionText = dv;
41817         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41818         this.value = vv;
41819         
41820         
41821     },
41822     // private
41823     reset : function(){
41824         // overridden so that last data is reset..
41825         this.setValue(this.resetValue);
41826         this.originalValue = this.getValue();
41827         this.clearInvalid();
41828         this.lastData = false;
41829         if (this.view) {
41830             this.view.clearSelections();
41831         }
41832     },
41833     // private
41834     findRecord : function(prop, value){
41835         var record;
41836         if(this.store.getCount() > 0){
41837             this.store.each(function(r){
41838                 if(r.data[prop] == value){
41839                     record = r;
41840                     return false;
41841                 }
41842                 return true;
41843             });
41844         }
41845         return record;
41846     },
41847     
41848     getName: function()
41849     {
41850         // returns hidden if it's set..
41851         if (!this.rendered) {return ''};
41852         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41853         
41854     },
41855     // private
41856     onViewMove : function(e, t){
41857         this.inKeyMode = false;
41858     },
41859
41860     // private
41861     onViewOver : function(e, t){
41862         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41863             return;
41864         }
41865         var item = this.view.findItemFromChild(t);
41866         if(item){
41867             var index = this.view.indexOf(item);
41868             this.select(index, false);
41869         }
41870     },
41871
41872     // private
41873     onViewClick : function(doFocus)
41874     {
41875         var index = this.view.getSelectedIndexes()[0];
41876         var r = this.store.getAt(index);
41877         if(r){
41878             this.onSelect(r, index);
41879         }
41880         if(doFocus !== false && !this.blockFocus){
41881             this.el.focus();
41882         }
41883     },
41884
41885     // private
41886     restrictHeight : function(){
41887         this.innerList.dom.style.height = '';
41888         var inner = this.innerList.dom;
41889         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41890         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41891         this.list.beginUpdate();
41892         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41893         this.list.alignTo(this.el, this.listAlign);
41894         this.list.endUpdate();
41895     },
41896
41897     // private
41898     onEmptyResults : function(){
41899         this.collapse();
41900     },
41901
41902     /**
41903      * Returns true if the dropdown list is expanded, else false.
41904      */
41905     isExpanded : function(){
41906         return this.list.isVisible();
41907     },
41908
41909     /**
41910      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41911      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41912      * @param {String} value The data value of the item to select
41913      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41914      * selected item if it is not currently in view (defaults to true)
41915      * @return {Boolean} True if the value matched an item in the list, else false
41916      */
41917     selectByValue : function(v, scrollIntoView){
41918         if(v !== undefined && v !== null){
41919             var r = this.findRecord(this.valueField || this.displayField, v);
41920             if(r){
41921                 this.select(this.store.indexOf(r), scrollIntoView);
41922                 return true;
41923             }
41924         }
41925         return false;
41926     },
41927
41928     /**
41929      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41930      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41931      * @param {Number} index The zero-based index of the list item to select
41932      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41933      * selected item if it is not currently in view (defaults to true)
41934      */
41935     select : function(index, scrollIntoView){
41936         this.selectedIndex = index;
41937         this.view.select(index);
41938         if(scrollIntoView !== false){
41939             var el = this.view.getNode(index);
41940             if(el){
41941                 this.innerList.scrollChildIntoView(el, false);
41942             }
41943         }
41944     },
41945
41946     // private
41947     selectNext : function(){
41948         var ct = this.store.getCount();
41949         if(ct > 0){
41950             if(this.selectedIndex == -1){
41951                 this.select(0);
41952             }else if(this.selectedIndex < ct-1){
41953                 this.select(this.selectedIndex+1);
41954             }
41955         }
41956     },
41957
41958     // private
41959     selectPrev : function(){
41960         var ct = this.store.getCount();
41961         if(ct > 0){
41962             if(this.selectedIndex == -1){
41963                 this.select(0);
41964             }else if(this.selectedIndex != 0){
41965                 this.select(this.selectedIndex-1);
41966             }
41967         }
41968     },
41969
41970     // private
41971     onKeyUp : function(e){
41972         if(this.editable !== false && !e.isSpecialKey()){
41973             this.lastKey = e.getKey();
41974             this.dqTask.delay(this.queryDelay);
41975         }
41976     },
41977
41978     // private
41979     validateBlur : function(){
41980         return !this.list || !this.list.isVisible();   
41981     },
41982
41983     // private
41984     initQuery : function(){
41985         this.doQuery(this.getRawValue());
41986     },
41987
41988     // private
41989     doForce : function(){
41990         if(this.el.dom.value.length > 0){
41991             this.el.dom.value =
41992                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
41993              
41994         }
41995     },
41996
41997     /**
41998      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
41999      * query allowing the query action to be canceled if needed.
42000      * @param {String} query The SQL query to execute
42001      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42002      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42003      * saved in the current store (defaults to false)
42004      */
42005     doQuery : function(q, forceAll){
42006         if(q === undefined || q === null){
42007             q = '';
42008         }
42009         var qe = {
42010             query: q,
42011             forceAll: forceAll,
42012             combo: this,
42013             cancel:false
42014         };
42015         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42016             return false;
42017         }
42018         q = qe.query;
42019         forceAll = qe.forceAll;
42020         if(forceAll === true || (q.length >= this.minChars)){
42021             if(this.lastQuery != q || this.alwaysQuery){
42022                 this.lastQuery = q;
42023                 if(this.mode == 'local'){
42024                     this.selectedIndex = -1;
42025                     if(forceAll){
42026                         this.store.clearFilter();
42027                     }else{
42028                         this.store.filter(this.displayField, q);
42029                     }
42030                     this.onLoad();
42031                 }else{
42032                     this.store.baseParams[this.queryParam] = q;
42033                     this.store.load({
42034                         params: this.getParams(q)
42035                     });
42036                     this.expand();
42037                 }
42038             }else{
42039                 this.selectedIndex = -1;
42040                 this.onLoad();   
42041             }
42042         }
42043     },
42044
42045     // private
42046     getParams : function(q){
42047         var p = {};
42048         //p[this.queryParam] = q;
42049         if(this.pageSize){
42050             p.start = 0;
42051             p.limit = this.pageSize;
42052         }
42053         return p;
42054     },
42055
42056     /**
42057      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42058      */
42059     collapse : function(){
42060         if(!this.isExpanded()){
42061             return;
42062         }
42063         this.list.hide();
42064         Roo.get(document).un('mousedown', this.collapseIf, this);
42065         Roo.get(document).un('mousewheel', this.collapseIf, this);
42066         if (!this.editable) {
42067             Roo.get(document).un('keydown', this.listKeyPress, this);
42068         }
42069         this.fireEvent('collapse', this);
42070     },
42071
42072     // private
42073     collapseIf : function(e){
42074         if(!e.within(this.wrap) && !e.within(this.list)){
42075             this.collapse();
42076         }
42077     },
42078
42079     /**
42080      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42081      */
42082     expand : function(){
42083         if(this.isExpanded() || !this.hasFocus){
42084             return;
42085         }
42086         this.list.alignTo(this.el, this.listAlign);
42087         this.list.show();
42088         Roo.get(document).on('mousedown', this.collapseIf, this);
42089         Roo.get(document).on('mousewheel', this.collapseIf, this);
42090         if (!this.editable) {
42091             Roo.get(document).on('keydown', this.listKeyPress, this);
42092         }
42093         
42094         this.fireEvent('expand', this);
42095     },
42096
42097     // private
42098     // Implements the default empty TriggerField.onTriggerClick function
42099     onTriggerClick : function(){
42100         if(this.disabled){
42101             return;
42102         }
42103         if(this.isExpanded()){
42104             this.collapse();
42105             if (!this.blockFocus) {
42106                 this.el.focus();
42107             }
42108             
42109         }else {
42110             this.hasFocus = true;
42111             if(this.triggerAction == 'all') {
42112                 this.doQuery(this.allQuery, true);
42113             } else {
42114                 this.doQuery(this.getRawValue());
42115             }
42116             if (!this.blockFocus) {
42117                 this.el.focus();
42118             }
42119         }
42120     },
42121     listKeyPress : function(e)
42122     {
42123         //Roo.log('listkeypress');
42124         // scroll to first matching element based on key pres..
42125         if (e.isSpecialKey()) {
42126             return false;
42127         }
42128         var k = String.fromCharCode(e.getKey()).toUpperCase();
42129         //Roo.log(k);
42130         var match  = false;
42131         var csel = this.view.getSelectedNodes();
42132         var cselitem = false;
42133         if (csel.length) {
42134             var ix = this.view.indexOf(csel[0]);
42135             cselitem  = this.store.getAt(ix);
42136             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42137                 cselitem = false;
42138             }
42139             
42140         }
42141         
42142         this.store.each(function(v) { 
42143             if (cselitem) {
42144                 // start at existing selection.
42145                 if (cselitem.id == v.id) {
42146                     cselitem = false;
42147                 }
42148                 return;
42149             }
42150                 
42151             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42152                 match = this.store.indexOf(v);
42153                 return false;
42154             }
42155         }, this);
42156         
42157         if (match === false) {
42158             return true; // no more action?
42159         }
42160         // scroll to?
42161         this.view.select(match);
42162         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42163         sn.scrollIntoView(sn.dom.parentNode, false);
42164     } 
42165
42166     /** 
42167     * @cfg {Boolean} grow 
42168     * @hide 
42169     */
42170     /** 
42171     * @cfg {Number} growMin 
42172     * @hide 
42173     */
42174     /** 
42175     * @cfg {Number} growMax 
42176     * @hide 
42177     */
42178     /**
42179      * @hide
42180      * @method autoSize
42181      */
42182 });/*
42183  * Copyright(c) 2010-2012, Roo J Solutions Limited
42184  *
42185  * Licence LGPL
42186  *
42187  */
42188
42189 /**
42190  * @class Roo.form.ComboBoxArray
42191  * @extends Roo.form.TextField
42192  * A facebook style adder... for lists of email / people / countries  etc...
42193  * pick multiple items from a combo box, and shows each one.
42194  *
42195  *  Fred [x]  Brian [x]  [Pick another |v]
42196  *
42197  *
42198  *  For this to work: it needs various extra information
42199  *    - normal combo problay has
42200  *      name, hiddenName
42201  *    + displayField, valueField
42202  *
42203  *    For our purpose...
42204  *
42205  *
42206  *   If we change from 'extends' to wrapping...
42207  *   
42208  *  
42209  *
42210  
42211  
42212  * @constructor
42213  * Create a new ComboBoxArray.
42214  * @param {Object} config Configuration options
42215  */
42216  
42217
42218 Roo.form.ComboBoxArray = function(config)
42219 {
42220     this.addEvents({
42221         /**
42222          * @event beforeremove
42223          * Fires before remove the value from the list
42224              * @param {Roo.form.ComboBoxArray} _self This combo box array
42225              * @param {Roo.form.ComboBoxArray.Item} item removed item
42226              */
42227         'beforeremove' : true,
42228         /**
42229          * @event remove
42230          * Fires when remove the value from the list
42231              * @param {Roo.form.ComboBoxArray} _self This combo box array
42232              * @param {Roo.form.ComboBoxArray.Item} item removed item
42233              */
42234         'remove' : true
42235         
42236         
42237     });
42238     
42239     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42240     
42241     this.items = new Roo.util.MixedCollection(false);
42242     
42243     // construct the child combo...
42244     
42245     
42246     
42247     
42248    
42249     
42250 }
42251
42252  
42253 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42254
42255     /**
42256      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42257      */
42258     
42259     lastData : false,
42260     
42261     // behavies liek a hiddne field
42262     inputType:      'hidden',
42263     /**
42264      * @cfg {Number} width The width of the box that displays the selected element
42265      */ 
42266     width:          300,
42267
42268     
42269     
42270     /**
42271      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42272      */
42273     name : false,
42274     /**
42275      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42276      */
42277     hiddenName : false,
42278     
42279     
42280     // private the array of items that are displayed..
42281     items  : false,
42282     // private - the hidden field el.
42283     hiddenEl : false,
42284     // private - the filed el..
42285     el : false,
42286     
42287     //validateValue : function() { return true; }, // all values are ok!
42288     //onAddClick: function() { },
42289     
42290     onRender : function(ct, position) 
42291     {
42292         
42293         // create the standard hidden element
42294         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42295         
42296         
42297         // give fake names to child combo;
42298         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42299         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42300         
42301         this.combo = Roo.factory(this.combo, Roo.form);
42302         this.combo.onRender(ct, position);
42303         if (typeof(this.combo.width) != 'undefined') {
42304             this.combo.onResize(this.combo.width,0);
42305         }
42306         
42307         this.combo.initEvents();
42308         
42309         // assigned so form know we need to do this..
42310         this.store          = this.combo.store;
42311         this.valueField     = this.combo.valueField;
42312         this.displayField   = this.combo.displayField ;
42313         
42314         
42315         this.combo.wrap.addClass('x-cbarray-grp');
42316         
42317         var cbwrap = this.combo.wrap.createChild(
42318             {tag: 'div', cls: 'x-cbarray-cb'},
42319             this.combo.el.dom
42320         );
42321         
42322              
42323         this.hiddenEl = this.combo.wrap.createChild({
42324             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42325         });
42326         this.el = this.combo.wrap.createChild({
42327             tag: 'input',  type:'hidden' , name: this.name, value : ''
42328         });
42329          //   this.el.dom.removeAttribute("name");
42330         
42331         
42332         this.outerWrap = this.combo.wrap;
42333         this.wrap = cbwrap;
42334         
42335         this.outerWrap.setWidth(this.width);
42336         this.outerWrap.dom.removeChild(this.el.dom);
42337         
42338         this.wrap.dom.appendChild(this.el.dom);
42339         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42340         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42341         
42342         this.combo.trigger.setStyle('position','relative');
42343         this.combo.trigger.setStyle('left', '0px');
42344         this.combo.trigger.setStyle('top', '2px');
42345         
42346         this.combo.el.setStyle('vertical-align', 'text-bottom');
42347         
42348         //this.trigger.setStyle('vertical-align', 'top');
42349         
42350         // this should use the code from combo really... on('add' ....)
42351         if (this.adder) {
42352             
42353         
42354             this.adder = this.outerWrap.createChild(
42355                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42356             var _t = this;
42357             this.adder.on('click', function(e) {
42358                 _t.fireEvent('adderclick', this, e);
42359             }, _t);
42360         }
42361         //var _t = this;
42362         //this.adder.on('click', this.onAddClick, _t);
42363         
42364         
42365         this.combo.on('select', function(cb, rec, ix) {
42366             this.addItem(rec.data);
42367             
42368             cb.setValue('');
42369             cb.el.dom.value = '';
42370             //cb.lastData = rec.data;
42371             // add to list
42372             
42373         }, this);
42374         
42375         
42376     },
42377     
42378     
42379     getName: function()
42380     {
42381         // returns hidden if it's set..
42382         if (!this.rendered) {return ''};
42383         return  this.hiddenName ? this.hiddenName : this.name;
42384         
42385     },
42386     
42387     
42388     onResize: function(w, h){
42389         
42390         return;
42391         // not sure if this is needed..
42392         //this.combo.onResize(w,h);
42393         
42394         if(typeof w != 'number'){
42395             // we do not handle it!?!?
42396             return;
42397         }
42398         var tw = this.combo.trigger.getWidth();
42399         tw += this.addicon ? this.addicon.getWidth() : 0;
42400         tw += this.editicon ? this.editicon.getWidth() : 0;
42401         var x = w - tw;
42402         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42403             
42404         this.combo.trigger.setStyle('left', '0px');
42405         
42406         if(this.list && this.listWidth === undefined){
42407             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42408             this.list.setWidth(lw);
42409             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42410         }
42411         
42412     
42413         
42414     },
42415     
42416     addItem: function(rec)
42417     {
42418         var valueField = this.combo.valueField;
42419         var displayField = this.combo.displayField;
42420         
42421         if (this.items.indexOfKey(rec[valueField]) > -1) {
42422             //console.log("GOT " + rec.data.id);
42423             return;
42424         }
42425         
42426         var x = new Roo.form.ComboBoxArray.Item({
42427             //id : rec[this.idField],
42428             data : rec,
42429             displayField : displayField ,
42430             tipField : displayField ,
42431             cb : this
42432         });
42433         // use the 
42434         this.items.add(rec[valueField],x);
42435         // add it before the element..
42436         this.updateHiddenEl();
42437         x.render(this.outerWrap, this.wrap.dom);
42438         // add the image handler..
42439     },
42440     
42441     updateHiddenEl : function()
42442     {
42443         this.validate();
42444         if (!this.hiddenEl) {
42445             return;
42446         }
42447         var ar = [];
42448         var idField = this.combo.valueField;
42449         
42450         this.items.each(function(f) {
42451             ar.push(f.data[idField]);
42452         });
42453         this.hiddenEl.dom.value = ar.join(',');
42454         this.validate();
42455     },
42456     
42457     reset : function()
42458     {
42459         this.items.clear();
42460         
42461         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42462            el.remove();
42463         });
42464         
42465         this.el.dom.value = '';
42466         if (this.hiddenEl) {
42467             this.hiddenEl.dom.value = '';
42468         }
42469         
42470     },
42471     getValue: function()
42472     {
42473         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42474     },
42475     setValue: function(v) // not a valid action - must use addItems..
42476     {
42477         
42478         this.reset();
42479          
42480         if (this.store.isLocal && (typeof(v) == 'string')) {
42481             // then we can use the store to find the values..
42482             // comma seperated at present.. this needs to allow JSON based encoding..
42483             this.hiddenEl.value  = v;
42484             var v_ar = [];
42485             Roo.each(v.split(','), function(k) {
42486                 Roo.log("CHECK " + this.valueField + ',' + k);
42487                 var li = this.store.query(this.valueField, k);
42488                 if (!li.length) {
42489                     return;
42490                 }
42491                 var add = {};
42492                 add[this.valueField] = k;
42493                 add[this.displayField] = li.item(0).data[this.displayField];
42494                 
42495                 this.addItem(add);
42496             }, this) 
42497              
42498         }
42499         if (typeof(v) == 'object' ) {
42500             // then let's assume it's an array of objects..
42501             Roo.each(v, function(l) {
42502                 this.addItem(l);
42503             }, this);
42504              
42505         }
42506         
42507         
42508     },
42509     setFromData: function(v)
42510     {
42511         // this recieves an object, if setValues is called.
42512         this.reset();
42513         this.el.dom.value = v[this.displayField];
42514         this.hiddenEl.dom.value = v[this.valueField];
42515         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42516             return;
42517         }
42518         var kv = v[this.valueField];
42519         var dv = v[this.displayField];
42520         kv = typeof(kv) != 'string' ? '' : kv;
42521         dv = typeof(dv) != 'string' ? '' : dv;
42522         
42523         
42524         var keys = kv.split(',');
42525         var display = dv.split(',');
42526         for (var i = 0 ; i < keys.length; i++) {
42527             
42528             add = {};
42529             add[this.valueField] = keys[i];
42530             add[this.displayField] = display[i];
42531             this.addItem(add);
42532         }
42533       
42534         
42535     },
42536     
42537     /**
42538      * Validates the combox array value
42539      * @return {Boolean} True if the value is valid, else false
42540      */
42541     validate : function(){
42542         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42543             this.clearInvalid();
42544             return true;
42545         }
42546         return false;
42547     },
42548     
42549     validateValue : function(value){
42550         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42551         
42552     },
42553     
42554     /*@
42555      * overide
42556      * 
42557      */
42558     isDirty : function() {
42559         if(this.disabled) {
42560             return false;
42561         }
42562         
42563         try {
42564             var d = Roo.decode(String(this.originalValue));
42565         } catch (e) {
42566             return String(this.getValue()) !== String(this.originalValue);
42567         }
42568         
42569         var originalValue = [];
42570         
42571         for (var i = 0; i < d.length; i++){
42572             originalValue.push(d[i][this.valueField]);
42573         }
42574         
42575         return String(this.getValue()) !== String(originalValue.join(','));
42576         
42577     }
42578     
42579 });
42580
42581
42582
42583 /**
42584  * @class Roo.form.ComboBoxArray.Item
42585  * @extends Roo.BoxComponent
42586  * A selected item in the list
42587  *  Fred [x]  Brian [x]  [Pick another |v]
42588  * 
42589  * @constructor
42590  * Create a new item.
42591  * @param {Object} config Configuration options
42592  */
42593  
42594 Roo.form.ComboBoxArray.Item = function(config) {
42595     config.id = Roo.id();
42596     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42597 }
42598
42599 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42600     data : {},
42601     cb: false,
42602     displayField : false,
42603     tipField : false,
42604     
42605     
42606     defaultAutoCreate : {
42607         tag: 'div',
42608         cls: 'x-cbarray-item',
42609         cn : [ 
42610             { tag: 'div' },
42611             {
42612                 tag: 'img',
42613                 width:16,
42614                 height : 16,
42615                 src : Roo.BLANK_IMAGE_URL ,
42616                 align: 'center'
42617             }
42618         ]
42619         
42620     },
42621     
42622  
42623     onRender : function(ct, position)
42624     {
42625         Roo.form.Field.superclass.onRender.call(this, ct, position);
42626         
42627         if(!this.el){
42628             var cfg = this.getAutoCreate();
42629             this.el = ct.createChild(cfg, position);
42630         }
42631         
42632         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42633         
42634         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42635             this.cb.renderer(this.data) :
42636             String.format('{0}',this.data[this.displayField]);
42637         
42638             
42639         this.el.child('div').dom.setAttribute('qtip',
42640                         String.format('{0}',this.data[this.tipField])
42641         );
42642         
42643         this.el.child('img').on('click', this.remove, this);
42644         
42645     },
42646    
42647     remove : function()
42648     {
42649         if(this.cb.disabled){
42650             return;
42651         }
42652         
42653         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42654             this.cb.items.remove(this);
42655             this.el.child('img').un('click', this.remove, this);
42656             this.el.remove();
42657             this.cb.updateHiddenEl();
42658
42659             this.cb.fireEvent('remove', this.cb, this);
42660         }
42661         
42662     }
42663 });/*
42664  * RooJS Library 1.1.1
42665  * Copyright(c) 2008-2011  Alan Knowles
42666  *
42667  * License - LGPL
42668  */
42669  
42670
42671 /**
42672  * @class Roo.form.ComboNested
42673  * @extends Roo.form.ComboBox
42674  * A combobox for that allows selection of nested items in a list,
42675  * eg.
42676  *
42677  *  Book
42678  *    -> red
42679  *    -> green
42680  *  Table
42681  *    -> square
42682  *      ->red
42683  *      ->green
42684  *    -> rectangle
42685  *      ->green
42686  *      
42687  * 
42688  * @constructor
42689  * Create a new ComboNested
42690  * @param {Object} config Configuration options
42691  */
42692 Roo.form.ComboNested = function(config){
42693     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42694     // should verify some data...
42695     // like
42696     // hiddenName = required..
42697     // displayField = required
42698     // valudField == required
42699     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42700     var _t = this;
42701     Roo.each(req, function(e) {
42702         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42703             throw "Roo.form.ComboNested : missing value for: " + e;
42704         }
42705     });
42706      
42707     
42708 };
42709
42710 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42711    
42712    
42713     list : null, // the outermost div..
42714     innerLists : null, // the
42715     views : null,
42716     stores : null,
42717     // private
42718     onRender : function(ct, position)
42719     {
42720         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42721         
42722         if(this.hiddenName){
42723             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42724                     'before', true);
42725             this.hiddenField.value =
42726                 this.hiddenValue !== undefined ? this.hiddenValue :
42727                 this.value !== undefined ? this.value : '';
42728
42729             // prevent input submission
42730             this.el.dom.removeAttribute('name');
42731              
42732              
42733         }
42734         
42735         if(Roo.isGecko){
42736             this.el.dom.setAttribute('autocomplete', 'off');
42737         }
42738
42739         var cls = 'x-combo-list';
42740
42741         this.list = new Roo.Layer({
42742             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42743         });
42744
42745         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42746         this.list.setWidth(lw);
42747         this.list.swallowEvent('mousewheel');
42748         this.assetHeight = 0;
42749
42750         if(this.title){
42751             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42752             this.assetHeight += this.header.getHeight();
42753         }
42754         this.innerLists = [];
42755         this.views = [];
42756         this.stores = [];
42757         for (var i =0 ; i < 3; i++) {
42758             this.onRenderList( cls, i);
42759         }
42760         
42761         // always needs footer, as we are going to have an 'OK' button.
42762         this.footer = this.list.createChild({cls:cls+'-ft'});
42763         this.pageTb = new Roo.Toolbar(this.footer);  
42764         var _this = this;
42765         this.pageTb.add(  {
42766             
42767             text: 'Done',
42768             handler: function()
42769             {
42770                 _this.collapse();
42771             }
42772         });
42773         
42774         if ( this.allowBlank && !this.disableClear) {
42775             
42776             this.pageTb.add(new Roo.Toolbar.Fill(), {
42777                 cls: 'x-btn-icon x-btn-clear',
42778                 text: '&#160;',
42779                 handler: function()
42780                 {
42781                     _this.collapse();
42782                     _this.clearValue();
42783                     _this.onSelect(false, -1);
42784                 }
42785             });
42786         }
42787         if (this.footer) {
42788             this.assetHeight += this.footer.getHeight();
42789         }
42790         
42791     },
42792     onRenderList : function (  cls, i)
42793     {
42794         
42795         var lw = Math.floor(
42796                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42797         );
42798         
42799         this.list.setWidth(lw); // default to '1'
42800
42801         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42802         //il.on('mouseover', this.onViewOver, this, { list:  i });
42803         //il.on('mousemove', this.onViewMove, this, { list:  i });
42804         il.setWidth(lw);
42805         il.setStyle({ 'overflow-x' : 'hidden'});
42806
42807         if(!this.tpl){
42808             this.tpl = new Roo.Template({
42809                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42810                 isEmpty: function (value, allValues) {
42811                     return value.length ? 'has-children' : 'no-children'
42812                 }
42813             });
42814         }
42815         
42816         var store  = this.store;
42817         if (i > 0) {
42818             store  = new Roo.data.SimpleStore({
42819                 reader : this.store.reader,
42820                 data : [ ]
42821             });
42822         }
42823         this.stores[i]  = store;
42824                 
42825         
42826         
42827         var view = this.views[i] = new Roo.View(
42828             il,
42829             this.tpl,
42830             {
42831                 singleSelect:true,
42832                 store: store,
42833                 selectedClass: this.selectedClass
42834             }
42835         );
42836         view.getEl().setWidth(lw);
42837         view.getEl().setStyle({
42838             position: i < 1 ? 'relative' : 'absolute',
42839             top: 0,
42840             left: (i * lw ) + 'px',
42841             display : i > 0 ? 'none' : 'block'
42842         });
42843         view.on('selectionchange', this.onSelectChange, this, {list : i });
42844         view.on('dblclick', this.onDoubleClick, this, {list : i });
42845         //view.on('click', this.onViewClick, this, { list : i });
42846
42847         store.on('beforeload', this.onBeforeLoad, this);
42848         store.on('load',  this.onStoreLoad, this, { list  : i});
42849         store.on('loadexception', this.onLoadException, this);
42850
42851         // hide the other vies..
42852         
42853         
42854         
42855     },
42856     onResize : function()  {},
42857     
42858     restrictHeight : function()
42859     {
42860         var mh = 0;
42861         Roo.each(this.innerLists, function(il,i) {
42862             var el = this.views[i].getEl();
42863             el.dom.style.height = '';
42864             var inner = el.dom;
42865             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42866             // only adjust heights on other ones..
42867             if (i < 1) {
42868                 
42869                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42870                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42871                 mh = Math.max(el.getHeight(), mh);
42872             }
42873             
42874             
42875         }, this);
42876         
42877         this.list.beginUpdate();
42878         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42879         this.list.alignTo(this.el, this.listAlign);
42880         this.list.endUpdate();
42881         
42882     },
42883      
42884     
42885     // -- store handlers..
42886     
42887     // private
42888     onLoad : function(a,b,c,d)
42889     {
42890         
42891         if(!this.hasFocus){
42892             return;
42893         }
42894         
42895         if(this.store.getCount() > 0) {
42896             this.expand();
42897             this.restrictHeight();   
42898         } else {
42899             this.onEmptyResults();
42900         }
42901         /*
42902         this.stores[1].loadData([]);
42903         this.stores[2].loadData([]);
42904         this.views
42905         */    
42906     
42907         //this.el.focus();
42908     },
42909     onStoreLoad : function ()
42910     {
42911         Roo.log(arguments);
42912     },
42913     
42914     // private
42915     onLoadException : function()
42916     {
42917         this.collapse();
42918         Roo.log(this.store.reader.jsonData);
42919         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42920             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42921         }
42922         
42923         
42924     } ,
42925      
42926      
42927
42928     onSelectChange : function (view, sels, opts )
42929     {
42930         var ix = view.getSelectedIndexes();
42931         
42932         
42933         if (opts.list > 1) {
42934              
42935             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42936             return;
42937         }
42938         
42939         if (!ix.length) {
42940             this.setFromData({});
42941             this.stores[opts.list+1].loadData( [] );
42942             return;
42943         }
42944         
42945         var rec = view.store.getAt(ix[0]);
42946         this.setFromData(rec.data);
42947         
42948         var lw = Math.floor(
42949                 ((this.listWidth * 3 || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / 3
42950         );
42951         
42952         this.stores[opts.list+1].loadData( typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn);
42953         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
42954         this.views[opts.list+1].getEl().setStyle({ display : rec.data.cn.length ? 'block' : 'none' });
42955         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
42956         this.list.setWidth(lw * (opts.list + (rec.data.cn.length ? 2 : 1))); 
42957     },
42958     onDoubleClick : function()
42959     {
42960         this.collapse(); //??
42961     },
42962     
42963      
42964     
42965     findRecord : function (prop,value)
42966     {
42967         return this.findRecordInStore(this.store, prop,value);
42968     },
42969     
42970      // private
42971     findRecordInStore : function(store, prop, value)
42972     {
42973         var cstore = new Roo.data.SimpleStore({
42974             reader : this.store.reader,
42975             data : [ ]
42976         });
42977         var _this = this;
42978         var record  = false;
42979         if(store.getCount() > 0){
42980            store.each(function(r){
42981                 if(r.data[prop] == value){
42982                     record = r;
42983                     return false;
42984                 }
42985                 if (r.data.cn && r.data.cn.length) {
42986                     cstore.loadData( r.data.cn);
42987                     var cret = _this.findRecordInStore(cstore, prop, value);
42988                     if (cret !== false) {
42989                         record = cret;
42990                         return false;
42991                     }
42992                 }
42993                 
42994                 return true;
42995             });
42996         }
42997         return record;
42998     }
42999     
43000 });/*
43001  * Based on:
43002  * Ext JS Library 1.1.1
43003  * Copyright(c) 2006-2007, Ext JS, LLC.
43004  *
43005  * Originally Released Under LGPL - original licence link has changed is not relivant.
43006  *
43007  * Fork - LGPL
43008  * <script type="text/javascript">
43009  */
43010 /**
43011  * @class Roo.form.Checkbox
43012  * @extends Roo.form.Field
43013  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43014  * @constructor
43015  * Creates a new Checkbox
43016  * @param {Object} config Configuration options
43017  */
43018 Roo.form.Checkbox = function(config){
43019     Roo.form.Checkbox.superclass.constructor.call(this, config);
43020     this.addEvents({
43021         /**
43022          * @event check
43023          * Fires when the checkbox is checked or unchecked.
43024              * @param {Roo.form.Checkbox} this This checkbox
43025              * @param {Boolean} checked The new checked value
43026              */
43027         check : true
43028     });
43029 };
43030
43031 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43032     /**
43033      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43034      */
43035     focusClass : undefined,
43036     /**
43037      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43038      */
43039     fieldClass: "x-form-field",
43040     /**
43041      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43042      */
43043     checked: false,
43044     /**
43045      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43046      * {tag: "input", type: "checkbox", autocomplete: "off"})
43047      */
43048     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43049     /**
43050      * @cfg {String} boxLabel The text that appears beside the checkbox
43051      */
43052     boxLabel : "",
43053     /**
43054      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43055      */  
43056     inputValue : '1',
43057     /**
43058      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43059      */
43060      valueOff: '0', // value when not checked..
43061
43062     actionMode : 'viewEl', 
43063     //
43064     // private
43065     itemCls : 'x-menu-check-item x-form-item',
43066     groupClass : 'x-menu-group-item',
43067     inputType : 'hidden',
43068     
43069     
43070     inSetChecked: false, // check that we are not calling self...
43071     
43072     inputElement: false, // real input element?
43073     basedOn: false, // ????
43074     
43075     isFormField: true, // not sure where this is needed!!!!
43076
43077     onResize : function(){
43078         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43079         if(!this.boxLabel){
43080             this.el.alignTo(this.wrap, 'c-c');
43081         }
43082     },
43083
43084     initEvents : function(){
43085         Roo.form.Checkbox.superclass.initEvents.call(this);
43086         this.el.on("click", this.onClick,  this);
43087         this.el.on("change", this.onClick,  this);
43088     },
43089
43090
43091     getResizeEl : function(){
43092         return this.wrap;
43093     },
43094
43095     getPositionEl : function(){
43096         return this.wrap;
43097     },
43098
43099     // private
43100     onRender : function(ct, position){
43101         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43102         /*
43103         if(this.inputValue !== undefined){
43104             this.el.dom.value = this.inputValue;
43105         }
43106         */
43107         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43108         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43109         var viewEl = this.wrap.createChild({ 
43110             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43111         this.viewEl = viewEl;   
43112         this.wrap.on('click', this.onClick,  this); 
43113         
43114         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43115         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43116         
43117         
43118         
43119         if(this.boxLabel){
43120             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43121         //    viewEl.on('click', this.onClick,  this); 
43122         }
43123         //if(this.checked){
43124             this.setChecked(this.checked);
43125         //}else{
43126             //this.checked = this.el.dom;
43127         //}
43128
43129     },
43130
43131     // private
43132     initValue : Roo.emptyFn,
43133
43134     /**
43135      * Returns the checked state of the checkbox.
43136      * @return {Boolean} True if checked, else false
43137      */
43138     getValue : function(){
43139         if(this.el){
43140             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43141         }
43142         return this.valueOff;
43143         
43144     },
43145
43146         // private
43147     onClick : function(){ 
43148         if (this.disabled) {
43149             return;
43150         }
43151         this.setChecked(!this.checked);
43152
43153         //if(this.el.dom.checked != this.checked){
43154         //    this.setValue(this.el.dom.checked);
43155        // }
43156     },
43157
43158     /**
43159      * Sets the checked state of the checkbox.
43160      * On is always based on a string comparison between inputValue and the param.
43161      * @param {Boolean/String} value - the value to set 
43162      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43163      */
43164     setValue : function(v,suppressEvent){
43165         
43166         
43167         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43168         //if(this.el && this.el.dom){
43169         //    this.el.dom.checked = this.checked;
43170         //    this.el.dom.defaultChecked = this.checked;
43171         //}
43172         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43173         //this.fireEvent("check", this, this.checked);
43174     },
43175     // private..
43176     setChecked : function(state,suppressEvent)
43177     {
43178         if (this.inSetChecked) {
43179             this.checked = state;
43180             return;
43181         }
43182         
43183     
43184         if(this.wrap){
43185             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43186         }
43187         this.checked = state;
43188         if(suppressEvent !== true){
43189             this.fireEvent('check', this, state);
43190         }
43191         this.inSetChecked = true;
43192         this.el.dom.value = state ? this.inputValue : this.valueOff;
43193         this.inSetChecked = false;
43194         
43195     },
43196     // handle setting of hidden value by some other method!!?!?
43197     setFromHidden: function()
43198     {
43199         if(!this.el){
43200             return;
43201         }
43202         //console.log("SET FROM HIDDEN");
43203         //alert('setFrom hidden');
43204         this.setValue(this.el.dom.value);
43205     },
43206     
43207     onDestroy : function()
43208     {
43209         if(this.viewEl){
43210             Roo.get(this.viewEl).remove();
43211         }
43212          
43213         Roo.form.Checkbox.superclass.onDestroy.call(this);
43214     },
43215     
43216     setBoxLabel : function(str)
43217     {
43218         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43219     }
43220
43221 });/*
43222  * Based on:
43223  * Ext JS Library 1.1.1
43224  * Copyright(c) 2006-2007, Ext JS, LLC.
43225  *
43226  * Originally Released Under LGPL - original licence link has changed is not relivant.
43227  *
43228  * Fork - LGPL
43229  * <script type="text/javascript">
43230  */
43231  
43232 /**
43233  * @class Roo.form.Radio
43234  * @extends Roo.form.Checkbox
43235  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43236  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43237  * @constructor
43238  * Creates a new Radio
43239  * @param {Object} config Configuration options
43240  */
43241 Roo.form.Radio = function(){
43242     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43243 };
43244 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43245     inputType: 'radio',
43246
43247     /**
43248      * If this radio is part of a group, it will return the selected value
43249      * @return {String}
43250      */
43251     getGroupValue : function(){
43252         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43253     },
43254     
43255     
43256     onRender : function(ct, position){
43257         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43258         
43259         if(this.inputValue !== undefined){
43260             this.el.dom.value = this.inputValue;
43261         }
43262          
43263         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43264         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43265         //var viewEl = this.wrap.createChild({ 
43266         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43267         //this.viewEl = viewEl;   
43268         //this.wrap.on('click', this.onClick,  this); 
43269         
43270         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43271         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43272         
43273         
43274         
43275         if(this.boxLabel){
43276             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43277         //    viewEl.on('click', this.onClick,  this); 
43278         }
43279          if(this.checked){
43280             this.el.dom.checked =   'checked' ;
43281         }
43282          
43283     } 
43284     
43285     
43286 });//<script type="text/javascript">
43287
43288 /*
43289  * Based  Ext JS Library 1.1.1
43290  * Copyright(c) 2006-2007, Ext JS, LLC.
43291  * LGPL
43292  *
43293  */
43294  
43295 /**
43296  * @class Roo.HtmlEditorCore
43297  * @extends Roo.Component
43298  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43299  *
43300  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43301  */
43302
43303 Roo.HtmlEditorCore = function(config){
43304     
43305     
43306     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43307     
43308     
43309     this.addEvents({
43310         /**
43311          * @event initialize
43312          * Fires when the editor is fully initialized (including the iframe)
43313          * @param {Roo.HtmlEditorCore} this
43314          */
43315         initialize: true,
43316         /**
43317          * @event activate
43318          * Fires when the editor is first receives the focus. Any insertion must wait
43319          * until after this event.
43320          * @param {Roo.HtmlEditorCore} this
43321          */
43322         activate: true,
43323          /**
43324          * @event beforesync
43325          * Fires before the textarea is updated with content from the editor iframe. Return false
43326          * to cancel the sync.
43327          * @param {Roo.HtmlEditorCore} this
43328          * @param {String} html
43329          */
43330         beforesync: true,
43331          /**
43332          * @event beforepush
43333          * Fires before the iframe editor is updated with content from the textarea. Return false
43334          * to cancel the push.
43335          * @param {Roo.HtmlEditorCore} this
43336          * @param {String} html
43337          */
43338         beforepush: true,
43339          /**
43340          * @event sync
43341          * Fires when the textarea is updated with content from the editor iframe.
43342          * @param {Roo.HtmlEditorCore} this
43343          * @param {String} html
43344          */
43345         sync: true,
43346          /**
43347          * @event push
43348          * Fires when the iframe editor is updated with content from the textarea.
43349          * @param {Roo.HtmlEditorCore} this
43350          * @param {String} html
43351          */
43352         push: true,
43353         
43354         /**
43355          * @event editorevent
43356          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43357          * @param {Roo.HtmlEditorCore} this
43358          */
43359         editorevent: true
43360         
43361     });
43362     
43363     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43364     
43365     // defaults : white / black...
43366     this.applyBlacklists();
43367     
43368     
43369     
43370 };
43371
43372
43373 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43374
43375
43376      /**
43377      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43378      */
43379     
43380     owner : false,
43381     
43382      /**
43383      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43384      *                        Roo.resizable.
43385      */
43386     resizable : false,
43387      /**
43388      * @cfg {Number} height (in pixels)
43389      */   
43390     height: 300,
43391    /**
43392      * @cfg {Number} width (in pixels)
43393      */   
43394     width: 500,
43395     
43396     /**
43397      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43398      * 
43399      */
43400     stylesheets: false,
43401     
43402     // id of frame..
43403     frameId: false,
43404     
43405     // private properties
43406     validationEvent : false,
43407     deferHeight: true,
43408     initialized : false,
43409     activated : false,
43410     sourceEditMode : false,
43411     onFocus : Roo.emptyFn,
43412     iframePad:3,
43413     hideMode:'offsets',
43414     
43415     clearUp: true,
43416     
43417     // blacklist + whitelisted elements..
43418     black: false,
43419     white: false,
43420      
43421     bodyCls : '',
43422
43423     /**
43424      * Protected method that will not generally be called directly. It
43425      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43426      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43427      */
43428     getDocMarkup : function(){
43429         // body styles..
43430         var st = '';
43431         
43432         // inherit styels from page...?? 
43433         if (this.stylesheets === false) {
43434             
43435             Roo.get(document.head).select('style').each(function(node) {
43436                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43437             });
43438             
43439             Roo.get(document.head).select('link').each(function(node) { 
43440                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43441             });
43442             
43443         } else if (!this.stylesheets.length) {
43444                 // simple..
43445                 st = '<style type="text/css">' +
43446                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43447                    '</style>';
43448         } else { 
43449             st = '<style type="text/css">' +
43450                     this.stylesheets +
43451                 '</style>';
43452         }
43453         
43454         st +=  '<style type="text/css">' +
43455             'IMG { cursor: pointer } ' +
43456         '</style>';
43457
43458         var cls = 'roo-htmleditor-body';
43459         
43460         if(this.bodyCls.length){
43461             cls += ' ' + this.bodyCls;
43462         }
43463         
43464         return '<html><head>' + st  +
43465             //<style type="text/css">' +
43466             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43467             //'</style>' +
43468             ' </head><body class="' +  cls + '"></body></html>';
43469     },
43470
43471     // private
43472     onRender : function(ct, position)
43473     {
43474         var _t = this;
43475         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43476         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43477         
43478         
43479         this.el.dom.style.border = '0 none';
43480         this.el.dom.setAttribute('tabIndex', -1);
43481         this.el.addClass('x-hidden hide');
43482         
43483         
43484         
43485         if(Roo.isIE){ // fix IE 1px bogus margin
43486             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43487         }
43488        
43489         
43490         this.frameId = Roo.id();
43491         
43492          
43493         
43494         var iframe = this.owner.wrap.createChild({
43495             tag: 'iframe',
43496             cls: 'form-control', // bootstrap..
43497             id: this.frameId,
43498             name: this.frameId,
43499             frameBorder : 'no',
43500             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43501         }, this.el
43502         );
43503         
43504         
43505         this.iframe = iframe.dom;
43506
43507          this.assignDocWin();
43508         
43509         this.doc.designMode = 'on';
43510        
43511         this.doc.open();
43512         this.doc.write(this.getDocMarkup());
43513         this.doc.close();
43514
43515         
43516         var task = { // must defer to wait for browser to be ready
43517             run : function(){
43518                 //console.log("run task?" + this.doc.readyState);
43519                 this.assignDocWin();
43520                 if(this.doc.body || this.doc.readyState == 'complete'){
43521                     try {
43522                         this.doc.designMode="on";
43523                     } catch (e) {
43524                         return;
43525                     }
43526                     Roo.TaskMgr.stop(task);
43527                     this.initEditor.defer(10, this);
43528                 }
43529             },
43530             interval : 10,
43531             duration: 10000,
43532             scope: this
43533         };
43534         Roo.TaskMgr.start(task);
43535
43536     },
43537
43538     // private
43539     onResize : function(w, h)
43540     {
43541          Roo.log('resize: ' +w + ',' + h );
43542         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43543         if(!this.iframe){
43544             return;
43545         }
43546         if(typeof w == 'number'){
43547             
43548             this.iframe.style.width = w + 'px';
43549         }
43550         if(typeof h == 'number'){
43551             
43552             this.iframe.style.height = h + 'px';
43553             if(this.doc){
43554                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43555             }
43556         }
43557         
43558     },
43559
43560     /**
43561      * Toggles the editor between standard and source edit mode.
43562      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43563      */
43564     toggleSourceEdit : function(sourceEditMode){
43565         
43566         this.sourceEditMode = sourceEditMode === true;
43567         
43568         if(this.sourceEditMode){
43569  
43570             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43571             
43572         }else{
43573             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43574             //this.iframe.className = '';
43575             this.deferFocus();
43576         }
43577         //this.setSize(this.owner.wrap.getSize());
43578         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43579     },
43580
43581     
43582   
43583
43584     /**
43585      * Protected method that will not generally be called directly. If you need/want
43586      * custom HTML cleanup, this is the method you should override.
43587      * @param {String} html The HTML to be cleaned
43588      * return {String} The cleaned HTML
43589      */
43590     cleanHtml : function(html){
43591         html = String(html);
43592         if(html.length > 5){
43593             if(Roo.isSafari){ // strip safari nonsense
43594                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43595             }
43596         }
43597         if(html == '&nbsp;'){
43598             html = '';
43599         }
43600         return html;
43601     },
43602
43603     /**
43604      * HTML Editor -> Textarea
43605      * Protected method that will not generally be called directly. Syncs the contents
43606      * of the editor iframe with the textarea.
43607      */
43608     syncValue : function(){
43609         if(this.initialized){
43610             var bd = (this.doc.body || this.doc.documentElement);
43611             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43612             var html = bd.innerHTML;
43613             if(Roo.isSafari){
43614                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43615                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43616                 if(m && m[1]){
43617                     html = '<div style="'+m[0]+'">' + html + '</div>';
43618                 }
43619             }
43620             html = this.cleanHtml(html);
43621             // fix up the special chars.. normaly like back quotes in word...
43622             // however we do not want to do this with chinese..
43623             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43624                 
43625                 var cc = match.charCodeAt();
43626
43627                 // Get the character value, handling surrogate pairs
43628                 if (match.length == 2) {
43629                     // It's a surrogate pair, calculate the Unicode code point
43630                     var high = match.charCodeAt(0) - 0xD800;
43631                     var low  = match.charCodeAt(1) - 0xDC00;
43632                     cc = (high * 0x400) + low + 0x10000;
43633                 }  else if (
43634                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43635                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43636                     (cc >= 0xf900 && cc < 0xfb00 )
43637                 ) {
43638                         return match;
43639                 }  
43640          
43641                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43642                 return "&#" + cc + ";";
43643                 
43644                 
43645             });
43646             
43647             
43648              
43649             if(this.owner.fireEvent('beforesync', this, html) !== false){
43650                 this.el.dom.value = html;
43651                 this.owner.fireEvent('sync', this, html);
43652             }
43653         }
43654     },
43655
43656     /**
43657      * Protected method that will not generally be called directly. Pushes the value of the textarea
43658      * into the iframe editor.
43659      */
43660     pushValue : function(){
43661         if(this.initialized){
43662             var v = this.el.dom.value.trim();
43663             
43664 //            if(v.length < 1){
43665 //                v = '&#160;';
43666 //            }
43667             
43668             if(this.owner.fireEvent('beforepush', this, v) !== false){
43669                 var d = (this.doc.body || this.doc.documentElement);
43670                 d.innerHTML = v;
43671                 this.cleanUpPaste();
43672                 this.el.dom.value = d.innerHTML;
43673                 this.owner.fireEvent('push', this, v);
43674             }
43675         }
43676     },
43677
43678     // private
43679     deferFocus : function(){
43680         this.focus.defer(10, this);
43681     },
43682
43683     // doc'ed in Field
43684     focus : function(){
43685         if(this.win && !this.sourceEditMode){
43686             this.win.focus();
43687         }else{
43688             this.el.focus();
43689         }
43690     },
43691     
43692     assignDocWin: function()
43693     {
43694         var iframe = this.iframe;
43695         
43696          if(Roo.isIE){
43697             this.doc = iframe.contentWindow.document;
43698             this.win = iframe.contentWindow;
43699         } else {
43700 //            if (!Roo.get(this.frameId)) {
43701 //                return;
43702 //            }
43703 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43704 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43705             
43706             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43707                 return;
43708             }
43709             
43710             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43711             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43712         }
43713     },
43714     
43715     // private
43716     initEditor : function(){
43717         //console.log("INIT EDITOR");
43718         this.assignDocWin();
43719         
43720         
43721         
43722         this.doc.designMode="on";
43723         this.doc.open();
43724         this.doc.write(this.getDocMarkup());
43725         this.doc.close();
43726         
43727         var dbody = (this.doc.body || this.doc.documentElement);
43728         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43729         // this copies styles from the containing element into thsi one..
43730         // not sure why we need all of this..
43731         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43732         
43733         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43734         //ss['background-attachment'] = 'fixed'; // w3c
43735         dbody.bgProperties = 'fixed'; // ie
43736         //Roo.DomHelper.applyStyles(dbody, ss);
43737         Roo.EventManager.on(this.doc, {
43738             //'mousedown': this.onEditorEvent,
43739             'mouseup': this.onEditorEvent,
43740             'dblclick': this.onEditorEvent,
43741             'click': this.onEditorEvent,
43742             'keyup': this.onEditorEvent,
43743             buffer:100,
43744             scope: this
43745         });
43746         if(Roo.isGecko){
43747             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43748         }
43749         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43750             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43751         }
43752         this.initialized = true;
43753
43754         this.owner.fireEvent('initialize', this);
43755         this.pushValue();
43756     },
43757
43758     // private
43759     onDestroy : function(){
43760         
43761         
43762         
43763         if(this.rendered){
43764             
43765             //for (var i =0; i < this.toolbars.length;i++) {
43766             //    // fixme - ask toolbars for heights?
43767             //    this.toolbars[i].onDestroy();
43768            // }
43769             
43770             //this.wrap.dom.innerHTML = '';
43771             //this.wrap.remove();
43772         }
43773     },
43774
43775     // private
43776     onFirstFocus : function(){
43777         
43778         this.assignDocWin();
43779         
43780         
43781         this.activated = true;
43782          
43783     
43784         if(Roo.isGecko){ // prevent silly gecko errors
43785             this.win.focus();
43786             var s = this.win.getSelection();
43787             if(!s.focusNode || s.focusNode.nodeType != 3){
43788                 var r = s.getRangeAt(0);
43789                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43790                 r.collapse(true);
43791                 this.deferFocus();
43792             }
43793             try{
43794                 this.execCmd('useCSS', true);
43795                 this.execCmd('styleWithCSS', false);
43796             }catch(e){}
43797         }
43798         this.owner.fireEvent('activate', this);
43799     },
43800
43801     // private
43802     adjustFont: function(btn){
43803         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43804         //if(Roo.isSafari){ // safari
43805         //    adjust *= 2;
43806        // }
43807         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43808         if(Roo.isSafari){ // safari
43809             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43810             v =  (v < 10) ? 10 : v;
43811             v =  (v > 48) ? 48 : v;
43812             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43813             
43814         }
43815         
43816         
43817         v = Math.max(1, v+adjust);
43818         
43819         this.execCmd('FontSize', v  );
43820     },
43821
43822     onEditorEvent : function(e)
43823     {
43824         this.owner.fireEvent('editorevent', this, e);
43825       //  this.updateToolbar();
43826         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43827     },
43828
43829     insertTag : function(tg)
43830     {
43831         // could be a bit smarter... -> wrap the current selected tRoo..
43832         if (tg.toLowerCase() == 'span' ||
43833             tg.toLowerCase() == 'code' ||
43834             tg.toLowerCase() == 'sup' ||
43835             tg.toLowerCase() == 'sub' 
43836             ) {
43837             
43838             range = this.createRange(this.getSelection());
43839             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43840             wrappingNode.appendChild(range.extractContents());
43841             range.insertNode(wrappingNode);
43842
43843             return;
43844             
43845             
43846             
43847         }
43848         this.execCmd("formatblock",   tg);
43849         
43850     },
43851     
43852     insertText : function(txt)
43853     {
43854         
43855         
43856         var range = this.createRange();
43857         range.deleteContents();
43858                //alert(Sender.getAttribute('label'));
43859                
43860         range.insertNode(this.doc.createTextNode(txt));
43861     } ,
43862     
43863      
43864
43865     /**
43866      * Executes a Midas editor command on the editor document and performs necessary focus and
43867      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43868      * @param {String} cmd The Midas command
43869      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43870      */
43871     relayCmd : function(cmd, value){
43872         this.win.focus();
43873         this.execCmd(cmd, value);
43874         this.owner.fireEvent('editorevent', this);
43875         //this.updateToolbar();
43876         this.owner.deferFocus();
43877     },
43878
43879     /**
43880      * Executes a Midas editor command directly on the editor document.
43881      * For visual commands, you should use {@link #relayCmd} instead.
43882      * <b>This should only be called after the editor is initialized.</b>
43883      * @param {String} cmd The Midas command
43884      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43885      */
43886     execCmd : function(cmd, value){
43887         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43888         this.syncValue();
43889     },
43890  
43891  
43892    
43893     /**
43894      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43895      * to insert tRoo.
43896      * @param {String} text | dom node.. 
43897      */
43898     insertAtCursor : function(text)
43899     {
43900         
43901         if(!this.activated){
43902             return;
43903         }
43904         /*
43905         if(Roo.isIE){
43906             this.win.focus();
43907             var r = this.doc.selection.createRange();
43908             if(r){
43909                 r.collapse(true);
43910                 r.pasteHTML(text);
43911                 this.syncValue();
43912                 this.deferFocus();
43913             
43914             }
43915             return;
43916         }
43917         */
43918         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43919             this.win.focus();
43920             
43921             
43922             // from jquery ui (MIT licenced)
43923             var range, node;
43924             var win = this.win;
43925             
43926             if (win.getSelection && win.getSelection().getRangeAt) {
43927                 range = win.getSelection().getRangeAt(0);
43928                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43929                 range.insertNode(node);
43930             } else if (win.document.selection && win.document.selection.createRange) {
43931                 // no firefox support
43932                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43933                 win.document.selection.createRange().pasteHTML(txt);
43934             } else {
43935                 // no firefox support
43936                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43937                 this.execCmd('InsertHTML', txt);
43938             } 
43939             
43940             this.syncValue();
43941             
43942             this.deferFocus();
43943         }
43944     },
43945  // private
43946     mozKeyPress : function(e){
43947         if(e.ctrlKey){
43948             var c = e.getCharCode(), cmd;
43949           
43950             if(c > 0){
43951                 c = String.fromCharCode(c).toLowerCase();
43952                 switch(c){
43953                     case 'b':
43954                         cmd = 'bold';
43955                         break;
43956                     case 'i':
43957                         cmd = 'italic';
43958                         break;
43959                     
43960                     case 'u':
43961                         cmd = 'underline';
43962                         break;
43963                     
43964                     case 'v':
43965                         this.cleanUpPaste.defer(100, this);
43966                         return;
43967                         
43968                 }
43969                 if(cmd){
43970                     this.win.focus();
43971                     this.execCmd(cmd);
43972                     this.deferFocus();
43973                     e.preventDefault();
43974                 }
43975                 
43976             }
43977         }
43978     },
43979
43980     // private
43981     fixKeys : function(){ // load time branching for fastest keydown performance
43982         if(Roo.isIE){
43983             return function(e){
43984                 var k = e.getKey(), r;
43985                 if(k == e.TAB){
43986                     e.stopEvent();
43987                     r = this.doc.selection.createRange();
43988                     if(r){
43989                         r.collapse(true);
43990                         r.pasteHTML('&#160;&#160;&#160;&#160;');
43991                         this.deferFocus();
43992                     }
43993                     return;
43994                 }
43995                 
43996                 if(k == e.ENTER){
43997                     r = this.doc.selection.createRange();
43998                     if(r){
43999                         var target = r.parentElement();
44000                         if(!target || target.tagName.toLowerCase() != 'li'){
44001                             e.stopEvent();
44002                             r.pasteHTML('<br />');
44003                             r.collapse(false);
44004                             r.select();
44005                         }
44006                     }
44007                 }
44008                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44009                     this.cleanUpPaste.defer(100, this);
44010                     return;
44011                 }
44012                 
44013                 
44014             };
44015         }else if(Roo.isOpera){
44016             return function(e){
44017                 var k = e.getKey();
44018                 if(k == e.TAB){
44019                     e.stopEvent();
44020                     this.win.focus();
44021                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44022                     this.deferFocus();
44023                 }
44024                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44025                     this.cleanUpPaste.defer(100, this);
44026                     return;
44027                 }
44028                 
44029             };
44030         }else if(Roo.isSafari){
44031             return function(e){
44032                 var k = e.getKey();
44033                 
44034                 if(k == e.TAB){
44035                     e.stopEvent();
44036                     this.execCmd('InsertText','\t');
44037                     this.deferFocus();
44038                     return;
44039                 }
44040                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44041                     this.cleanUpPaste.defer(100, this);
44042                     return;
44043                 }
44044                 
44045              };
44046         }
44047     }(),
44048     
44049     getAllAncestors: function()
44050     {
44051         var p = this.getSelectedNode();
44052         var a = [];
44053         if (!p) {
44054             a.push(p); // push blank onto stack..
44055             p = this.getParentElement();
44056         }
44057         
44058         
44059         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44060             a.push(p);
44061             p = p.parentNode;
44062         }
44063         a.push(this.doc.body);
44064         return a;
44065     },
44066     lastSel : false,
44067     lastSelNode : false,
44068     
44069     
44070     getSelection : function() 
44071     {
44072         this.assignDocWin();
44073         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44074     },
44075     
44076     getSelectedNode: function() 
44077     {
44078         // this may only work on Gecko!!!
44079         
44080         // should we cache this!!!!
44081         
44082         
44083         
44084          
44085         var range = this.createRange(this.getSelection()).cloneRange();
44086         
44087         if (Roo.isIE) {
44088             var parent = range.parentElement();
44089             while (true) {
44090                 var testRange = range.duplicate();
44091                 testRange.moveToElementText(parent);
44092                 if (testRange.inRange(range)) {
44093                     break;
44094                 }
44095                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44096                     break;
44097                 }
44098                 parent = parent.parentElement;
44099             }
44100             return parent;
44101         }
44102         
44103         // is ancestor a text element.
44104         var ac =  range.commonAncestorContainer;
44105         if (ac.nodeType == 3) {
44106             ac = ac.parentNode;
44107         }
44108         
44109         var ar = ac.childNodes;
44110          
44111         var nodes = [];
44112         var other_nodes = [];
44113         var has_other_nodes = false;
44114         for (var i=0;i<ar.length;i++) {
44115             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44116                 continue;
44117             }
44118             // fullly contained node.
44119             
44120             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44121                 nodes.push(ar[i]);
44122                 continue;
44123             }
44124             
44125             // probably selected..
44126             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44127                 other_nodes.push(ar[i]);
44128                 continue;
44129             }
44130             // outer..
44131             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44132                 continue;
44133             }
44134             
44135             
44136             has_other_nodes = true;
44137         }
44138         if (!nodes.length && other_nodes.length) {
44139             nodes= other_nodes;
44140         }
44141         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44142             return false;
44143         }
44144         
44145         return nodes[0];
44146     },
44147     createRange: function(sel)
44148     {
44149         // this has strange effects when using with 
44150         // top toolbar - not sure if it's a great idea.
44151         //this.editor.contentWindow.focus();
44152         if (typeof sel != "undefined") {
44153             try {
44154                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44155             } catch(e) {
44156                 return this.doc.createRange();
44157             }
44158         } else {
44159             return this.doc.createRange();
44160         }
44161     },
44162     getParentElement: function()
44163     {
44164         
44165         this.assignDocWin();
44166         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44167         
44168         var range = this.createRange(sel);
44169          
44170         try {
44171             var p = range.commonAncestorContainer;
44172             while (p.nodeType == 3) { // text node
44173                 p = p.parentNode;
44174             }
44175             return p;
44176         } catch (e) {
44177             return null;
44178         }
44179     
44180     },
44181     /***
44182      *
44183      * Range intersection.. the hard stuff...
44184      *  '-1' = before
44185      *  '0' = hits..
44186      *  '1' = after.
44187      *         [ -- selected range --- ]
44188      *   [fail]                        [fail]
44189      *
44190      *    basically..
44191      *      if end is before start or  hits it. fail.
44192      *      if start is after end or hits it fail.
44193      *
44194      *   if either hits (but other is outside. - then it's not 
44195      *   
44196      *    
44197      **/
44198     
44199     
44200     // @see http://www.thismuchiknow.co.uk/?p=64.
44201     rangeIntersectsNode : function(range, node)
44202     {
44203         var nodeRange = node.ownerDocument.createRange();
44204         try {
44205             nodeRange.selectNode(node);
44206         } catch (e) {
44207             nodeRange.selectNodeContents(node);
44208         }
44209     
44210         var rangeStartRange = range.cloneRange();
44211         rangeStartRange.collapse(true);
44212     
44213         var rangeEndRange = range.cloneRange();
44214         rangeEndRange.collapse(false);
44215     
44216         var nodeStartRange = nodeRange.cloneRange();
44217         nodeStartRange.collapse(true);
44218     
44219         var nodeEndRange = nodeRange.cloneRange();
44220         nodeEndRange.collapse(false);
44221     
44222         return rangeStartRange.compareBoundaryPoints(
44223                  Range.START_TO_START, nodeEndRange) == -1 &&
44224                rangeEndRange.compareBoundaryPoints(
44225                  Range.START_TO_START, nodeStartRange) == 1;
44226         
44227          
44228     },
44229     rangeCompareNode : function(range, node)
44230     {
44231         var nodeRange = node.ownerDocument.createRange();
44232         try {
44233             nodeRange.selectNode(node);
44234         } catch (e) {
44235             nodeRange.selectNodeContents(node);
44236         }
44237         
44238         
44239         range.collapse(true);
44240     
44241         nodeRange.collapse(true);
44242      
44243         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44244         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44245          
44246         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44247         
44248         var nodeIsBefore   =  ss == 1;
44249         var nodeIsAfter    = ee == -1;
44250         
44251         if (nodeIsBefore && nodeIsAfter) {
44252             return 0; // outer
44253         }
44254         if (!nodeIsBefore && nodeIsAfter) {
44255             return 1; //right trailed.
44256         }
44257         
44258         if (nodeIsBefore && !nodeIsAfter) {
44259             return 2;  // left trailed.
44260         }
44261         // fully contined.
44262         return 3;
44263     },
44264
44265     // private? - in a new class?
44266     cleanUpPaste :  function()
44267     {
44268         // cleans up the whole document..
44269         Roo.log('cleanuppaste');
44270         
44271         this.cleanUpChildren(this.doc.body);
44272         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44273         if (clean != this.doc.body.innerHTML) {
44274             this.doc.body.innerHTML = clean;
44275         }
44276         
44277     },
44278     
44279     cleanWordChars : function(input) {// change the chars to hex code
44280         var he = Roo.HtmlEditorCore;
44281         
44282         var output = input;
44283         Roo.each(he.swapCodes, function(sw) { 
44284             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44285             
44286             output = output.replace(swapper, sw[1]);
44287         });
44288         
44289         return output;
44290     },
44291     
44292     
44293     cleanUpChildren : function (n)
44294     {
44295         if (!n.childNodes.length) {
44296             return;
44297         }
44298         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44299            this.cleanUpChild(n.childNodes[i]);
44300         }
44301     },
44302     
44303     
44304         
44305     
44306     cleanUpChild : function (node)
44307     {
44308         var ed = this;
44309         //console.log(node);
44310         if (node.nodeName == "#text") {
44311             // clean up silly Windows -- stuff?
44312             return; 
44313         }
44314         if (node.nodeName == "#comment") {
44315             node.parentNode.removeChild(node);
44316             // clean up silly Windows -- stuff?
44317             return; 
44318         }
44319         var lcname = node.tagName.toLowerCase();
44320         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44321         // whitelist of tags..
44322         
44323         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44324             // remove node.
44325             node.parentNode.removeChild(node);
44326             return;
44327             
44328         }
44329         
44330         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44331         
44332         // spans with no attributes - just remove them..
44333         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44334             remove_keep_children = true;
44335         }
44336         
44337         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44338         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44339         
44340         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44341         //    remove_keep_children = true;
44342         //}
44343         
44344         if (remove_keep_children) {
44345             this.cleanUpChildren(node);
44346             // inserts everything just before this node...
44347             while (node.childNodes.length) {
44348                 var cn = node.childNodes[0];
44349                 node.removeChild(cn);
44350                 node.parentNode.insertBefore(cn, node);
44351             }
44352             node.parentNode.removeChild(node);
44353             return;
44354         }
44355         
44356         if (!node.attributes || !node.attributes.length) {
44357             
44358           
44359             
44360             
44361             this.cleanUpChildren(node);
44362             return;
44363         }
44364         
44365         function cleanAttr(n,v)
44366         {
44367             
44368             if (v.match(/^\./) || v.match(/^\//)) {
44369                 return;
44370             }
44371             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44372                 return;
44373             }
44374             if (v.match(/^#/)) {
44375                 return;
44376             }
44377 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44378             node.removeAttribute(n);
44379             
44380         }
44381         
44382         var cwhite = this.cwhite;
44383         var cblack = this.cblack;
44384             
44385         function cleanStyle(n,v)
44386         {
44387             if (v.match(/expression/)) { //XSS?? should we even bother..
44388                 node.removeAttribute(n);
44389                 return;
44390             }
44391             
44392             var parts = v.split(/;/);
44393             var clean = [];
44394             
44395             Roo.each(parts, function(p) {
44396                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44397                 if (!p.length) {
44398                     return true;
44399                 }
44400                 var l = p.split(':').shift().replace(/\s+/g,'');
44401                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44402                 
44403                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44404 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44405                     //node.removeAttribute(n);
44406                     return true;
44407                 }
44408                 //Roo.log()
44409                 // only allow 'c whitelisted system attributes'
44410                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44411 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44412                     //node.removeAttribute(n);
44413                     return true;
44414                 }
44415                 
44416                 
44417                  
44418                 
44419                 clean.push(p);
44420                 return true;
44421             });
44422             if (clean.length) { 
44423                 node.setAttribute(n, clean.join(';'));
44424             } else {
44425                 node.removeAttribute(n);
44426             }
44427             
44428         }
44429         
44430         
44431         for (var i = node.attributes.length-1; i > -1 ; i--) {
44432             var a = node.attributes[i];
44433             //console.log(a);
44434             
44435             if (a.name.toLowerCase().substr(0,2)=='on')  {
44436                 node.removeAttribute(a.name);
44437                 continue;
44438             }
44439             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44440                 node.removeAttribute(a.name);
44441                 continue;
44442             }
44443             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44444                 cleanAttr(a.name,a.value); // fixme..
44445                 continue;
44446             }
44447             if (a.name == 'style') {
44448                 cleanStyle(a.name,a.value);
44449                 continue;
44450             }
44451             /// clean up MS crap..
44452             // tecnically this should be a list of valid class'es..
44453             
44454             
44455             if (a.name == 'class') {
44456                 if (a.value.match(/^Mso/)) {
44457                     node.removeAttribute('class');
44458                 }
44459                 
44460                 if (a.value.match(/^body$/)) {
44461                     node.removeAttribute('class');
44462                 }
44463                 continue;
44464             }
44465             
44466             // style cleanup!?
44467             // class cleanup?
44468             
44469         }
44470         
44471         
44472         this.cleanUpChildren(node);
44473         
44474         
44475     },
44476     
44477     /**
44478      * Clean up MS wordisms...
44479      */
44480     cleanWord : function(node)
44481     {
44482         if (!node) {
44483             this.cleanWord(this.doc.body);
44484             return;
44485         }
44486         
44487         if(
44488                 node.nodeName == 'SPAN' &&
44489                 !node.hasAttributes() &&
44490                 node.childNodes.length == 1 &&
44491                 node.firstChild.nodeName == "#text"  
44492         ) {
44493             var textNode = node.firstChild;
44494             node.removeChild(textNode);
44495             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44496                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44497             }
44498             node.parentNode.insertBefore(textNode, node);
44499             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44500                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44501             }
44502             node.parentNode.removeChild(node);
44503         }
44504         
44505         if (node.nodeName == "#text") {
44506             // clean up silly Windows -- stuff?
44507             return; 
44508         }
44509         if (node.nodeName == "#comment") {
44510             node.parentNode.removeChild(node);
44511             // clean up silly Windows -- stuff?
44512             return; 
44513         }
44514         
44515         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44516             node.parentNode.removeChild(node);
44517             return;
44518         }
44519         //Roo.log(node.tagName);
44520         // remove - but keep children..
44521         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44522             //Roo.log('-- removed');
44523             while (node.childNodes.length) {
44524                 var cn = node.childNodes[0];
44525                 node.removeChild(cn);
44526                 node.parentNode.insertBefore(cn, node);
44527                 // move node to parent - and clean it..
44528                 this.cleanWord(cn);
44529             }
44530             node.parentNode.removeChild(node);
44531             /// no need to iterate chidlren = it's got none..
44532             //this.iterateChildren(node, this.cleanWord);
44533             return;
44534         }
44535         // clean styles
44536         if (node.className.length) {
44537             
44538             var cn = node.className.split(/\W+/);
44539             var cna = [];
44540             Roo.each(cn, function(cls) {
44541                 if (cls.match(/Mso[a-zA-Z]+/)) {
44542                     return;
44543                 }
44544                 cna.push(cls);
44545             });
44546             node.className = cna.length ? cna.join(' ') : '';
44547             if (!cna.length) {
44548                 node.removeAttribute("class");
44549             }
44550         }
44551         
44552         if (node.hasAttribute("lang")) {
44553             node.removeAttribute("lang");
44554         }
44555         
44556         if (node.hasAttribute("style")) {
44557             
44558             var styles = node.getAttribute("style").split(";");
44559             var nstyle = [];
44560             Roo.each(styles, function(s) {
44561                 if (!s.match(/:/)) {
44562                     return;
44563                 }
44564                 var kv = s.split(":");
44565                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44566                     return;
44567                 }
44568                 // what ever is left... we allow.
44569                 nstyle.push(s);
44570             });
44571             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44572             if (!nstyle.length) {
44573                 node.removeAttribute('style');
44574             }
44575         }
44576         this.iterateChildren(node, this.cleanWord);
44577         
44578         
44579         
44580     },
44581     /**
44582      * iterateChildren of a Node, calling fn each time, using this as the scole..
44583      * @param {DomNode} node node to iterate children of.
44584      * @param {Function} fn method of this class to call on each item.
44585      */
44586     iterateChildren : function(node, fn)
44587     {
44588         if (!node.childNodes.length) {
44589                 return;
44590         }
44591         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44592            fn.call(this, node.childNodes[i])
44593         }
44594     },
44595     
44596     
44597     /**
44598      * cleanTableWidths.
44599      *
44600      * Quite often pasting from word etc.. results in tables with column and widths.
44601      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44602      *
44603      */
44604     cleanTableWidths : function(node)
44605     {
44606          
44607          
44608         if (!node) {
44609             this.cleanTableWidths(this.doc.body);
44610             return;
44611         }
44612         
44613         // ignore list...
44614         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44615             return; 
44616         }
44617         Roo.log(node.tagName);
44618         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44619             this.iterateChildren(node, this.cleanTableWidths);
44620             return;
44621         }
44622         if (node.hasAttribute('width')) {
44623             node.removeAttribute('width');
44624         }
44625         
44626          
44627         if (node.hasAttribute("style")) {
44628             // pretty basic...
44629             
44630             var styles = node.getAttribute("style").split(";");
44631             var nstyle = [];
44632             Roo.each(styles, function(s) {
44633                 if (!s.match(/:/)) {
44634                     return;
44635                 }
44636                 var kv = s.split(":");
44637                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44638                     return;
44639                 }
44640                 // what ever is left... we allow.
44641                 nstyle.push(s);
44642             });
44643             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44644             if (!nstyle.length) {
44645                 node.removeAttribute('style');
44646             }
44647         }
44648         
44649         this.iterateChildren(node, this.cleanTableWidths);
44650         
44651         
44652     },
44653     
44654     
44655     
44656     
44657     domToHTML : function(currentElement, depth, nopadtext) {
44658         
44659         depth = depth || 0;
44660         nopadtext = nopadtext || false;
44661     
44662         if (!currentElement) {
44663             return this.domToHTML(this.doc.body);
44664         }
44665         
44666         //Roo.log(currentElement);
44667         var j;
44668         var allText = false;
44669         var nodeName = currentElement.nodeName;
44670         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44671         
44672         if  (nodeName == '#text') {
44673             
44674             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44675         }
44676         
44677         
44678         var ret = '';
44679         if (nodeName != 'BODY') {
44680              
44681             var i = 0;
44682             // Prints the node tagName, such as <A>, <IMG>, etc
44683             if (tagName) {
44684                 var attr = [];
44685                 for(i = 0; i < currentElement.attributes.length;i++) {
44686                     // quoting?
44687                     var aname = currentElement.attributes.item(i).name;
44688                     if (!currentElement.attributes.item(i).value.length) {
44689                         continue;
44690                     }
44691                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44692                 }
44693                 
44694                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44695             } 
44696             else {
44697                 
44698                 // eack
44699             }
44700         } else {
44701             tagName = false;
44702         }
44703         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44704             return ret;
44705         }
44706         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44707             nopadtext = true;
44708         }
44709         
44710         
44711         // Traverse the tree
44712         i = 0;
44713         var currentElementChild = currentElement.childNodes.item(i);
44714         var allText = true;
44715         var innerHTML  = '';
44716         lastnode = '';
44717         while (currentElementChild) {
44718             // Formatting code (indent the tree so it looks nice on the screen)
44719             var nopad = nopadtext;
44720             if (lastnode == 'SPAN') {
44721                 nopad  = true;
44722             }
44723             // text
44724             if  (currentElementChild.nodeName == '#text') {
44725                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44726                 toadd = nopadtext ? toadd : toadd.trim();
44727                 if (!nopad && toadd.length > 80) {
44728                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44729                 }
44730                 innerHTML  += toadd;
44731                 
44732                 i++;
44733                 currentElementChild = currentElement.childNodes.item(i);
44734                 lastNode = '';
44735                 continue;
44736             }
44737             allText = false;
44738             
44739             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44740                 
44741             // Recursively traverse the tree structure of the child node
44742             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44743             lastnode = currentElementChild.nodeName;
44744             i++;
44745             currentElementChild=currentElement.childNodes.item(i);
44746         }
44747         
44748         ret += innerHTML;
44749         
44750         if (!allText) {
44751                 // The remaining code is mostly for formatting the tree
44752             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44753         }
44754         
44755         
44756         if (tagName) {
44757             ret+= "</"+tagName+">";
44758         }
44759         return ret;
44760         
44761     },
44762         
44763     applyBlacklists : function()
44764     {
44765         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44766         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44767         
44768         this.white = [];
44769         this.black = [];
44770         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44771             if (b.indexOf(tag) > -1) {
44772                 return;
44773             }
44774             this.white.push(tag);
44775             
44776         }, this);
44777         
44778         Roo.each(w, function(tag) {
44779             if (b.indexOf(tag) > -1) {
44780                 return;
44781             }
44782             if (this.white.indexOf(tag) > -1) {
44783                 return;
44784             }
44785             this.white.push(tag);
44786             
44787         }, this);
44788         
44789         
44790         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44791             if (w.indexOf(tag) > -1) {
44792                 return;
44793             }
44794             this.black.push(tag);
44795             
44796         }, this);
44797         
44798         Roo.each(b, function(tag) {
44799             if (w.indexOf(tag) > -1) {
44800                 return;
44801             }
44802             if (this.black.indexOf(tag) > -1) {
44803                 return;
44804             }
44805             this.black.push(tag);
44806             
44807         }, this);
44808         
44809         
44810         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44811         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44812         
44813         this.cwhite = [];
44814         this.cblack = [];
44815         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44816             if (b.indexOf(tag) > -1) {
44817                 return;
44818             }
44819             this.cwhite.push(tag);
44820             
44821         }, this);
44822         
44823         Roo.each(w, function(tag) {
44824             if (b.indexOf(tag) > -1) {
44825                 return;
44826             }
44827             if (this.cwhite.indexOf(tag) > -1) {
44828                 return;
44829             }
44830             this.cwhite.push(tag);
44831             
44832         }, this);
44833         
44834         
44835         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44836             if (w.indexOf(tag) > -1) {
44837                 return;
44838             }
44839             this.cblack.push(tag);
44840             
44841         }, this);
44842         
44843         Roo.each(b, function(tag) {
44844             if (w.indexOf(tag) > -1) {
44845                 return;
44846             }
44847             if (this.cblack.indexOf(tag) > -1) {
44848                 return;
44849             }
44850             this.cblack.push(tag);
44851             
44852         }, this);
44853     },
44854     
44855     setStylesheets : function(stylesheets)
44856     {
44857         if(typeof(stylesheets) == 'string'){
44858             Roo.get(this.iframe.contentDocument.head).createChild({
44859                 tag : 'link',
44860                 rel : 'stylesheet',
44861                 type : 'text/css',
44862                 href : stylesheets
44863             });
44864             
44865             return;
44866         }
44867         var _this = this;
44868      
44869         Roo.each(stylesheets, function(s) {
44870             if(!s.length){
44871                 return;
44872             }
44873             
44874             Roo.get(_this.iframe.contentDocument.head).createChild({
44875                 tag : 'link',
44876                 rel : 'stylesheet',
44877                 type : 'text/css',
44878                 href : s
44879             });
44880         });
44881
44882         
44883     },
44884     
44885     removeStylesheets : function()
44886     {
44887         var _this = this;
44888         
44889         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44890             s.remove();
44891         });
44892     },
44893     
44894     setStyle : function(style)
44895     {
44896         Roo.get(this.iframe.contentDocument.head).createChild({
44897             tag : 'style',
44898             type : 'text/css',
44899             html : style
44900         });
44901
44902         return;
44903     }
44904     
44905     // hide stuff that is not compatible
44906     /**
44907      * @event blur
44908      * @hide
44909      */
44910     /**
44911      * @event change
44912      * @hide
44913      */
44914     /**
44915      * @event focus
44916      * @hide
44917      */
44918     /**
44919      * @event specialkey
44920      * @hide
44921      */
44922     /**
44923      * @cfg {String} fieldClass @hide
44924      */
44925     /**
44926      * @cfg {String} focusClass @hide
44927      */
44928     /**
44929      * @cfg {String} autoCreate @hide
44930      */
44931     /**
44932      * @cfg {String} inputType @hide
44933      */
44934     /**
44935      * @cfg {String} invalidClass @hide
44936      */
44937     /**
44938      * @cfg {String} invalidText @hide
44939      */
44940     /**
44941      * @cfg {String} msgFx @hide
44942      */
44943     /**
44944      * @cfg {String} validateOnBlur @hide
44945      */
44946 });
44947
44948 Roo.HtmlEditorCore.white = [
44949         'area', 'br', 'img', 'input', 'hr', 'wbr',
44950         
44951        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
44952        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
44953        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
44954        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
44955        'table',   'ul',         'xmp', 
44956        
44957        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
44958       'thead',   'tr', 
44959      
44960       'dir', 'menu', 'ol', 'ul', 'dl',
44961        
44962       'embed',  'object'
44963 ];
44964
44965
44966 Roo.HtmlEditorCore.black = [
44967     //    'embed',  'object', // enable - backend responsiblity to clean thiese
44968         'applet', // 
44969         'base',   'basefont', 'bgsound', 'blink',  'body', 
44970         'frame',  'frameset', 'head',    'html',   'ilayer', 
44971         'iframe', 'layer',  'link',     'meta',    'object',   
44972         'script', 'style' ,'title',  'xml' // clean later..
44973 ];
44974 Roo.HtmlEditorCore.clean = [
44975     'script', 'style', 'title', 'xml'
44976 ];
44977 Roo.HtmlEditorCore.remove = [
44978     'font'
44979 ];
44980 // attributes..
44981
44982 Roo.HtmlEditorCore.ablack = [
44983     'on'
44984 ];
44985     
44986 Roo.HtmlEditorCore.aclean = [ 
44987     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
44988 ];
44989
44990 // protocols..
44991 Roo.HtmlEditorCore.pwhite= [
44992         'http',  'https',  'mailto'
44993 ];
44994
44995 // white listed style attributes.
44996 Roo.HtmlEditorCore.cwhite= [
44997       //  'text-align', /// default is to allow most things..
44998       
44999          
45000 //        'font-size'//??
45001 ];
45002
45003 // black listed style attributes.
45004 Roo.HtmlEditorCore.cblack= [
45005       //  'font-size' -- this can be set by the project 
45006 ];
45007
45008
45009 Roo.HtmlEditorCore.swapCodes   =[ 
45010     [    8211, "--" ], 
45011     [    8212, "--" ], 
45012     [    8216,  "'" ],  
45013     [    8217, "'" ],  
45014     [    8220, '"' ],  
45015     [    8221, '"' ],  
45016     [    8226, "*" ],  
45017     [    8230, "..." ]
45018 ]; 
45019
45020     //<script type="text/javascript">
45021
45022 /*
45023  * Ext JS Library 1.1.1
45024  * Copyright(c) 2006-2007, Ext JS, LLC.
45025  * Licence LGPL
45026  * 
45027  */
45028  
45029  
45030 Roo.form.HtmlEditor = function(config){
45031     
45032     
45033     
45034     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45035     
45036     if (!this.toolbars) {
45037         this.toolbars = [];
45038     }
45039     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45040     
45041     
45042 };
45043
45044 /**
45045  * @class Roo.form.HtmlEditor
45046  * @extends Roo.form.Field
45047  * Provides a lightweight HTML Editor component.
45048  *
45049  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45050  * 
45051  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45052  * supported by this editor.</b><br/><br/>
45053  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45054  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45055  */
45056 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45057     /**
45058      * @cfg {Boolean} clearUp
45059      */
45060     clearUp : true,
45061       /**
45062      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45063      */
45064     toolbars : false,
45065    
45066      /**
45067      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45068      *                        Roo.resizable.
45069      */
45070     resizable : false,
45071      /**
45072      * @cfg {Number} height (in pixels)
45073      */   
45074     height: 300,
45075    /**
45076      * @cfg {Number} width (in pixels)
45077      */   
45078     width: 500,
45079     
45080     /**
45081      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45082      * 
45083      */
45084     stylesheets: false,
45085     
45086     
45087      /**
45088      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45089      * 
45090      */
45091     cblack: false,
45092     /**
45093      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45094      * 
45095      */
45096     cwhite: false,
45097     
45098      /**
45099      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45100      * 
45101      */
45102     black: false,
45103     /**
45104      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45105      * 
45106      */
45107     white: false,
45108     
45109     // id of frame..
45110     frameId: false,
45111     
45112     // private properties
45113     validationEvent : false,
45114     deferHeight: true,
45115     initialized : false,
45116     activated : false,
45117     
45118     onFocus : Roo.emptyFn,
45119     iframePad:3,
45120     hideMode:'offsets',
45121     
45122     actionMode : 'container', // defaults to hiding it...
45123     
45124     defaultAutoCreate : { // modified by initCompnoent..
45125         tag: "textarea",
45126         style:"width:500px;height:300px;",
45127         autocomplete: "new-password"
45128     },
45129
45130     // private
45131     initComponent : function(){
45132         this.addEvents({
45133             /**
45134              * @event initialize
45135              * Fires when the editor is fully initialized (including the iframe)
45136              * @param {HtmlEditor} this
45137              */
45138             initialize: true,
45139             /**
45140              * @event activate
45141              * Fires when the editor is first receives the focus. Any insertion must wait
45142              * until after this event.
45143              * @param {HtmlEditor} this
45144              */
45145             activate: true,
45146              /**
45147              * @event beforesync
45148              * Fires before the textarea is updated with content from the editor iframe. Return false
45149              * to cancel the sync.
45150              * @param {HtmlEditor} this
45151              * @param {String} html
45152              */
45153             beforesync: true,
45154              /**
45155              * @event beforepush
45156              * Fires before the iframe editor is updated with content from the textarea. Return false
45157              * to cancel the push.
45158              * @param {HtmlEditor} this
45159              * @param {String} html
45160              */
45161             beforepush: true,
45162              /**
45163              * @event sync
45164              * Fires when the textarea is updated with content from the editor iframe.
45165              * @param {HtmlEditor} this
45166              * @param {String} html
45167              */
45168             sync: true,
45169              /**
45170              * @event push
45171              * Fires when the iframe editor is updated with content from the textarea.
45172              * @param {HtmlEditor} this
45173              * @param {String} html
45174              */
45175             push: true,
45176              /**
45177              * @event editmodechange
45178              * Fires when the editor switches edit modes
45179              * @param {HtmlEditor} this
45180              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45181              */
45182             editmodechange: true,
45183             /**
45184              * @event editorevent
45185              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45186              * @param {HtmlEditor} this
45187              */
45188             editorevent: true,
45189             /**
45190              * @event firstfocus
45191              * Fires when on first focus - needed by toolbars..
45192              * @param {HtmlEditor} this
45193              */
45194             firstfocus: true,
45195             /**
45196              * @event autosave
45197              * Auto save the htmlEditor value as a file into Events
45198              * @param {HtmlEditor} this
45199              */
45200             autosave: true,
45201             /**
45202              * @event savedpreview
45203              * preview the saved version of htmlEditor
45204              * @param {HtmlEditor} this
45205              */
45206             savedpreview: true,
45207             
45208             /**
45209             * @event stylesheetsclick
45210             * Fires when press the Sytlesheets button
45211             * @param {Roo.HtmlEditorCore} this
45212             */
45213             stylesheetsclick: true
45214         });
45215         this.defaultAutoCreate =  {
45216             tag: "textarea",
45217             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45218             autocomplete: "new-password"
45219         };
45220     },
45221
45222     /**
45223      * Protected method that will not generally be called directly. It
45224      * is called when the editor creates its toolbar. Override this method if you need to
45225      * add custom toolbar buttons.
45226      * @param {HtmlEditor} editor
45227      */
45228     createToolbar : function(editor){
45229         Roo.log("create toolbars");
45230         if (!editor.toolbars || !editor.toolbars.length) {
45231             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45232         }
45233         
45234         for (var i =0 ; i < editor.toolbars.length;i++) {
45235             editor.toolbars[i] = Roo.factory(
45236                     typeof(editor.toolbars[i]) == 'string' ?
45237                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45238                 Roo.form.HtmlEditor);
45239             editor.toolbars[i].init(editor);
45240         }
45241          
45242         
45243     },
45244
45245      
45246     // private
45247     onRender : function(ct, position)
45248     {
45249         var _t = this;
45250         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45251         
45252         this.wrap = this.el.wrap({
45253             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45254         });
45255         
45256         this.editorcore.onRender(ct, position);
45257          
45258         if (this.resizable) {
45259             this.resizeEl = new Roo.Resizable(this.wrap, {
45260                 pinned : true,
45261                 wrap: true,
45262                 dynamic : true,
45263                 minHeight : this.height,
45264                 height: this.height,
45265                 handles : this.resizable,
45266                 width: this.width,
45267                 listeners : {
45268                     resize : function(r, w, h) {
45269                         _t.onResize(w,h); // -something
45270                     }
45271                 }
45272             });
45273             
45274         }
45275         this.createToolbar(this);
45276        
45277         
45278         if(!this.width){
45279             this.setSize(this.wrap.getSize());
45280         }
45281         if (this.resizeEl) {
45282             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45283             // should trigger onReize..
45284         }
45285         
45286         this.keyNav = new Roo.KeyNav(this.el, {
45287             
45288             "tab" : function(e){
45289                 e.preventDefault();
45290                 
45291                 var value = this.getValue();
45292                 
45293                 var start = this.el.dom.selectionStart;
45294                 var end = this.el.dom.selectionEnd;
45295                 
45296                 if(!e.shiftKey){
45297                     
45298                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45299                     this.el.dom.setSelectionRange(end + 1, end + 1);
45300                     return;
45301                 }
45302                 
45303                 var f = value.substring(0, start).split("\t");
45304                 
45305                 if(f.pop().length != 0){
45306                     return;
45307                 }
45308                 
45309                 this.setValue(f.join("\t") + value.substring(end));
45310                 this.el.dom.setSelectionRange(start - 1, start - 1);
45311                 
45312             },
45313             
45314             "home" : function(e){
45315                 e.preventDefault();
45316                 
45317                 var curr = this.el.dom.selectionStart;
45318                 var lines = this.getValue().split("\n");
45319                 
45320                 if(!lines.length){
45321                     return;
45322                 }
45323                 
45324                 if(e.ctrlKey){
45325                     this.el.dom.setSelectionRange(0, 0);
45326                     return;
45327                 }
45328                 
45329                 var pos = 0;
45330                 
45331                 for (var i = 0; i < lines.length;i++) {
45332                     pos += lines[i].length;
45333                     
45334                     if(i != 0){
45335                         pos += 1;
45336                     }
45337                     
45338                     if(pos < curr){
45339                         continue;
45340                     }
45341                     
45342                     pos -= lines[i].length;
45343                     
45344                     break;
45345                 }
45346                 
45347                 if(!e.shiftKey){
45348                     this.el.dom.setSelectionRange(pos, pos);
45349                     return;
45350                 }
45351                 
45352                 this.el.dom.selectionStart = pos;
45353                 this.el.dom.selectionEnd = curr;
45354             },
45355             
45356             "end" : function(e){
45357                 e.preventDefault();
45358                 
45359                 var curr = this.el.dom.selectionStart;
45360                 var lines = this.getValue().split("\n");
45361                 
45362                 if(!lines.length){
45363                     return;
45364                 }
45365                 
45366                 if(e.ctrlKey){
45367                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45368                     return;
45369                 }
45370                 
45371                 var pos = 0;
45372                 
45373                 for (var i = 0; i < lines.length;i++) {
45374                     
45375                     pos += lines[i].length;
45376                     
45377                     if(i != 0){
45378                         pos += 1;
45379                     }
45380                     
45381                     if(pos < curr){
45382                         continue;
45383                     }
45384                     
45385                     break;
45386                 }
45387                 
45388                 if(!e.shiftKey){
45389                     this.el.dom.setSelectionRange(pos, pos);
45390                     return;
45391                 }
45392                 
45393                 this.el.dom.selectionStart = curr;
45394                 this.el.dom.selectionEnd = pos;
45395             },
45396
45397             scope : this,
45398
45399             doRelay : function(foo, bar, hname){
45400                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45401             },
45402
45403             forceKeyDown: true
45404         });
45405         
45406 //        if(this.autosave && this.w){
45407 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45408 //        }
45409     },
45410
45411     // private
45412     onResize : function(w, h)
45413     {
45414         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45415         var ew = false;
45416         var eh = false;
45417         
45418         if(this.el ){
45419             if(typeof w == 'number'){
45420                 var aw = w - this.wrap.getFrameWidth('lr');
45421                 this.el.setWidth(this.adjustWidth('textarea', aw));
45422                 ew = aw;
45423             }
45424             if(typeof h == 'number'){
45425                 var tbh = 0;
45426                 for (var i =0; i < this.toolbars.length;i++) {
45427                     // fixme - ask toolbars for heights?
45428                     tbh += this.toolbars[i].tb.el.getHeight();
45429                     if (this.toolbars[i].footer) {
45430                         tbh += this.toolbars[i].footer.el.getHeight();
45431                     }
45432                 }
45433                 
45434                 
45435                 
45436                 
45437                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45438                 ah -= 5; // knock a few pixes off for look..
45439 //                Roo.log(ah);
45440                 this.el.setHeight(this.adjustWidth('textarea', ah));
45441                 var eh = ah;
45442             }
45443         }
45444         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45445         this.editorcore.onResize(ew,eh);
45446         
45447     },
45448
45449     /**
45450      * Toggles the editor between standard and source edit mode.
45451      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45452      */
45453     toggleSourceEdit : function(sourceEditMode)
45454     {
45455         this.editorcore.toggleSourceEdit(sourceEditMode);
45456         
45457         if(this.editorcore.sourceEditMode){
45458             Roo.log('editor - showing textarea');
45459             
45460 //            Roo.log('in');
45461 //            Roo.log(this.syncValue());
45462             this.editorcore.syncValue();
45463             this.el.removeClass('x-hidden');
45464             this.el.dom.removeAttribute('tabIndex');
45465             this.el.focus();
45466             
45467             for (var i = 0; i < this.toolbars.length; i++) {
45468                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45469                     this.toolbars[i].tb.hide();
45470                     this.toolbars[i].footer.hide();
45471                 }
45472             }
45473             
45474         }else{
45475             Roo.log('editor - hiding textarea');
45476 //            Roo.log('out')
45477 //            Roo.log(this.pushValue()); 
45478             this.editorcore.pushValue();
45479             
45480             this.el.addClass('x-hidden');
45481             this.el.dom.setAttribute('tabIndex', -1);
45482             
45483             for (var i = 0; i < this.toolbars.length; i++) {
45484                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45485                     this.toolbars[i].tb.show();
45486                     this.toolbars[i].footer.show();
45487                 }
45488             }
45489             
45490             //this.deferFocus();
45491         }
45492         
45493         this.setSize(this.wrap.getSize());
45494         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45495         
45496         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45497     },
45498  
45499     // private (for BoxComponent)
45500     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45501
45502     // private (for BoxComponent)
45503     getResizeEl : function(){
45504         return this.wrap;
45505     },
45506
45507     // private (for BoxComponent)
45508     getPositionEl : function(){
45509         return this.wrap;
45510     },
45511
45512     // private
45513     initEvents : function(){
45514         this.originalValue = this.getValue();
45515     },
45516
45517     /**
45518      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45519      * @method
45520      */
45521     markInvalid : Roo.emptyFn,
45522     /**
45523      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45524      * @method
45525      */
45526     clearInvalid : Roo.emptyFn,
45527
45528     setValue : function(v){
45529         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45530         this.editorcore.pushValue();
45531     },
45532
45533      
45534     // private
45535     deferFocus : function(){
45536         this.focus.defer(10, this);
45537     },
45538
45539     // doc'ed in Field
45540     focus : function(){
45541         this.editorcore.focus();
45542         
45543     },
45544       
45545
45546     // private
45547     onDestroy : function(){
45548         
45549         
45550         
45551         if(this.rendered){
45552             
45553             for (var i =0; i < this.toolbars.length;i++) {
45554                 // fixme - ask toolbars for heights?
45555                 this.toolbars[i].onDestroy();
45556             }
45557             
45558             this.wrap.dom.innerHTML = '';
45559             this.wrap.remove();
45560         }
45561     },
45562
45563     // private
45564     onFirstFocus : function(){
45565         //Roo.log("onFirstFocus");
45566         this.editorcore.onFirstFocus();
45567          for (var i =0; i < this.toolbars.length;i++) {
45568             this.toolbars[i].onFirstFocus();
45569         }
45570         
45571     },
45572     
45573     // private
45574     syncValue : function()
45575     {
45576         this.editorcore.syncValue();
45577     },
45578     
45579     pushValue : function()
45580     {
45581         this.editorcore.pushValue();
45582     },
45583     
45584     setStylesheets : function(stylesheets)
45585     {
45586         this.editorcore.setStylesheets(stylesheets);
45587     },
45588     
45589     removeStylesheets : function()
45590     {
45591         this.editorcore.removeStylesheets();
45592     }
45593      
45594     
45595     // hide stuff that is not compatible
45596     /**
45597      * @event blur
45598      * @hide
45599      */
45600     /**
45601      * @event change
45602      * @hide
45603      */
45604     /**
45605      * @event focus
45606      * @hide
45607      */
45608     /**
45609      * @event specialkey
45610      * @hide
45611      */
45612     /**
45613      * @cfg {String} fieldClass @hide
45614      */
45615     /**
45616      * @cfg {String} focusClass @hide
45617      */
45618     /**
45619      * @cfg {String} autoCreate @hide
45620      */
45621     /**
45622      * @cfg {String} inputType @hide
45623      */
45624     /**
45625      * @cfg {String} invalidClass @hide
45626      */
45627     /**
45628      * @cfg {String} invalidText @hide
45629      */
45630     /**
45631      * @cfg {String} msgFx @hide
45632      */
45633     /**
45634      * @cfg {String} validateOnBlur @hide
45635      */
45636 });
45637  
45638     // <script type="text/javascript">
45639 /*
45640  * Based on
45641  * Ext JS Library 1.1.1
45642  * Copyright(c) 2006-2007, Ext JS, LLC.
45643  *  
45644  
45645  */
45646
45647 /**
45648  * @class Roo.form.HtmlEditorToolbar1
45649  * Basic Toolbar
45650  * 
45651  * Usage:
45652  *
45653  new Roo.form.HtmlEditor({
45654     ....
45655     toolbars : [
45656         new Roo.form.HtmlEditorToolbar1({
45657             disable : { fonts: 1 , format: 1, ..., ... , ...],
45658             btns : [ .... ]
45659         })
45660     }
45661      
45662  * 
45663  * @cfg {Object} disable List of elements to disable..
45664  * @cfg {Array} btns List of additional buttons.
45665  * 
45666  * 
45667  * NEEDS Extra CSS? 
45668  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45669  */
45670  
45671 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45672 {
45673     
45674     Roo.apply(this, config);
45675     
45676     // default disabled, based on 'good practice'..
45677     this.disable = this.disable || {};
45678     Roo.applyIf(this.disable, {
45679         fontSize : true,
45680         colors : true,
45681         specialElements : true
45682     });
45683     
45684     
45685     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45686     // dont call parent... till later.
45687 }
45688
45689 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45690     
45691     tb: false,
45692     
45693     rendered: false,
45694     
45695     editor : false,
45696     editorcore : false,
45697     /**
45698      * @cfg {Object} disable  List of toolbar elements to disable
45699          
45700      */
45701     disable : false,
45702     
45703     
45704      /**
45705      * @cfg {String} createLinkText The default text for the create link prompt
45706      */
45707     createLinkText : 'Please enter the URL for the link:',
45708     /**
45709      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45710      */
45711     defaultLinkValue : 'http:/'+'/',
45712    
45713     
45714       /**
45715      * @cfg {Array} fontFamilies An array of available font families
45716      */
45717     fontFamilies : [
45718         'Arial',
45719         'Courier New',
45720         'Tahoma',
45721         'Times New Roman',
45722         'Verdana'
45723     ],
45724     
45725     specialChars : [
45726            "&#169;",
45727           "&#174;",     
45728           "&#8482;",    
45729           "&#163;" ,    
45730          // "&#8212;",    
45731           "&#8230;",    
45732           "&#247;" ,    
45733         //  "&#225;" ,     ?? a acute?
45734            "&#8364;"    , //Euro
45735        //   "&#8220;"    ,
45736         //  "&#8221;"    ,
45737         //  "&#8226;"    ,
45738           "&#176;"  //   , // degrees
45739
45740          // "&#233;"     , // e ecute
45741          // "&#250;"     , // u ecute?
45742     ],
45743     
45744     specialElements : [
45745         {
45746             text: "Insert Table",
45747             xtype: 'MenuItem',
45748             xns : Roo.Menu,
45749             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45750                 
45751         },
45752         {    
45753             text: "Insert Image",
45754             xtype: 'MenuItem',
45755             xns : Roo.Menu,
45756             ihtml : '<img src="about:blank"/>'
45757             
45758         }
45759         
45760          
45761     ],
45762     
45763     
45764     inputElements : [ 
45765             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45766             "input:submit", "input:button", "select", "textarea", "label" ],
45767     formats : [
45768         ["p"] ,  
45769         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45770         ["pre"],[ "code"], 
45771         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45772         ['div'],['span'],
45773         ['sup'],['sub']
45774     ],
45775     
45776     cleanStyles : [
45777         "font-size"
45778     ],
45779      /**
45780      * @cfg {String} defaultFont default font to use.
45781      */
45782     defaultFont: 'tahoma',
45783    
45784     fontSelect : false,
45785     
45786     
45787     formatCombo : false,
45788     
45789     init : function(editor)
45790     {
45791         this.editor = editor;
45792         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45793         var editorcore = this.editorcore;
45794         
45795         var _t = this;
45796         
45797         var fid = editorcore.frameId;
45798         var etb = this;
45799         function btn(id, toggle, handler){
45800             var xid = fid + '-'+ id ;
45801             return {
45802                 id : xid,
45803                 cmd : id,
45804                 cls : 'x-btn-icon x-edit-'+id,
45805                 enableToggle:toggle !== false,
45806                 scope: _t, // was editor...
45807                 handler:handler||_t.relayBtnCmd,
45808                 clickEvent:'mousedown',
45809                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45810                 tabIndex:-1
45811             };
45812         }
45813         
45814         
45815         
45816         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45817         this.tb = tb;
45818          // stop form submits
45819         tb.el.on('click', function(e){
45820             e.preventDefault(); // what does this do?
45821         });
45822
45823         if(!this.disable.font) { // && !Roo.isSafari){
45824             /* why no safari for fonts 
45825             editor.fontSelect = tb.el.createChild({
45826                 tag:'select',
45827                 tabIndex: -1,
45828                 cls:'x-font-select',
45829                 html: this.createFontOptions()
45830             });
45831             
45832             editor.fontSelect.on('change', function(){
45833                 var font = editor.fontSelect.dom.value;
45834                 editor.relayCmd('fontname', font);
45835                 editor.deferFocus();
45836             }, editor);
45837             
45838             tb.add(
45839                 editor.fontSelect.dom,
45840                 '-'
45841             );
45842             */
45843             
45844         };
45845         if(!this.disable.formats){
45846             this.formatCombo = new Roo.form.ComboBox({
45847                 store: new Roo.data.SimpleStore({
45848                     id : 'tag',
45849                     fields: ['tag'],
45850                     data : this.formats // from states.js
45851                 }),
45852                 blockFocus : true,
45853                 name : '',
45854                 //autoCreate : {tag: "div",  size: "20"},
45855                 displayField:'tag',
45856                 typeAhead: false,
45857                 mode: 'local',
45858                 editable : false,
45859                 triggerAction: 'all',
45860                 emptyText:'Add tag',
45861                 selectOnFocus:true,
45862                 width:135,
45863                 listeners : {
45864                     'select': function(c, r, i) {
45865                         editorcore.insertTag(r.get('tag'));
45866                         editor.focus();
45867                     }
45868                 }
45869
45870             });
45871             tb.addField(this.formatCombo);
45872             
45873         }
45874         
45875         if(!this.disable.format){
45876             tb.add(
45877                 btn('bold'),
45878                 btn('italic'),
45879                 btn('underline'),
45880                 btn('strikethrough')
45881             );
45882         };
45883         if(!this.disable.fontSize){
45884             tb.add(
45885                 '-',
45886                 
45887                 
45888                 btn('increasefontsize', false, editorcore.adjustFont),
45889                 btn('decreasefontsize', false, editorcore.adjustFont)
45890             );
45891         };
45892         
45893         
45894         if(!this.disable.colors){
45895             tb.add(
45896                 '-', {
45897                     id:editorcore.frameId +'-forecolor',
45898                     cls:'x-btn-icon x-edit-forecolor',
45899                     clickEvent:'mousedown',
45900                     tooltip: this.buttonTips['forecolor'] || undefined,
45901                     tabIndex:-1,
45902                     menu : new Roo.menu.ColorMenu({
45903                         allowReselect: true,
45904                         focus: Roo.emptyFn,
45905                         value:'000000',
45906                         plain:true,
45907                         selectHandler: function(cp, color){
45908                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45909                             editor.deferFocus();
45910                         },
45911                         scope: editorcore,
45912                         clickEvent:'mousedown'
45913                     })
45914                 }, {
45915                     id:editorcore.frameId +'backcolor',
45916                     cls:'x-btn-icon x-edit-backcolor',
45917                     clickEvent:'mousedown',
45918                     tooltip: this.buttonTips['backcolor'] || undefined,
45919                     tabIndex:-1,
45920                     menu : new Roo.menu.ColorMenu({
45921                         focus: Roo.emptyFn,
45922                         value:'FFFFFF',
45923                         plain:true,
45924                         allowReselect: true,
45925                         selectHandler: function(cp, color){
45926                             if(Roo.isGecko){
45927                                 editorcore.execCmd('useCSS', false);
45928                                 editorcore.execCmd('hilitecolor', color);
45929                                 editorcore.execCmd('useCSS', true);
45930                                 editor.deferFocus();
45931                             }else{
45932                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45933                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
45934                                 editor.deferFocus();
45935                             }
45936                         },
45937                         scope:editorcore,
45938                         clickEvent:'mousedown'
45939                     })
45940                 }
45941             );
45942         };
45943         // now add all the items...
45944         
45945
45946         if(!this.disable.alignments){
45947             tb.add(
45948                 '-',
45949                 btn('justifyleft'),
45950                 btn('justifycenter'),
45951                 btn('justifyright')
45952             );
45953         };
45954
45955         //if(!Roo.isSafari){
45956             if(!this.disable.links){
45957                 tb.add(
45958                     '-',
45959                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
45960                 );
45961             };
45962
45963             if(!this.disable.lists){
45964                 tb.add(
45965                     '-',
45966                     btn('insertorderedlist'),
45967                     btn('insertunorderedlist')
45968                 );
45969             }
45970             if(!this.disable.sourceEdit){
45971                 tb.add(
45972                     '-',
45973                     btn('sourceedit', true, function(btn){
45974                         this.toggleSourceEdit(btn.pressed);
45975                     })
45976                 );
45977             }
45978         //}
45979         
45980         var smenu = { };
45981         // special menu.. - needs to be tidied up..
45982         if (!this.disable.special) {
45983             smenu = {
45984                 text: "&#169;",
45985                 cls: 'x-edit-none',
45986                 
45987                 menu : {
45988                     items : []
45989                 }
45990             };
45991             for (var i =0; i < this.specialChars.length; i++) {
45992                 smenu.menu.items.push({
45993                     
45994                     html: this.specialChars[i],
45995                     handler: function(a,b) {
45996                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
45997                         //editor.insertAtCursor(a.html);
45998                         
45999                     },
46000                     tabIndex:-1
46001                 });
46002             }
46003             
46004             
46005             tb.add(smenu);
46006             
46007             
46008         }
46009         
46010         var cmenu = { };
46011         if (!this.disable.cleanStyles) {
46012             cmenu = {
46013                 cls: 'x-btn-icon x-btn-clear',
46014                 
46015                 menu : {
46016                     items : []
46017                 }
46018             };
46019             for (var i =0; i < this.cleanStyles.length; i++) {
46020                 cmenu.menu.items.push({
46021                     actiontype : this.cleanStyles[i],
46022                     html: 'Remove ' + this.cleanStyles[i],
46023                     handler: function(a,b) {
46024 //                        Roo.log(a);
46025 //                        Roo.log(b);
46026                         var c = Roo.get(editorcore.doc.body);
46027                         c.select('[style]').each(function(s) {
46028                             s.dom.style.removeProperty(a.actiontype);
46029                         });
46030                         editorcore.syncValue();
46031                     },
46032                     tabIndex:-1
46033                 });
46034             }
46035              cmenu.menu.items.push({
46036                 actiontype : 'tablewidths',
46037                 html: 'Remove Table Widths',
46038                 handler: function(a,b) {
46039                     editorcore.cleanTableWidths();
46040                     editorcore.syncValue();
46041                 },
46042                 tabIndex:-1
46043             });
46044             cmenu.menu.items.push({
46045                 actiontype : 'word',
46046                 html: 'Remove MS Word Formating',
46047                 handler: function(a,b) {
46048                     editorcore.cleanWord();
46049                     editorcore.syncValue();
46050                 },
46051                 tabIndex:-1
46052             });
46053             
46054             cmenu.menu.items.push({
46055                 actiontype : 'all',
46056                 html: 'Remove All Styles',
46057                 handler: function(a,b) {
46058                     
46059                     var c = Roo.get(editorcore.doc.body);
46060                     c.select('[style]').each(function(s) {
46061                         s.dom.removeAttribute('style');
46062                     });
46063                     editorcore.syncValue();
46064                 },
46065                 tabIndex:-1
46066             });
46067             
46068             cmenu.menu.items.push({
46069                 actiontype : 'all',
46070                 html: 'Remove All CSS Classes',
46071                 handler: function(a,b) {
46072                     
46073                     var c = Roo.get(editorcore.doc.body);
46074                     c.select('[class]').each(function(s) {
46075                         s.dom.removeAttribute('class');
46076                     });
46077                     editorcore.cleanWord();
46078                     editorcore.syncValue();
46079                 },
46080                 tabIndex:-1
46081             });
46082             
46083              cmenu.menu.items.push({
46084                 actiontype : 'tidy',
46085                 html: 'Tidy HTML Source',
46086                 handler: function(a,b) {
46087                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46088                     editorcore.syncValue();
46089                 },
46090                 tabIndex:-1
46091             });
46092             
46093             
46094             tb.add(cmenu);
46095         }
46096          
46097         if (!this.disable.specialElements) {
46098             var semenu = {
46099                 text: "Other;",
46100                 cls: 'x-edit-none',
46101                 menu : {
46102                     items : []
46103                 }
46104             };
46105             for (var i =0; i < this.specialElements.length; i++) {
46106                 semenu.menu.items.push(
46107                     Roo.apply({ 
46108                         handler: function(a,b) {
46109                             editor.insertAtCursor(this.ihtml);
46110                         }
46111                     }, this.specialElements[i])
46112                 );
46113                     
46114             }
46115             
46116             tb.add(semenu);
46117             
46118             
46119         }
46120          
46121         
46122         if (this.btns) {
46123             for(var i =0; i< this.btns.length;i++) {
46124                 var b = Roo.factory(this.btns[i],Roo.form);
46125                 b.cls =  'x-edit-none';
46126                 
46127                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46128                     b.cls += ' x-init-enable';
46129                 }
46130                 
46131                 b.scope = editorcore;
46132                 tb.add(b);
46133             }
46134         
46135         }
46136         
46137         
46138         
46139         // disable everything...
46140         
46141         this.tb.items.each(function(item){
46142             
46143            if(
46144                 item.id != editorcore.frameId+ '-sourceedit' && 
46145                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46146             ){
46147                 
46148                 item.disable();
46149             }
46150         });
46151         this.rendered = true;
46152         
46153         // the all the btns;
46154         editor.on('editorevent', this.updateToolbar, this);
46155         // other toolbars need to implement this..
46156         //editor.on('editmodechange', this.updateToolbar, this);
46157     },
46158     
46159     
46160     relayBtnCmd : function(btn) {
46161         this.editorcore.relayCmd(btn.cmd);
46162     },
46163     // private used internally
46164     createLink : function(){
46165         Roo.log("create link?");
46166         var url = prompt(this.createLinkText, this.defaultLinkValue);
46167         if(url && url != 'http:/'+'/'){
46168             this.editorcore.relayCmd('createlink', url);
46169         }
46170     },
46171
46172     
46173     /**
46174      * Protected method that will not generally be called directly. It triggers
46175      * a toolbar update by reading the markup state of the current selection in the editor.
46176      */
46177     updateToolbar: function(){
46178
46179         if(!this.editorcore.activated){
46180             this.editor.onFirstFocus();
46181             return;
46182         }
46183
46184         var btns = this.tb.items.map, 
46185             doc = this.editorcore.doc,
46186             frameId = this.editorcore.frameId;
46187
46188         if(!this.disable.font && !Roo.isSafari){
46189             /*
46190             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46191             if(name != this.fontSelect.dom.value){
46192                 this.fontSelect.dom.value = name;
46193             }
46194             */
46195         }
46196         if(!this.disable.format){
46197             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46198             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46199             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46200             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46201         }
46202         if(!this.disable.alignments){
46203             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46204             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46205             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46206         }
46207         if(!Roo.isSafari && !this.disable.lists){
46208             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46209             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46210         }
46211         
46212         var ans = this.editorcore.getAllAncestors();
46213         if (this.formatCombo) {
46214             
46215             
46216             var store = this.formatCombo.store;
46217             this.formatCombo.setValue("");
46218             for (var i =0; i < ans.length;i++) {
46219                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46220                     // select it..
46221                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46222                     break;
46223                 }
46224             }
46225         }
46226         
46227         
46228         
46229         // hides menus... - so this cant be on a menu...
46230         Roo.menu.MenuMgr.hideAll();
46231
46232         //this.editorsyncValue();
46233     },
46234    
46235     
46236     createFontOptions : function(){
46237         var buf = [], fs = this.fontFamilies, ff, lc;
46238         
46239         
46240         
46241         for(var i = 0, len = fs.length; i< len; i++){
46242             ff = fs[i];
46243             lc = ff.toLowerCase();
46244             buf.push(
46245                 '<option value="',lc,'" style="font-family:',ff,';"',
46246                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46247                     ff,
46248                 '</option>'
46249             );
46250         }
46251         return buf.join('');
46252     },
46253     
46254     toggleSourceEdit : function(sourceEditMode){
46255         
46256         Roo.log("toolbar toogle");
46257         if(sourceEditMode === undefined){
46258             sourceEditMode = !this.sourceEditMode;
46259         }
46260         this.sourceEditMode = sourceEditMode === true;
46261         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46262         // just toggle the button?
46263         if(btn.pressed !== this.sourceEditMode){
46264             btn.toggle(this.sourceEditMode);
46265             return;
46266         }
46267         
46268         if(sourceEditMode){
46269             Roo.log("disabling buttons");
46270             this.tb.items.each(function(item){
46271                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46272                     item.disable();
46273                 }
46274             });
46275           
46276         }else{
46277             Roo.log("enabling buttons");
46278             if(this.editorcore.initialized){
46279                 this.tb.items.each(function(item){
46280                     item.enable();
46281                 });
46282             }
46283             
46284         }
46285         Roo.log("calling toggole on editor");
46286         // tell the editor that it's been pressed..
46287         this.editor.toggleSourceEdit(sourceEditMode);
46288        
46289     },
46290      /**
46291      * Object collection of toolbar tooltips for the buttons in the editor. The key
46292      * is the command id associated with that button and the value is a valid QuickTips object.
46293      * For example:
46294 <pre><code>
46295 {
46296     bold : {
46297         title: 'Bold (Ctrl+B)',
46298         text: 'Make the selected text bold.',
46299         cls: 'x-html-editor-tip'
46300     },
46301     italic : {
46302         title: 'Italic (Ctrl+I)',
46303         text: 'Make the selected text italic.',
46304         cls: 'x-html-editor-tip'
46305     },
46306     ...
46307 </code></pre>
46308     * @type Object
46309      */
46310     buttonTips : {
46311         bold : {
46312             title: 'Bold (Ctrl+B)',
46313             text: 'Make the selected text bold.',
46314             cls: 'x-html-editor-tip'
46315         },
46316         italic : {
46317             title: 'Italic (Ctrl+I)',
46318             text: 'Make the selected text italic.',
46319             cls: 'x-html-editor-tip'
46320         },
46321         underline : {
46322             title: 'Underline (Ctrl+U)',
46323             text: 'Underline the selected text.',
46324             cls: 'x-html-editor-tip'
46325         },
46326         strikethrough : {
46327             title: 'Strikethrough',
46328             text: 'Strikethrough the selected text.',
46329             cls: 'x-html-editor-tip'
46330         },
46331         increasefontsize : {
46332             title: 'Grow Text',
46333             text: 'Increase the font size.',
46334             cls: 'x-html-editor-tip'
46335         },
46336         decreasefontsize : {
46337             title: 'Shrink Text',
46338             text: 'Decrease the font size.',
46339             cls: 'x-html-editor-tip'
46340         },
46341         backcolor : {
46342             title: 'Text Highlight Color',
46343             text: 'Change the background color of the selected text.',
46344             cls: 'x-html-editor-tip'
46345         },
46346         forecolor : {
46347             title: 'Font Color',
46348             text: 'Change the color of the selected text.',
46349             cls: 'x-html-editor-tip'
46350         },
46351         justifyleft : {
46352             title: 'Align Text Left',
46353             text: 'Align text to the left.',
46354             cls: 'x-html-editor-tip'
46355         },
46356         justifycenter : {
46357             title: 'Center Text',
46358             text: 'Center text in the editor.',
46359             cls: 'x-html-editor-tip'
46360         },
46361         justifyright : {
46362             title: 'Align Text Right',
46363             text: 'Align text to the right.',
46364             cls: 'x-html-editor-tip'
46365         },
46366         insertunorderedlist : {
46367             title: 'Bullet List',
46368             text: 'Start a bulleted list.',
46369             cls: 'x-html-editor-tip'
46370         },
46371         insertorderedlist : {
46372             title: 'Numbered List',
46373             text: 'Start a numbered list.',
46374             cls: 'x-html-editor-tip'
46375         },
46376         createlink : {
46377             title: 'Hyperlink',
46378             text: 'Make the selected text a hyperlink.',
46379             cls: 'x-html-editor-tip'
46380         },
46381         sourceedit : {
46382             title: 'Source Edit',
46383             text: 'Switch to source editing mode.',
46384             cls: 'x-html-editor-tip'
46385         }
46386     },
46387     // private
46388     onDestroy : function(){
46389         if(this.rendered){
46390             
46391             this.tb.items.each(function(item){
46392                 if(item.menu){
46393                     item.menu.removeAll();
46394                     if(item.menu.el){
46395                         item.menu.el.destroy();
46396                     }
46397                 }
46398                 item.destroy();
46399             });
46400              
46401         }
46402     },
46403     onFirstFocus: function() {
46404         this.tb.items.each(function(item){
46405            item.enable();
46406         });
46407     }
46408 });
46409
46410
46411
46412
46413 // <script type="text/javascript">
46414 /*
46415  * Based on
46416  * Ext JS Library 1.1.1
46417  * Copyright(c) 2006-2007, Ext JS, LLC.
46418  *  
46419  
46420  */
46421
46422  
46423 /**
46424  * @class Roo.form.HtmlEditor.ToolbarContext
46425  * Context Toolbar
46426  * 
46427  * Usage:
46428  *
46429  new Roo.form.HtmlEditor({
46430     ....
46431     toolbars : [
46432         { xtype: 'ToolbarStandard', styles : {} }
46433         { xtype: 'ToolbarContext', disable : {} }
46434     ]
46435 })
46436
46437      
46438  * 
46439  * @config : {Object} disable List of elements to disable.. (not done yet.)
46440  * @config : {Object} styles  Map of styles available.
46441  * 
46442  */
46443
46444 Roo.form.HtmlEditor.ToolbarContext = function(config)
46445 {
46446     
46447     Roo.apply(this, config);
46448     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46449     // dont call parent... till later.
46450     this.styles = this.styles || {};
46451 }
46452
46453  
46454
46455 Roo.form.HtmlEditor.ToolbarContext.types = {
46456     'IMG' : {
46457         width : {
46458             title: "Width",
46459             width: 40
46460         },
46461         height:  {
46462             title: "Height",
46463             width: 40
46464         },
46465         align: {
46466             title: "Align",
46467             opts : [ [""],[ "left"],[ "right"],[ "center"],[ "top"]],
46468             width : 80
46469             
46470         },
46471         border: {
46472             title: "Border",
46473             width: 40
46474         },
46475         alt: {
46476             title: "Alt",
46477             width: 120
46478         },
46479         src : {
46480             title: "Src",
46481             width: 220
46482         }
46483         
46484     },
46485     'A' : {
46486         name : {
46487             title: "Name",
46488             width: 50
46489         },
46490         target:  {
46491             title: "Target",
46492             width: 120
46493         },
46494         href:  {
46495             title: "Href",
46496             width: 220
46497         } // border?
46498         
46499     },
46500     'TABLE' : {
46501         rows : {
46502             title: "Rows",
46503             width: 20
46504         },
46505         cols : {
46506             title: "Cols",
46507             width: 20
46508         },
46509         width : {
46510             title: "Width",
46511             width: 40
46512         },
46513         height : {
46514             title: "Height",
46515             width: 40
46516         },
46517         border : {
46518             title: "Border",
46519             width: 20
46520         }
46521     },
46522     'TD' : {
46523         width : {
46524             title: "Width",
46525             width: 40
46526         },
46527         height : {
46528             title: "Height",
46529             width: 40
46530         },   
46531         align: {
46532             title: "Align",
46533             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46534             width: 80
46535         },
46536         valign: {
46537             title: "Valign",
46538             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46539             width: 80
46540         },
46541         colspan: {
46542             title: "Colspan",
46543             width: 20
46544             
46545         },
46546          'font-family'  : {
46547             title : "Font",
46548             style : 'fontFamily',
46549             displayField: 'display',
46550             optname : 'font-family',
46551             width: 140
46552         }
46553     },
46554     'INPUT' : {
46555         name : {
46556             title: "name",
46557             width: 120
46558         },
46559         value : {
46560             title: "Value",
46561             width: 120
46562         },
46563         width : {
46564             title: "Width",
46565             width: 40
46566         }
46567     },
46568     'LABEL' : {
46569         'for' : {
46570             title: "For",
46571             width: 120
46572         }
46573     },
46574     'TEXTAREA' : {
46575           name : {
46576             title: "name",
46577             width: 120
46578         },
46579         rows : {
46580             title: "Rows",
46581             width: 20
46582         },
46583         cols : {
46584             title: "Cols",
46585             width: 20
46586         }
46587     },
46588     'SELECT' : {
46589         name : {
46590             title: "name",
46591             width: 120
46592         },
46593         selectoptions : {
46594             title: "Options",
46595             width: 200
46596         }
46597     },
46598     
46599     // should we really allow this??
46600     // should this just be 
46601     'BODY' : {
46602         title : {
46603             title: "Title",
46604             width: 200,
46605             disabled : true
46606         }
46607     },
46608     'SPAN' : {
46609         'font-family'  : {
46610             title : "Font",
46611             style : 'fontFamily',
46612             displayField: 'display',
46613             optname : 'font-family',
46614             width: 140
46615         }
46616     },
46617     'DIV' : {
46618         'font-family'  : {
46619             title : "Font",
46620             style : 'fontFamily',
46621             displayField: 'display',
46622             optname : 'font-family',
46623             width: 140
46624         }
46625     },
46626      'P' : {
46627         'font-family'  : {
46628             title : "Font",
46629             style : 'fontFamily',
46630             displayField: 'display',
46631             optname : 'font-family',
46632             width: 140
46633         }
46634     },
46635     
46636     '*' : {
46637         // empty..
46638     }
46639
46640 };
46641
46642 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46643 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46644
46645 Roo.form.HtmlEditor.ToolbarContext.options = {
46646         'font-family'  : [ 
46647                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46648                 [ 'Courier New', 'Courier New'],
46649                 [ 'Tahoma', 'Tahoma'],
46650                 [ 'Times New Roman,serif', 'Times'],
46651                 [ 'Verdana','Verdana' ]
46652         ]
46653 };
46654
46655 // fixme - these need to be configurable..
46656  
46657
46658 //Roo.form.HtmlEditor.ToolbarContext.types
46659
46660
46661 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46662     
46663     tb: false,
46664     
46665     rendered: false,
46666     
46667     editor : false,
46668     editorcore : false,
46669     /**
46670      * @cfg {Object} disable  List of toolbar elements to disable
46671          
46672      */
46673     disable : false,
46674     /**
46675      * @cfg {Object} styles List of styles 
46676      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46677      *
46678      * These must be defined in the page, so they get rendered correctly..
46679      * .headline { }
46680      * TD.underline { }
46681      * 
46682      */
46683     styles : false,
46684     
46685     options: false,
46686     
46687     toolbars : false,
46688     
46689     init : function(editor)
46690     {
46691         this.editor = editor;
46692         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46693         var editorcore = this.editorcore;
46694         
46695         var fid = editorcore.frameId;
46696         var etb = this;
46697         function btn(id, toggle, handler){
46698             var xid = fid + '-'+ id ;
46699             return {
46700                 id : xid,
46701                 cmd : id,
46702                 cls : 'x-btn-icon x-edit-'+id,
46703                 enableToggle:toggle !== false,
46704                 scope: editorcore, // was editor...
46705                 handler:handler||editorcore.relayBtnCmd,
46706                 clickEvent:'mousedown',
46707                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46708                 tabIndex:-1
46709             };
46710         }
46711         // create a new element.
46712         var wdiv = editor.wrap.createChild({
46713                 tag: 'div'
46714             }, editor.wrap.dom.firstChild.nextSibling, true);
46715         
46716         // can we do this more than once??
46717         
46718          // stop form submits
46719       
46720  
46721         // disable everything...
46722         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46723         this.toolbars = {};
46724            
46725         for (var i in  ty) {
46726           
46727             this.toolbars[i] = this.buildToolbar(ty[i],i);
46728         }
46729         this.tb = this.toolbars.BODY;
46730         this.tb.el.show();
46731         this.buildFooter();
46732         this.footer.show();
46733         editor.on('hide', function( ) { this.footer.hide() }, this);
46734         editor.on('show', function( ) { this.footer.show() }, this);
46735         
46736          
46737         this.rendered = true;
46738         
46739         // the all the btns;
46740         editor.on('editorevent', this.updateToolbar, this);
46741         // other toolbars need to implement this..
46742         //editor.on('editmodechange', this.updateToolbar, this);
46743     },
46744     
46745     
46746     
46747     /**
46748      * Protected method that will not generally be called directly. It triggers
46749      * a toolbar update by reading the markup state of the current selection in the editor.
46750      *
46751      * Note you can force an update by calling on('editorevent', scope, false)
46752      */
46753     updateToolbar: function(editor,ev,sel){
46754
46755         //Roo.log(ev);
46756         // capture mouse up - this is handy for selecting images..
46757         // perhaps should go somewhere else...
46758         if(!this.editorcore.activated){
46759              this.editor.onFirstFocus();
46760             return;
46761         }
46762         
46763         
46764         
46765         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46766         // selectNode - might want to handle IE?
46767         if (ev &&
46768             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46769             ev.target && ev.target.tagName == 'IMG') {
46770             // they have click on an image...
46771             // let's see if we can change the selection...
46772             sel = ev.target;
46773          
46774               var nodeRange = sel.ownerDocument.createRange();
46775             try {
46776                 nodeRange.selectNode(sel);
46777             } catch (e) {
46778                 nodeRange.selectNodeContents(sel);
46779             }
46780             //nodeRange.collapse(true);
46781             var s = this.editorcore.win.getSelection();
46782             s.removeAllRanges();
46783             s.addRange(nodeRange);
46784         }  
46785         
46786       
46787         var updateFooter = sel ? false : true;
46788         
46789         
46790         var ans = this.editorcore.getAllAncestors();
46791         
46792         // pick
46793         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46794         
46795         if (!sel) { 
46796             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46797             sel = sel ? sel : this.editorcore.doc.body;
46798             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46799             
46800         }
46801         // pick a menu that exists..
46802         var tn = sel.tagName.toUpperCase();
46803         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46804         
46805         tn = sel.tagName.toUpperCase();
46806         
46807         var lastSel = this.tb.selectedNode;
46808         
46809         this.tb.selectedNode = sel;
46810         
46811         // if current menu does not match..
46812         
46813         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46814                 
46815             this.tb.el.hide();
46816             ///console.log("show: " + tn);
46817             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46818             this.tb.el.show();
46819             // update name
46820             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46821             
46822             
46823             // update attributes
46824             if (this.tb.fields) {
46825                 this.tb.fields.each(function(e) {
46826                     if (e.stylename) {
46827                         e.setValue(sel.style[e.stylename]);
46828                         return;
46829                     } 
46830                    e.setValue(sel.getAttribute(e.attrname));
46831                 });
46832             }
46833             
46834             var hasStyles = false;
46835             for(var i in this.styles) {
46836                 hasStyles = true;
46837                 break;
46838             }
46839             
46840             // update styles
46841             if (hasStyles) { 
46842                 var st = this.tb.fields.item(0);
46843                 
46844                 st.store.removeAll();
46845                
46846                 
46847                 var cn = sel.className.split(/\s+/);
46848                 
46849                 var avs = [];
46850                 if (this.styles['*']) {
46851                     
46852                     Roo.each(this.styles['*'], function(v) {
46853                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46854                     });
46855                 }
46856                 if (this.styles[tn]) { 
46857                     Roo.each(this.styles[tn], function(v) {
46858                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46859                     });
46860                 }
46861                 
46862                 st.store.loadData(avs);
46863                 st.collapse();
46864                 st.setValue(cn);
46865             }
46866             // flag our selected Node.
46867             this.tb.selectedNode = sel;
46868            
46869            
46870             Roo.menu.MenuMgr.hideAll();
46871
46872         }
46873         
46874         if (!updateFooter) {
46875             //this.footDisp.dom.innerHTML = ''; 
46876             return;
46877         }
46878         // update the footer
46879         //
46880         var html = '';
46881         
46882         this.footerEls = ans.reverse();
46883         Roo.each(this.footerEls, function(a,i) {
46884             if (!a) { return; }
46885             html += html.length ? ' &gt; '  :  '';
46886             
46887             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46888             
46889         });
46890        
46891         // 
46892         var sz = this.footDisp.up('td').getSize();
46893         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46894         this.footDisp.dom.style.marginLeft = '5px';
46895         
46896         this.footDisp.dom.style.overflow = 'hidden';
46897         
46898         this.footDisp.dom.innerHTML = html;
46899             
46900         //this.editorsyncValue();
46901     },
46902      
46903     
46904    
46905        
46906     // private
46907     onDestroy : function(){
46908         if(this.rendered){
46909             
46910             this.tb.items.each(function(item){
46911                 if(item.menu){
46912                     item.menu.removeAll();
46913                     if(item.menu.el){
46914                         item.menu.el.destroy();
46915                     }
46916                 }
46917                 item.destroy();
46918             });
46919              
46920         }
46921     },
46922     onFirstFocus: function() {
46923         // need to do this for all the toolbars..
46924         this.tb.items.each(function(item){
46925            item.enable();
46926         });
46927     },
46928     buildToolbar: function(tlist, nm)
46929     {
46930         var editor = this.editor;
46931         var editorcore = this.editorcore;
46932          // create a new element.
46933         var wdiv = editor.wrap.createChild({
46934                 tag: 'div'
46935             }, editor.wrap.dom.firstChild.nextSibling, true);
46936         
46937        
46938         var tb = new Roo.Toolbar(wdiv);
46939         // add the name..
46940         
46941         tb.add(nm+ ":&nbsp;");
46942         
46943         var styles = [];
46944         for(var i in this.styles) {
46945             styles.push(i);
46946         }
46947         
46948         // styles...
46949         if (styles && styles.length) {
46950             
46951             // this needs a multi-select checkbox...
46952             tb.addField( new Roo.form.ComboBox({
46953                 store: new Roo.data.SimpleStore({
46954                     id : 'val',
46955                     fields: ['val', 'selected'],
46956                     data : [] 
46957                 }),
46958                 name : '-roo-edit-className',
46959                 attrname : 'className',
46960                 displayField: 'val',
46961                 typeAhead: false,
46962                 mode: 'local',
46963                 editable : false,
46964                 triggerAction: 'all',
46965                 emptyText:'Select Style',
46966                 selectOnFocus:true,
46967                 width: 130,
46968                 listeners : {
46969                     'select': function(c, r, i) {
46970                         // initial support only for on class per el..
46971                         tb.selectedNode.className =  r ? r.get('val') : '';
46972                         editorcore.syncValue();
46973                     }
46974                 }
46975     
46976             }));
46977         }
46978         
46979         var tbc = Roo.form.HtmlEditor.ToolbarContext;
46980         var tbops = tbc.options;
46981         
46982         for (var i in tlist) {
46983             
46984             var item = tlist[i];
46985             tb.add(item.title + ":&nbsp;");
46986             
46987             
46988             //optname == used so you can configure the options available..
46989             var opts = item.opts ? item.opts : false;
46990             if (item.optname) {
46991                 opts = tbops[item.optname];
46992            
46993             }
46994             
46995             if (opts) {
46996                 // opts == pulldown..
46997                 tb.addField( new Roo.form.ComboBox({
46998                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
46999                         id : 'val',
47000                         fields: ['val', 'display'],
47001                         data : opts  
47002                     }),
47003                     name : '-roo-edit-' + i,
47004                     attrname : i,
47005                     stylename : item.style ? item.style : false,
47006                     displayField: item.displayField ? item.displayField : 'val',
47007                     valueField :  'val',
47008                     typeAhead: false,
47009                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47010                     editable : false,
47011                     triggerAction: 'all',
47012                     emptyText:'Select',
47013                     selectOnFocus:true,
47014                     width: item.width ? item.width  : 130,
47015                     listeners : {
47016                         'select': function(c, r, i) {
47017                             if (c.stylename) {
47018                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47019                                 return;
47020                             }
47021                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47022                         }
47023                     }
47024
47025                 }));
47026                 continue;
47027                     
47028                  
47029                 
47030                 tb.addField( new Roo.form.TextField({
47031                     name: i,
47032                     width: 100,
47033                     //allowBlank:false,
47034                     value: ''
47035                 }));
47036                 continue;
47037             }
47038             tb.addField( new Roo.form.TextField({
47039                 name: '-roo-edit-' + i,
47040                 attrname : i,
47041                 
47042                 width: item.width,
47043                 //allowBlank:true,
47044                 value: '',
47045                 listeners: {
47046                     'change' : function(f, nv, ov) {
47047                         tb.selectedNode.setAttribute(f.attrname, nv);
47048                         editorcore.syncValue();
47049                     }
47050                 }
47051             }));
47052              
47053         }
47054         
47055         var _this = this;
47056         
47057         if(nm == 'BODY'){
47058             tb.addSeparator();
47059         
47060             tb.addButton( {
47061                 text: 'Stylesheets',
47062
47063                 listeners : {
47064                     click : function ()
47065                     {
47066                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47067                     }
47068                 }
47069             });
47070         }
47071         
47072         tb.addFill();
47073         tb.addButton( {
47074             text: 'Remove Tag',
47075     
47076             listeners : {
47077                 click : function ()
47078                 {
47079                     // remove
47080                     // undo does not work.
47081                      
47082                     var sn = tb.selectedNode;
47083                     
47084                     var pn = sn.parentNode;
47085                     
47086                     var stn =  sn.childNodes[0];
47087                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47088                     while (sn.childNodes.length) {
47089                         var node = sn.childNodes[0];
47090                         sn.removeChild(node);
47091                         //Roo.log(node);
47092                         pn.insertBefore(node, sn);
47093                         
47094                     }
47095                     pn.removeChild(sn);
47096                     var range = editorcore.createRange();
47097         
47098                     range.setStart(stn,0);
47099                     range.setEnd(en,0); //????
47100                     //range.selectNode(sel);
47101                     
47102                     
47103                     var selection = editorcore.getSelection();
47104                     selection.removeAllRanges();
47105                     selection.addRange(range);
47106                     
47107                     
47108                     
47109                     //_this.updateToolbar(null, null, pn);
47110                     _this.updateToolbar(null, null, null);
47111                     _this.footDisp.dom.innerHTML = ''; 
47112                 }
47113             }
47114             
47115                     
47116                 
47117             
47118         });
47119         
47120         
47121         tb.el.on('click', function(e){
47122             e.preventDefault(); // what does this do?
47123         });
47124         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47125         tb.el.hide();
47126         tb.name = nm;
47127         // dont need to disable them... as they will get hidden
47128         return tb;
47129          
47130         
47131     },
47132     buildFooter : function()
47133     {
47134         
47135         var fel = this.editor.wrap.createChild();
47136         this.footer = new Roo.Toolbar(fel);
47137         // toolbar has scrolly on left / right?
47138         var footDisp= new Roo.Toolbar.Fill();
47139         var _t = this;
47140         this.footer.add(
47141             {
47142                 text : '&lt;',
47143                 xtype: 'Button',
47144                 handler : function() {
47145                     _t.footDisp.scrollTo('left',0,true)
47146                 }
47147             }
47148         );
47149         this.footer.add( footDisp );
47150         this.footer.add( 
47151             {
47152                 text : '&gt;',
47153                 xtype: 'Button',
47154                 handler : function() {
47155                     // no animation..
47156                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47157                 }
47158             }
47159         );
47160         var fel = Roo.get(footDisp.el);
47161         fel.addClass('x-editor-context');
47162         this.footDispWrap = fel; 
47163         this.footDispWrap.overflow  = 'hidden';
47164         
47165         this.footDisp = fel.createChild();
47166         this.footDispWrap.on('click', this.onContextClick, this)
47167         
47168         
47169     },
47170     onContextClick : function (ev,dom)
47171     {
47172         ev.preventDefault();
47173         var  cn = dom.className;
47174         //Roo.log(cn);
47175         if (!cn.match(/x-ed-loc-/)) {
47176             return;
47177         }
47178         var n = cn.split('-').pop();
47179         var ans = this.footerEls;
47180         var sel = ans[n];
47181         
47182          // pick
47183         var range = this.editorcore.createRange();
47184         
47185         range.selectNodeContents(sel);
47186         //range.selectNode(sel);
47187         
47188         
47189         var selection = this.editorcore.getSelection();
47190         selection.removeAllRanges();
47191         selection.addRange(range);
47192         
47193         
47194         
47195         this.updateToolbar(null, null, sel);
47196         
47197         
47198     }
47199     
47200     
47201     
47202     
47203     
47204 });
47205
47206
47207
47208
47209
47210 /*
47211  * Based on:
47212  * Ext JS Library 1.1.1
47213  * Copyright(c) 2006-2007, Ext JS, LLC.
47214  *
47215  * Originally Released Under LGPL - original licence link has changed is not relivant.
47216  *
47217  * Fork - LGPL
47218  * <script type="text/javascript">
47219  */
47220  
47221 /**
47222  * @class Roo.form.BasicForm
47223  * @extends Roo.util.Observable
47224  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47225  * @constructor
47226  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47227  * @param {Object} config Configuration options
47228  */
47229 Roo.form.BasicForm = function(el, config){
47230     this.allItems = [];
47231     this.childForms = [];
47232     Roo.apply(this, config);
47233     /*
47234      * The Roo.form.Field items in this form.
47235      * @type MixedCollection
47236      */
47237      
47238      
47239     this.items = new Roo.util.MixedCollection(false, function(o){
47240         return o.id || (o.id = Roo.id());
47241     });
47242     this.addEvents({
47243         /**
47244          * @event beforeaction
47245          * Fires before any action is performed. Return false to cancel the action.
47246          * @param {Form} this
47247          * @param {Action} action The action to be performed
47248          */
47249         beforeaction: true,
47250         /**
47251          * @event actionfailed
47252          * Fires when an action fails.
47253          * @param {Form} this
47254          * @param {Action} action The action that failed
47255          */
47256         actionfailed : true,
47257         /**
47258          * @event actioncomplete
47259          * Fires when an action is completed.
47260          * @param {Form} this
47261          * @param {Action} action The action that completed
47262          */
47263         actioncomplete : true
47264     });
47265     if(el){
47266         this.initEl(el);
47267     }
47268     Roo.form.BasicForm.superclass.constructor.call(this);
47269     
47270     Roo.form.BasicForm.popover.apply();
47271 };
47272
47273 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47274     /**
47275      * @cfg {String} method
47276      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47277      */
47278     /**
47279      * @cfg {DataReader} reader
47280      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47281      * This is optional as there is built-in support for processing JSON.
47282      */
47283     /**
47284      * @cfg {DataReader} errorReader
47285      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47286      * This is completely optional as there is built-in support for processing JSON.
47287      */
47288     /**
47289      * @cfg {String} url
47290      * The URL to use for form actions if one isn't supplied in the action options.
47291      */
47292     /**
47293      * @cfg {Boolean} fileUpload
47294      * Set to true if this form is a file upload.
47295      */
47296      
47297     /**
47298      * @cfg {Object} baseParams
47299      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47300      */
47301      /**
47302      
47303     /**
47304      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47305      */
47306     timeout: 30,
47307
47308     // private
47309     activeAction : null,
47310
47311     /**
47312      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47313      * or setValues() data instead of when the form was first created.
47314      */
47315     trackResetOnLoad : false,
47316     
47317     
47318     /**
47319      * childForms - used for multi-tab forms
47320      * @type {Array}
47321      */
47322     childForms : false,
47323     
47324     /**
47325      * allItems - full list of fields.
47326      * @type {Array}
47327      */
47328     allItems : false,
47329     
47330     /**
47331      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47332      * element by passing it or its id or mask the form itself by passing in true.
47333      * @type Mixed
47334      */
47335     waitMsgTarget : false,
47336     
47337     /**
47338      * @type Boolean
47339      */
47340     disableMask : false,
47341     
47342     /**
47343      * @cfg {Boolean} errorMask (true|false) default false
47344      */
47345     errorMask : false,
47346     
47347     /**
47348      * @cfg {Number} maskOffset Default 100
47349      */
47350     maskOffset : 100,
47351
47352     // private
47353     initEl : function(el){
47354         this.el = Roo.get(el);
47355         this.id = this.el.id || Roo.id();
47356         this.el.on('submit', this.onSubmit, this);
47357         this.el.addClass('x-form');
47358     },
47359
47360     // private
47361     onSubmit : function(e){
47362         e.stopEvent();
47363     },
47364
47365     /**
47366      * Returns true if client-side validation on the form is successful.
47367      * @return Boolean
47368      */
47369     isValid : function(){
47370         var valid = true;
47371         var target = false;
47372         this.items.each(function(f){
47373             if(f.validate()){
47374                 return;
47375             }
47376             
47377             valid = false;
47378                 
47379             if(!target && f.el.isVisible(true)){
47380                 target = f;
47381             }
47382         });
47383         
47384         if(this.errorMask && !valid){
47385             Roo.form.BasicForm.popover.mask(this, target);
47386         }
47387         
47388         return valid;
47389     },
47390
47391     /**
47392      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47393      * @return Boolean
47394      */
47395     isDirty : function(){
47396         var dirty = false;
47397         this.items.each(function(f){
47398            if(f.isDirty()){
47399                dirty = true;
47400                return false;
47401            }
47402         });
47403         return dirty;
47404     },
47405     
47406     /**
47407      * Returns true if any fields in this form have changed since their original load. (New version)
47408      * @return Boolean
47409      */
47410     
47411     hasChanged : function()
47412     {
47413         var dirty = false;
47414         this.items.each(function(f){
47415            if(f.hasChanged()){
47416                dirty = true;
47417                return false;
47418            }
47419         });
47420         return dirty;
47421         
47422     },
47423     /**
47424      * Resets all hasChanged to 'false' -
47425      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47426      * So hasChanged storage is only to be used for this purpose
47427      * @return Boolean
47428      */
47429     resetHasChanged : function()
47430     {
47431         this.items.each(function(f){
47432            f.resetHasChanged();
47433         });
47434         
47435     },
47436     
47437     
47438     /**
47439      * Performs a predefined action (submit or load) or custom actions you define on this form.
47440      * @param {String} actionName The name of the action type
47441      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47442      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47443      * accept other config options):
47444      * <pre>
47445 Property          Type             Description
47446 ----------------  ---------------  ----------------------------------------------------------------------------------
47447 url               String           The url for the action (defaults to the form's url)
47448 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47449 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47450 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47451                                    validate the form on the client (defaults to false)
47452      * </pre>
47453      * @return {BasicForm} this
47454      */
47455     doAction : function(action, options){
47456         if(typeof action == 'string'){
47457             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47458         }
47459         if(this.fireEvent('beforeaction', this, action) !== false){
47460             this.beforeAction(action);
47461             action.run.defer(100, action);
47462         }
47463         return this;
47464     },
47465
47466     /**
47467      * Shortcut to do a submit action.
47468      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47469      * @return {BasicForm} this
47470      */
47471     submit : function(options){
47472         this.doAction('submit', options);
47473         return this;
47474     },
47475
47476     /**
47477      * Shortcut to do a load action.
47478      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47479      * @return {BasicForm} this
47480      */
47481     load : function(options){
47482         this.doAction('load', options);
47483         return this;
47484     },
47485
47486     /**
47487      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47488      * @param {Record} record The record to edit
47489      * @return {BasicForm} this
47490      */
47491     updateRecord : function(record){
47492         record.beginEdit();
47493         var fs = record.fields;
47494         fs.each(function(f){
47495             var field = this.findField(f.name);
47496             if(field){
47497                 record.set(f.name, field.getValue());
47498             }
47499         }, this);
47500         record.endEdit();
47501         return this;
47502     },
47503
47504     /**
47505      * Loads an Roo.data.Record into this form.
47506      * @param {Record} record The record to load
47507      * @return {BasicForm} this
47508      */
47509     loadRecord : function(record){
47510         this.setValues(record.data);
47511         return this;
47512     },
47513
47514     // private
47515     beforeAction : function(action){
47516         var o = action.options;
47517         
47518         if(!this.disableMask) {
47519             if(this.waitMsgTarget === true){
47520                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47521             }else if(this.waitMsgTarget){
47522                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47523                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47524             }else {
47525                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47526             }
47527         }
47528         
47529          
47530     },
47531
47532     // private
47533     afterAction : function(action, success){
47534         this.activeAction = null;
47535         var o = action.options;
47536         
47537         if(!this.disableMask) {
47538             if(this.waitMsgTarget === true){
47539                 this.el.unmask();
47540             }else if(this.waitMsgTarget){
47541                 this.waitMsgTarget.unmask();
47542             }else{
47543                 Roo.MessageBox.updateProgress(1);
47544                 Roo.MessageBox.hide();
47545             }
47546         }
47547         
47548         if(success){
47549             if(o.reset){
47550                 this.reset();
47551             }
47552             Roo.callback(o.success, o.scope, [this, action]);
47553             this.fireEvent('actioncomplete', this, action);
47554             
47555         }else{
47556             
47557             // failure condition..
47558             // we have a scenario where updates need confirming.
47559             // eg. if a locking scenario exists..
47560             // we look for { errors : { needs_confirm : true }} in the response.
47561             if (
47562                 (typeof(action.result) != 'undefined')  &&
47563                 (typeof(action.result.errors) != 'undefined')  &&
47564                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47565            ){
47566                 var _t = this;
47567                 Roo.MessageBox.confirm(
47568                     "Change requires confirmation",
47569                     action.result.errorMsg,
47570                     function(r) {
47571                         if (r != 'yes') {
47572                             return;
47573                         }
47574                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47575                     }
47576                     
47577                 );
47578                 
47579                 
47580                 
47581                 return;
47582             }
47583             
47584             Roo.callback(o.failure, o.scope, [this, action]);
47585             // show an error message if no failed handler is set..
47586             if (!this.hasListener('actionfailed')) {
47587                 Roo.MessageBox.alert("Error",
47588                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47589                         action.result.errorMsg :
47590                         "Saving Failed, please check your entries or try again"
47591                 );
47592             }
47593             
47594             this.fireEvent('actionfailed', this, action);
47595         }
47596         
47597     },
47598
47599     /**
47600      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47601      * @param {String} id The value to search for
47602      * @return Field
47603      */
47604     findField : function(id){
47605         var field = this.items.get(id);
47606         if(!field){
47607             this.items.each(function(f){
47608                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47609                     field = f;
47610                     return false;
47611                 }
47612             });
47613         }
47614         return field || null;
47615     },
47616
47617     /**
47618      * Add a secondary form to this one, 
47619      * Used to provide tabbed forms. One form is primary, with hidden values 
47620      * which mirror the elements from the other forms.
47621      * 
47622      * @param {Roo.form.Form} form to add.
47623      * 
47624      */
47625     addForm : function(form)
47626     {
47627        
47628         if (this.childForms.indexOf(form) > -1) {
47629             // already added..
47630             return;
47631         }
47632         this.childForms.push(form);
47633         var n = '';
47634         Roo.each(form.allItems, function (fe) {
47635             
47636             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47637             if (this.findField(n)) { // already added..
47638                 return;
47639             }
47640             var add = new Roo.form.Hidden({
47641                 name : n
47642             });
47643             add.render(this.el);
47644             
47645             this.add( add );
47646         }, this);
47647         
47648     },
47649     /**
47650      * Mark fields in this form invalid in bulk.
47651      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47652      * @return {BasicForm} this
47653      */
47654     markInvalid : function(errors){
47655         if(errors instanceof Array){
47656             for(var i = 0, len = errors.length; i < len; i++){
47657                 var fieldError = errors[i];
47658                 var f = this.findField(fieldError.id);
47659                 if(f){
47660                     f.markInvalid(fieldError.msg);
47661                 }
47662             }
47663         }else{
47664             var field, id;
47665             for(id in errors){
47666                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47667                     field.markInvalid(errors[id]);
47668                 }
47669             }
47670         }
47671         Roo.each(this.childForms || [], function (f) {
47672             f.markInvalid(errors);
47673         });
47674         
47675         return this;
47676     },
47677
47678     /**
47679      * Set values for fields in this form in bulk.
47680      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47681      * @return {BasicForm} this
47682      */
47683     setValues : function(values){
47684         if(values instanceof Array){ // array of objects
47685             for(var i = 0, len = values.length; i < len; i++){
47686                 var v = values[i];
47687                 var f = this.findField(v.id);
47688                 if(f){
47689                     f.setValue(v.value);
47690                     if(this.trackResetOnLoad){
47691                         f.originalValue = f.getValue();
47692                     }
47693                 }
47694             }
47695         }else{ // object hash
47696             var field, id;
47697             for(id in values){
47698                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47699                     
47700                     if (field.setFromData && 
47701                         field.valueField && 
47702                         field.displayField &&
47703                         // combos' with local stores can 
47704                         // be queried via setValue()
47705                         // to set their value..
47706                         (field.store && !field.store.isLocal)
47707                         ) {
47708                         // it's a combo
47709                         var sd = { };
47710                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47711                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47712                         field.setFromData(sd);
47713                         
47714                     } else {
47715                         field.setValue(values[id]);
47716                     }
47717                     
47718                     
47719                     if(this.trackResetOnLoad){
47720                         field.originalValue = field.getValue();
47721                     }
47722                 }
47723             }
47724         }
47725         this.resetHasChanged();
47726         
47727         
47728         Roo.each(this.childForms || [], function (f) {
47729             f.setValues(values);
47730             f.resetHasChanged();
47731         });
47732                 
47733         return this;
47734     },
47735  
47736     /**
47737      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47738      * they are returned as an array.
47739      * @param {Boolean} asString
47740      * @return {Object}
47741      */
47742     getValues : function(asString){
47743         if (this.childForms) {
47744             // copy values from the child forms
47745             Roo.each(this.childForms, function (f) {
47746                 this.setValues(f.getValues());
47747             }, this);
47748         }
47749         
47750         // use formdata
47751         if (typeof(FormData) != 'undefined' && asString !== true) {
47752             var fd = (new FormData(this.el.dom)).entries();
47753             var ret = {};
47754             var ent = fd.next();
47755             while (!ent.done) {
47756                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47757                 ent = fd.next();
47758             };
47759             return ret;
47760         }
47761         
47762         
47763         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47764         if(asString === true){
47765             return fs;
47766         }
47767         return Roo.urlDecode(fs);
47768     },
47769     
47770     /**
47771      * Returns the fields in this form as an object with key/value pairs. 
47772      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47773      * @return {Object}
47774      */
47775     getFieldValues : function(with_hidden)
47776     {
47777         if (this.childForms) {
47778             // copy values from the child forms
47779             // should this call getFieldValues - probably not as we do not currently copy
47780             // hidden fields when we generate..
47781             Roo.each(this.childForms, function (f) {
47782                 this.setValues(f.getValues());
47783             }, this);
47784         }
47785         
47786         var ret = {};
47787         this.items.each(function(f){
47788             if (!f.getName()) {
47789                 return;
47790             }
47791             var v = f.getValue();
47792             if (f.inputType =='radio') {
47793                 if (typeof(ret[f.getName()]) == 'undefined') {
47794                     ret[f.getName()] = ''; // empty..
47795                 }
47796                 
47797                 if (!f.el.dom.checked) {
47798                     return;
47799                     
47800                 }
47801                 v = f.el.dom.value;
47802                 
47803             }
47804             
47805             // not sure if this supported any more..
47806             if ((typeof(v) == 'object') && f.getRawValue) {
47807                 v = f.getRawValue() ; // dates..
47808             }
47809             // combo boxes where name != hiddenName...
47810             if (f.name != f.getName()) {
47811                 ret[f.name] = f.getRawValue();
47812             }
47813             ret[f.getName()] = v;
47814         });
47815         
47816         return ret;
47817     },
47818
47819     /**
47820      * Clears all invalid messages in this form.
47821      * @return {BasicForm} this
47822      */
47823     clearInvalid : function(){
47824         this.items.each(function(f){
47825            f.clearInvalid();
47826         });
47827         
47828         Roo.each(this.childForms || [], function (f) {
47829             f.clearInvalid();
47830         });
47831         
47832         
47833         return this;
47834     },
47835
47836     /**
47837      * Resets this form.
47838      * @return {BasicForm} this
47839      */
47840     reset : function(){
47841         this.items.each(function(f){
47842             f.reset();
47843         });
47844         
47845         Roo.each(this.childForms || [], function (f) {
47846             f.reset();
47847         });
47848         this.resetHasChanged();
47849         
47850         return this;
47851     },
47852
47853     /**
47854      * Add Roo.form components to this form.
47855      * @param {Field} field1
47856      * @param {Field} field2 (optional)
47857      * @param {Field} etc (optional)
47858      * @return {BasicForm} this
47859      */
47860     add : function(){
47861         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47862         return this;
47863     },
47864
47865
47866     /**
47867      * Removes a field from the items collection (does NOT remove its markup).
47868      * @param {Field} field
47869      * @return {BasicForm} this
47870      */
47871     remove : function(field){
47872         this.items.remove(field);
47873         return this;
47874     },
47875
47876     /**
47877      * Looks at the fields in this form, checks them for an id attribute,
47878      * and calls applyTo on the existing dom element with that id.
47879      * @return {BasicForm} this
47880      */
47881     render : function(){
47882         this.items.each(function(f){
47883             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47884                 f.applyTo(f.id);
47885             }
47886         });
47887         return this;
47888     },
47889
47890     /**
47891      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47892      * @param {Object} values
47893      * @return {BasicForm} this
47894      */
47895     applyToFields : function(o){
47896         this.items.each(function(f){
47897            Roo.apply(f, o);
47898         });
47899         return this;
47900     },
47901
47902     /**
47903      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47904      * @param {Object} values
47905      * @return {BasicForm} this
47906      */
47907     applyIfToFields : function(o){
47908         this.items.each(function(f){
47909            Roo.applyIf(f, o);
47910         });
47911         return this;
47912     }
47913 });
47914
47915 // back compat
47916 Roo.BasicForm = Roo.form.BasicForm;
47917
47918 Roo.apply(Roo.form.BasicForm, {
47919     
47920     popover : {
47921         
47922         padding : 5,
47923         
47924         isApplied : false,
47925         
47926         isMasked : false,
47927         
47928         form : false,
47929         
47930         target : false,
47931         
47932         intervalID : false,
47933         
47934         maskEl : false,
47935         
47936         apply : function()
47937         {
47938             if(this.isApplied){
47939                 return;
47940             }
47941             
47942             this.maskEl = {
47943                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
47944                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
47945                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
47946                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
47947             };
47948             
47949             this.maskEl.top.enableDisplayMode("block");
47950             this.maskEl.left.enableDisplayMode("block");
47951             this.maskEl.bottom.enableDisplayMode("block");
47952             this.maskEl.right.enableDisplayMode("block");
47953             
47954             Roo.get(document.body).on('click', function(){
47955                 this.unmask();
47956             }, this);
47957             
47958             Roo.get(document.body).on('touchstart', function(){
47959                 this.unmask();
47960             }, this);
47961             
47962             this.isApplied = true
47963         },
47964         
47965         mask : function(form, target)
47966         {
47967             this.form = form;
47968             
47969             this.target = target;
47970             
47971             if(!this.form.errorMask || !target.el){
47972                 return;
47973             }
47974             
47975             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
47976             
47977             var ot = this.target.el.calcOffsetsTo(scrollable);
47978             
47979             var scrollTo = ot[1] - this.form.maskOffset;
47980             
47981             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
47982             
47983             scrollable.scrollTo('top', scrollTo);
47984             
47985             var el = this.target.wrap || this.target.el;
47986             
47987             var box = el.getBox();
47988             
47989             this.maskEl.top.setStyle('position', 'absolute');
47990             this.maskEl.top.setStyle('z-index', 10000);
47991             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
47992             this.maskEl.top.setLeft(0);
47993             this.maskEl.top.setTop(0);
47994             this.maskEl.top.show();
47995             
47996             this.maskEl.left.setStyle('position', 'absolute');
47997             this.maskEl.left.setStyle('z-index', 10000);
47998             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
47999             this.maskEl.left.setLeft(0);
48000             this.maskEl.left.setTop(box.y - this.padding);
48001             this.maskEl.left.show();
48002
48003             this.maskEl.bottom.setStyle('position', 'absolute');
48004             this.maskEl.bottom.setStyle('z-index', 10000);
48005             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48006             this.maskEl.bottom.setLeft(0);
48007             this.maskEl.bottom.setTop(box.bottom + this.padding);
48008             this.maskEl.bottom.show();
48009
48010             this.maskEl.right.setStyle('position', 'absolute');
48011             this.maskEl.right.setStyle('z-index', 10000);
48012             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48013             this.maskEl.right.setLeft(box.right + this.padding);
48014             this.maskEl.right.setTop(box.y - this.padding);
48015             this.maskEl.right.show();
48016
48017             this.intervalID = window.setInterval(function() {
48018                 Roo.form.BasicForm.popover.unmask();
48019             }, 10000);
48020
48021             window.onwheel = function(){ return false;};
48022             
48023             (function(){ this.isMasked = true; }).defer(500, this);
48024             
48025         },
48026         
48027         unmask : function()
48028         {
48029             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48030                 return;
48031             }
48032             
48033             this.maskEl.top.setStyle('position', 'absolute');
48034             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48035             this.maskEl.top.hide();
48036
48037             this.maskEl.left.setStyle('position', 'absolute');
48038             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48039             this.maskEl.left.hide();
48040
48041             this.maskEl.bottom.setStyle('position', 'absolute');
48042             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48043             this.maskEl.bottom.hide();
48044
48045             this.maskEl.right.setStyle('position', 'absolute');
48046             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48047             this.maskEl.right.hide();
48048             
48049             window.onwheel = function(){ return true;};
48050             
48051             if(this.intervalID){
48052                 window.clearInterval(this.intervalID);
48053                 this.intervalID = false;
48054             }
48055             
48056             this.isMasked = false;
48057             
48058         }
48059         
48060     }
48061     
48062 });/*
48063  * Based on:
48064  * Ext JS Library 1.1.1
48065  * Copyright(c) 2006-2007, Ext JS, LLC.
48066  *
48067  * Originally Released Under LGPL - original licence link has changed is not relivant.
48068  *
48069  * Fork - LGPL
48070  * <script type="text/javascript">
48071  */
48072
48073 /**
48074  * @class Roo.form.Form
48075  * @extends Roo.form.BasicForm
48076  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48077  * @constructor
48078  * @param {Object} config Configuration options
48079  */
48080 Roo.form.Form = function(config){
48081     var xitems =  [];
48082     if (config.items) {
48083         xitems = config.items;
48084         delete config.items;
48085     }
48086    
48087     
48088     Roo.form.Form.superclass.constructor.call(this, null, config);
48089     this.url = this.url || this.action;
48090     if(!this.root){
48091         this.root = new Roo.form.Layout(Roo.applyIf({
48092             id: Roo.id()
48093         }, config));
48094     }
48095     this.active = this.root;
48096     /**
48097      * Array of all the buttons that have been added to this form via {@link addButton}
48098      * @type Array
48099      */
48100     this.buttons = [];
48101     this.allItems = [];
48102     this.addEvents({
48103         /**
48104          * @event clientvalidation
48105          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48106          * @param {Form} this
48107          * @param {Boolean} valid true if the form has passed client-side validation
48108          */
48109         clientvalidation: true,
48110         /**
48111          * @event rendered
48112          * Fires when the form is rendered
48113          * @param {Roo.form.Form} form
48114          */
48115         rendered : true
48116     });
48117     
48118     if (this.progressUrl) {
48119             // push a hidden field onto the list of fields..
48120             this.addxtype( {
48121                     xns: Roo.form, 
48122                     xtype : 'Hidden', 
48123                     name : 'UPLOAD_IDENTIFIER' 
48124             });
48125         }
48126         
48127     
48128     Roo.each(xitems, this.addxtype, this);
48129     
48130 };
48131
48132 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48133     /**
48134      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48135      */
48136     /**
48137      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48138      */
48139     /**
48140      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48141      */
48142     buttonAlign:'center',
48143
48144     /**
48145      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48146      */
48147     minButtonWidth:75,
48148
48149     /**
48150      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48151      * This property cascades to child containers if not set.
48152      */
48153     labelAlign:'left',
48154
48155     /**
48156      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48157      * fires a looping event with that state. This is required to bind buttons to the valid
48158      * state using the config value formBind:true on the button.
48159      */
48160     monitorValid : false,
48161
48162     /**
48163      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48164      */
48165     monitorPoll : 200,
48166     
48167     /**
48168      * @cfg {String} progressUrl - Url to return progress data 
48169      */
48170     
48171     progressUrl : false,
48172     /**
48173      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48174      * sending a formdata with extra parameters - eg uploaded elements.
48175      */
48176     
48177     formData : false,
48178     
48179     /**
48180      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48181      * fields are added and the column is closed. If no fields are passed the column remains open
48182      * until end() is called.
48183      * @param {Object} config The config to pass to the column
48184      * @param {Field} field1 (optional)
48185      * @param {Field} field2 (optional)
48186      * @param {Field} etc (optional)
48187      * @return Column The column container object
48188      */
48189     column : function(c){
48190         var col = new Roo.form.Column(c);
48191         this.start(col);
48192         if(arguments.length > 1){ // duplicate code required because of Opera
48193             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48194             this.end();
48195         }
48196         return col;
48197     },
48198
48199     /**
48200      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48201      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48202      * until end() is called.
48203      * @param {Object} config The config to pass to the fieldset
48204      * @param {Field} field1 (optional)
48205      * @param {Field} field2 (optional)
48206      * @param {Field} etc (optional)
48207      * @return FieldSet The fieldset container object
48208      */
48209     fieldset : function(c){
48210         var fs = new Roo.form.FieldSet(c);
48211         this.start(fs);
48212         if(arguments.length > 1){ // duplicate code required because of Opera
48213             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48214             this.end();
48215         }
48216         return fs;
48217     },
48218
48219     /**
48220      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48221      * fields are added and the container is closed. If no fields are passed the container remains open
48222      * until end() is called.
48223      * @param {Object} config The config to pass to the Layout
48224      * @param {Field} field1 (optional)
48225      * @param {Field} field2 (optional)
48226      * @param {Field} etc (optional)
48227      * @return Layout The container object
48228      */
48229     container : function(c){
48230         var l = new Roo.form.Layout(c);
48231         this.start(l);
48232         if(arguments.length > 1){ // duplicate code required because of Opera
48233             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48234             this.end();
48235         }
48236         return l;
48237     },
48238
48239     /**
48240      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48241      * @param {Object} container A Roo.form.Layout or subclass of Layout
48242      * @return {Form} this
48243      */
48244     start : function(c){
48245         // cascade label info
48246         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48247         this.active.stack.push(c);
48248         c.ownerCt = this.active;
48249         this.active = c;
48250         return this;
48251     },
48252
48253     /**
48254      * Closes the current open container
48255      * @return {Form} this
48256      */
48257     end : function(){
48258         if(this.active == this.root){
48259             return this;
48260         }
48261         this.active = this.active.ownerCt;
48262         return this;
48263     },
48264
48265     /**
48266      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48267      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48268      * as the label of the field.
48269      * @param {Field} field1
48270      * @param {Field} field2 (optional)
48271      * @param {Field} etc. (optional)
48272      * @return {Form} this
48273      */
48274     add : function(){
48275         this.active.stack.push.apply(this.active.stack, arguments);
48276         this.allItems.push.apply(this.allItems,arguments);
48277         var r = [];
48278         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48279             if(a[i].isFormField){
48280                 r.push(a[i]);
48281             }
48282         }
48283         if(r.length > 0){
48284             Roo.form.Form.superclass.add.apply(this, r);
48285         }
48286         return this;
48287     },
48288     
48289
48290     
48291     
48292     
48293      /**
48294      * Find any element that has been added to a form, using it's ID or name
48295      * This can include framesets, columns etc. along with regular fields..
48296      * @param {String} id - id or name to find.
48297      
48298      * @return {Element} e - or false if nothing found.
48299      */
48300     findbyId : function(id)
48301     {
48302         var ret = false;
48303         if (!id) {
48304             return ret;
48305         }
48306         Roo.each(this.allItems, function(f){
48307             if (f.id == id || f.name == id ){
48308                 ret = f;
48309                 return false;
48310             }
48311         });
48312         return ret;
48313     },
48314
48315     
48316     
48317     /**
48318      * Render this form into the passed container. This should only be called once!
48319      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48320      * @return {Form} this
48321      */
48322     render : function(ct)
48323     {
48324         
48325         
48326         
48327         ct = Roo.get(ct);
48328         var o = this.autoCreate || {
48329             tag: 'form',
48330             method : this.method || 'POST',
48331             id : this.id || Roo.id()
48332         };
48333         this.initEl(ct.createChild(o));
48334
48335         this.root.render(this.el);
48336         
48337        
48338              
48339         this.items.each(function(f){
48340             f.render('x-form-el-'+f.id);
48341         });
48342
48343         if(this.buttons.length > 0){
48344             // tables are required to maintain order and for correct IE layout
48345             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48346                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48347                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48348             }}, null, true);
48349             var tr = tb.getElementsByTagName('tr')[0];
48350             for(var i = 0, len = this.buttons.length; i < len; i++) {
48351                 var b = this.buttons[i];
48352                 var td = document.createElement('td');
48353                 td.className = 'x-form-btn-td';
48354                 b.render(tr.appendChild(td));
48355             }
48356         }
48357         if(this.monitorValid){ // initialize after render
48358             this.startMonitoring();
48359         }
48360         this.fireEvent('rendered', this);
48361         return this;
48362     },
48363
48364     /**
48365      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48366      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48367      * object or a valid Roo.DomHelper element config
48368      * @param {Function} handler The function called when the button is clicked
48369      * @param {Object} scope (optional) The scope of the handler function
48370      * @return {Roo.Button}
48371      */
48372     addButton : function(config, handler, scope){
48373         var bc = {
48374             handler: handler,
48375             scope: scope,
48376             minWidth: this.minButtonWidth,
48377             hideParent:true
48378         };
48379         if(typeof config == "string"){
48380             bc.text = config;
48381         }else{
48382             Roo.apply(bc, config);
48383         }
48384         var btn = new Roo.Button(null, bc);
48385         this.buttons.push(btn);
48386         return btn;
48387     },
48388
48389      /**
48390      * Adds a series of form elements (using the xtype property as the factory method.
48391      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48392      * @param {Object} config 
48393      */
48394     
48395     addxtype : function()
48396     {
48397         var ar = Array.prototype.slice.call(arguments, 0);
48398         var ret = false;
48399         for(var i = 0; i < ar.length; i++) {
48400             if (!ar[i]) {
48401                 continue; // skip -- if this happends something invalid got sent, we 
48402                 // should ignore it, as basically that interface element will not show up
48403                 // and that should be pretty obvious!!
48404             }
48405             
48406             if (Roo.form[ar[i].xtype]) {
48407                 ar[i].form = this;
48408                 var fe = Roo.factory(ar[i], Roo.form);
48409                 if (!ret) {
48410                     ret = fe;
48411                 }
48412                 fe.form = this;
48413                 if (fe.store) {
48414                     fe.store.form = this;
48415                 }
48416                 if (fe.isLayout) {  
48417                          
48418                     this.start(fe);
48419                     this.allItems.push(fe);
48420                     if (fe.items && fe.addxtype) {
48421                         fe.addxtype.apply(fe, fe.items);
48422                         delete fe.items;
48423                     }
48424                      this.end();
48425                     continue;
48426                 }
48427                 
48428                 
48429                  
48430                 this.add(fe);
48431               //  console.log('adding ' + ar[i].xtype);
48432             }
48433             if (ar[i].xtype == 'Button') {  
48434                 //console.log('adding button');
48435                 //console.log(ar[i]);
48436                 this.addButton(ar[i]);
48437                 this.allItems.push(fe);
48438                 continue;
48439             }
48440             
48441             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48442                 alert('end is not supported on xtype any more, use items');
48443             //    this.end();
48444             //    //console.log('adding end');
48445             }
48446             
48447         }
48448         return ret;
48449     },
48450     
48451     /**
48452      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48453      * option "monitorValid"
48454      */
48455     startMonitoring : function(){
48456         if(!this.bound){
48457             this.bound = true;
48458             Roo.TaskMgr.start({
48459                 run : this.bindHandler,
48460                 interval : this.monitorPoll || 200,
48461                 scope: this
48462             });
48463         }
48464     },
48465
48466     /**
48467      * Stops monitoring of the valid state of this form
48468      */
48469     stopMonitoring : function(){
48470         this.bound = false;
48471     },
48472
48473     // private
48474     bindHandler : function(){
48475         if(!this.bound){
48476             return false; // stops binding
48477         }
48478         var valid = true;
48479         this.items.each(function(f){
48480             if(!f.isValid(true)){
48481                 valid = false;
48482                 return false;
48483             }
48484         });
48485         for(var i = 0, len = this.buttons.length; i < len; i++){
48486             var btn = this.buttons[i];
48487             if(btn.formBind === true && btn.disabled === valid){
48488                 btn.setDisabled(!valid);
48489             }
48490         }
48491         this.fireEvent('clientvalidation', this, valid);
48492     }
48493     
48494     
48495     
48496     
48497     
48498     
48499     
48500     
48501 });
48502
48503
48504 // back compat
48505 Roo.Form = Roo.form.Form;
48506 /*
48507  * Based on:
48508  * Ext JS Library 1.1.1
48509  * Copyright(c) 2006-2007, Ext JS, LLC.
48510  *
48511  * Originally Released Under LGPL - original licence link has changed is not relivant.
48512  *
48513  * Fork - LGPL
48514  * <script type="text/javascript">
48515  */
48516
48517 // as we use this in bootstrap.
48518 Roo.namespace('Roo.form');
48519  /**
48520  * @class Roo.form.Action
48521  * Internal Class used to handle form actions
48522  * @constructor
48523  * @param {Roo.form.BasicForm} el The form element or its id
48524  * @param {Object} config Configuration options
48525  */
48526
48527  
48528  
48529 // define the action interface
48530 Roo.form.Action = function(form, options){
48531     this.form = form;
48532     this.options = options || {};
48533 };
48534 /**
48535  * Client Validation Failed
48536  * @const 
48537  */
48538 Roo.form.Action.CLIENT_INVALID = 'client';
48539 /**
48540  * Server Validation Failed
48541  * @const 
48542  */
48543 Roo.form.Action.SERVER_INVALID = 'server';
48544  /**
48545  * Connect to Server Failed
48546  * @const 
48547  */
48548 Roo.form.Action.CONNECT_FAILURE = 'connect';
48549 /**
48550  * Reading Data from Server Failed
48551  * @const 
48552  */
48553 Roo.form.Action.LOAD_FAILURE = 'load';
48554
48555 Roo.form.Action.prototype = {
48556     type : 'default',
48557     failureType : undefined,
48558     response : undefined,
48559     result : undefined,
48560
48561     // interface method
48562     run : function(options){
48563
48564     },
48565
48566     // interface method
48567     success : function(response){
48568
48569     },
48570
48571     // interface method
48572     handleResponse : function(response){
48573
48574     },
48575
48576     // default connection failure
48577     failure : function(response){
48578         
48579         this.response = response;
48580         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48581         this.form.afterAction(this, false);
48582     },
48583
48584     processResponse : function(response){
48585         this.response = response;
48586         if(!response.responseText){
48587             return true;
48588         }
48589         this.result = this.handleResponse(response);
48590         return this.result;
48591     },
48592
48593     // utility functions used internally
48594     getUrl : function(appendParams){
48595         var url = this.options.url || this.form.url || this.form.el.dom.action;
48596         if(appendParams){
48597             var p = this.getParams();
48598             if(p){
48599                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48600             }
48601         }
48602         return url;
48603     },
48604
48605     getMethod : function(){
48606         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48607     },
48608
48609     getParams : function(){
48610         var bp = this.form.baseParams;
48611         var p = this.options.params;
48612         if(p){
48613             if(typeof p == "object"){
48614                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48615             }else if(typeof p == 'string' && bp){
48616                 p += '&' + Roo.urlEncode(bp);
48617             }
48618         }else if(bp){
48619             p = Roo.urlEncode(bp);
48620         }
48621         return p;
48622     },
48623
48624     createCallback : function(){
48625         return {
48626             success: this.success,
48627             failure: this.failure,
48628             scope: this,
48629             timeout: (this.form.timeout*1000),
48630             upload: this.form.fileUpload ? this.success : undefined
48631         };
48632     }
48633 };
48634
48635 Roo.form.Action.Submit = function(form, options){
48636     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48637 };
48638
48639 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48640     type : 'submit',
48641
48642     haveProgress : false,
48643     uploadComplete : false,
48644     
48645     // uploadProgress indicator.
48646     uploadProgress : function()
48647     {
48648         if (!this.form.progressUrl) {
48649             return;
48650         }
48651         
48652         if (!this.haveProgress) {
48653             Roo.MessageBox.progress("Uploading", "Uploading");
48654         }
48655         if (this.uploadComplete) {
48656            Roo.MessageBox.hide();
48657            return;
48658         }
48659         
48660         this.haveProgress = true;
48661    
48662         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48663         
48664         var c = new Roo.data.Connection();
48665         c.request({
48666             url : this.form.progressUrl,
48667             params: {
48668                 id : uid
48669             },
48670             method: 'GET',
48671             success : function(req){
48672                //console.log(data);
48673                 var rdata = false;
48674                 var edata;
48675                 try  {
48676                    rdata = Roo.decode(req.responseText)
48677                 } catch (e) {
48678                     Roo.log("Invalid data from server..");
48679                     Roo.log(edata);
48680                     return;
48681                 }
48682                 if (!rdata || !rdata.success) {
48683                     Roo.log(rdata);
48684                     Roo.MessageBox.alert(Roo.encode(rdata));
48685                     return;
48686                 }
48687                 var data = rdata.data;
48688                 
48689                 if (this.uploadComplete) {
48690                    Roo.MessageBox.hide();
48691                    return;
48692                 }
48693                    
48694                 if (data){
48695                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48696                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48697                     );
48698                 }
48699                 this.uploadProgress.defer(2000,this);
48700             },
48701        
48702             failure: function(data) {
48703                 Roo.log('progress url failed ');
48704                 Roo.log(data);
48705             },
48706             scope : this
48707         });
48708            
48709     },
48710     
48711     
48712     run : function()
48713     {
48714         // run get Values on the form, so it syncs any secondary forms.
48715         this.form.getValues();
48716         
48717         var o = this.options;
48718         var method = this.getMethod();
48719         var isPost = method == 'POST';
48720         if(o.clientValidation === false || this.form.isValid()){
48721             
48722             if (this.form.progressUrl) {
48723                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48724                     (new Date() * 1) + '' + Math.random());
48725                     
48726             } 
48727             
48728             
48729             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48730                 form:this.form.el.dom,
48731                 url:this.getUrl(!isPost),
48732                 method: method,
48733                 params:isPost ? this.getParams() : null,
48734                 isUpload: this.form.fileUpload,
48735                 formData : this.form.formData
48736             }));
48737             
48738             this.uploadProgress();
48739
48740         }else if (o.clientValidation !== false){ // client validation failed
48741             this.failureType = Roo.form.Action.CLIENT_INVALID;
48742             this.form.afterAction(this, false);
48743         }
48744     },
48745
48746     success : function(response)
48747     {
48748         this.uploadComplete= true;
48749         if (this.haveProgress) {
48750             Roo.MessageBox.hide();
48751         }
48752         
48753         
48754         var result = this.processResponse(response);
48755         if(result === true || result.success){
48756             this.form.afterAction(this, true);
48757             return;
48758         }
48759         if(result.errors){
48760             this.form.markInvalid(result.errors);
48761             this.failureType = Roo.form.Action.SERVER_INVALID;
48762         }
48763         this.form.afterAction(this, false);
48764     },
48765     failure : function(response)
48766     {
48767         this.uploadComplete= true;
48768         if (this.haveProgress) {
48769             Roo.MessageBox.hide();
48770         }
48771         
48772         this.response = response;
48773         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48774         this.form.afterAction(this, false);
48775     },
48776     
48777     handleResponse : function(response){
48778         if(this.form.errorReader){
48779             var rs = this.form.errorReader.read(response);
48780             var errors = [];
48781             if(rs.records){
48782                 for(var i = 0, len = rs.records.length; i < len; i++) {
48783                     var r = rs.records[i];
48784                     errors[i] = r.data;
48785                 }
48786             }
48787             if(errors.length < 1){
48788                 errors = null;
48789             }
48790             return {
48791                 success : rs.success,
48792                 errors : errors
48793             };
48794         }
48795         var ret = false;
48796         try {
48797             ret = Roo.decode(response.responseText);
48798         } catch (e) {
48799             ret = {
48800                 success: false,
48801                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48802                 errors : []
48803             };
48804         }
48805         return ret;
48806         
48807     }
48808 });
48809
48810
48811 Roo.form.Action.Load = function(form, options){
48812     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48813     this.reader = this.form.reader;
48814 };
48815
48816 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48817     type : 'load',
48818
48819     run : function(){
48820         
48821         Roo.Ajax.request(Roo.apply(
48822                 this.createCallback(), {
48823                     method:this.getMethod(),
48824                     url:this.getUrl(false),
48825                     params:this.getParams()
48826         }));
48827     },
48828
48829     success : function(response){
48830         
48831         var result = this.processResponse(response);
48832         if(result === true || !result.success || !result.data){
48833             this.failureType = Roo.form.Action.LOAD_FAILURE;
48834             this.form.afterAction(this, false);
48835             return;
48836         }
48837         this.form.clearInvalid();
48838         this.form.setValues(result.data);
48839         this.form.afterAction(this, true);
48840     },
48841
48842     handleResponse : function(response){
48843         if(this.form.reader){
48844             var rs = this.form.reader.read(response);
48845             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48846             return {
48847                 success : rs.success,
48848                 data : data
48849             };
48850         }
48851         return Roo.decode(response.responseText);
48852     }
48853 });
48854
48855 Roo.form.Action.ACTION_TYPES = {
48856     'load' : Roo.form.Action.Load,
48857     'submit' : Roo.form.Action.Submit
48858 };/*
48859  * Based on:
48860  * Ext JS Library 1.1.1
48861  * Copyright(c) 2006-2007, Ext JS, LLC.
48862  *
48863  * Originally Released Under LGPL - original licence link has changed is not relivant.
48864  *
48865  * Fork - LGPL
48866  * <script type="text/javascript">
48867  */
48868  
48869 /**
48870  * @class Roo.form.Layout
48871  * @extends Roo.Component
48872  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48873  * @constructor
48874  * @param {Object} config Configuration options
48875  */
48876 Roo.form.Layout = function(config){
48877     var xitems = [];
48878     if (config.items) {
48879         xitems = config.items;
48880         delete config.items;
48881     }
48882     Roo.form.Layout.superclass.constructor.call(this, config);
48883     this.stack = [];
48884     Roo.each(xitems, this.addxtype, this);
48885      
48886 };
48887
48888 Roo.extend(Roo.form.Layout, Roo.Component, {
48889     /**
48890      * @cfg {String/Object} autoCreate
48891      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48892      */
48893     /**
48894      * @cfg {String/Object/Function} style
48895      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48896      * a function which returns such a specification.
48897      */
48898     /**
48899      * @cfg {String} labelAlign
48900      * Valid values are "left," "top" and "right" (defaults to "left")
48901      */
48902     /**
48903      * @cfg {Number} labelWidth
48904      * Fixed width in pixels of all field labels (defaults to undefined)
48905      */
48906     /**
48907      * @cfg {Boolean} clear
48908      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48909      */
48910     clear : true,
48911     /**
48912      * @cfg {String} labelSeparator
48913      * The separator to use after field labels (defaults to ':')
48914      */
48915     labelSeparator : ':',
48916     /**
48917      * @cfg {Boolean} hideLabels
48918      * True to suppress the display of field labels in this layout (defaults to false)
48919      */
48920     hideLabels : false,
48921
48922     // private
48923     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48924     
48925     isLayout : true,
48926     
48927     // private
48928     onRender : function(ct, position){
48929         if(this.el){ // from markup
48930             this.el = Roo.get(this.el);
48931         }else {  // generate
48932             var cfg = this.getAutoCreate();
48933             this.el = ct.createChild(cfg, position);
48934         }
48935         if(this.style){
48936             this.el.applyStyles(this.style);
48937         }
48938         if(this.labelAlign){
48939             this.el.addClass('x-form-label-'+this.labelAlign);
48940         }
48941         if(this.hideLabels){
48942             this.labelStyle = "display:none";
48943             this.elementStyle = "padding-left:0;";
48944         }else{
48945             if(typeof this.labelWidth == 'number'){
48946                 this.labelStyle = "width:"+this.labelWidth+"px;";
48947                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
48948             }
48949             if(this.labelAlign == 'top'){
48950                 this.labelStyle = "width:auto;";
48951                 this.elementStyle = "padding-left:0;";
48952             }
48953         }
48954         var stack = this.stack;
48955         var slen = stack.length;
48956         if(slen > 0){
48957             if(!this.fieldTpl){
48958                 var t = new Roo.Template(
48959                     '<div class="x-form-item {5}">',
48960                         '<label for="{0}" style="{2}">{1}{4}</label>',
48961                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
48962                         '</div>',
48963                     '</div><div class="x-form-clear-left"></div>'
48964                 );
48965                 t.disableFormats = true;
48966                 t.compile();
48967                 Roo.form.Layout.prototype.fieldTpl = t;
48968             }
48969             for(var i = 0; i < slen; i++) {
48970                 if(stack[i].isFormField){
48971                     this.renderField(stack[i]);
48972                 }else{
48973                     this.renderComponent(stack[i]);
48974                 }
48975             }
48976         }
48977         if(this.clear){
48978             this.el.createChild({cls:'x-form-clear'});
48979         }
48980     },
48981
48982     // private
48983     renderField : function(f){
48984         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
48985                f.id, //0
48986                f.fieldLabel, //1
48987                f.labelStyle||this.labelStyle||'', //2
48988                this.elementStyle||'', //3
48989                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
48990                f.itemCls||this.itemCls||''  //5
48991        ], true).getPrevSibling());
48992     },
48993
48994     // private
48995     renderComponent : function(c){
48996         c.render(c.isLayout ? this.el : this.el.createChild());    
48997     },
48998     /**
48999      * Adds a object form elements (using the xtype property as the factory method.)
49000      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49001      * @param {Object} config 
49002      */
49003     addxtype : function(o)
49004     {
49005         // create the lement.
49006         o.form = this.form;
49007         var fe = Roo.factory(o, Roo.form);
49008         this.form.allItems.push(fe);
49009         this.stack.push(fe);
49010         
49011         if (fe.isFormField) {
49012             this.form.items.add(fe);
49013         }
49014          
49015         return fe;
49016     }
49017 });
49018
49019 /**
49020  * @class Roo.form.Column
49021  * @extends Roo.form.Layout
49022  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49023  * @constructor
49024  * @param {Object} config Configuration options
49025  */
49026 Roo.form.Column = function(config){
49027     Roo.form.Column.superclass.constructor.call(this, config);
49028 };
49029
49030 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49031     /**
49032      * @cfg {Number/String} width
49033      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49034      */
49035     /**
49036      * @cfg {String/Object} autoCreate
49037      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49038      */
49039
49040     // private
49041     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49042
49043     // private
49044     onRender : function(ct, position){
49045         Roo.form.Column.superclass.onRender.call(this, ct, position);
49046         if(this.width){
49047             this.el.setWidth(this.width);
49048         }
49049     }
49050 });
49051
49052
49053 /**
49054  * @class Roo.form.Row
49055  * @extends Roo.form.Layout
49056  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49057  * @constructor
49058  * @param {Object} config Configuration options
49059  */
49060
49061  
49062 Roo.form.Row = function(config){
49063     Roo.form.Row.superclass.constructor.call(this, config);
49064 };
49065  
49066 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49067       /**
49068      * @cfg {Number/String} width
49069      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49070      */
49071     /**
49072      * @cfg {Number/String} height
49073      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49074      */
49075     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49076     
49077     padWidth : 20,
49078     // private
49079     onRender : function(ct, position){
49080         //console.log('row render');
49081         if(!this.rowTpl){
49082             var t = new Roo.Template(
49083                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49084                     '<label for="{0}" style="{2}">{1}{4}</label>',
49085                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49086                     '</div>',
49087                 '</div>'
49088             );
49089             t.disableFormats = true;
49090             t.compile();
49091             Roo.form.Layout.prototype.rowTpl = t;
49092         }
49093         this.fieldTpl = this.rowTpl;
49094         
49095         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49096         var labelWidth = 100;
49097         
49098         if ((this.labelAlign != 'top')) {
49099             if (typeof this.labelWidth == 'number') {
49100                 labelWidth = this.labelWidth
49101             }
49102             this.padWidth =  20 + labelWidth;
49103             
49104         }
49105         
49106         Roo.form.Column.superclass.onRender.call(this, ct, position);
49107         if(this.width){
49108             this.el.setWidth(this.width);
49109         }
49110         if(this.height){
49111             this.el.setHeight(this.height);
49112         }
49113     },
49114     
49115     // private
49116     renderField : function(f){
49117         f.fieldEl = this.fieldTpl.append(this.el, [
49118                f.id, f.fieldLabel,
49119                f.labelStyle||this.labelStyle||'',
49120                this.elementStyle||'',
49121                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49122                f.itemCls||this.itemCls||'',
49123                f.width ? f.width + this.padWidth : 160 + this.padWidth
49124        ],true);
49125     }
49126 });
49127  
49128
49129 /**
49130  * @class Roo.form.FieldSet
49131  * @extends Roo.form.Layout
49132  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49133  * @constructor
49134  * @param {Object} config Configuration options
49135  */
49136 Roo.form.FieldSet = function(config){
49137     Roo.form.FieldSet.superclass.constructor.call(this, config);
49138 };
49139
49140 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49141     /**
49142      * @cfg {String} legend
49143      * The text to display as the legend for the FieldSet (defaults to '')
49144      */
49145     /**
49146      * @cfg {String/Object} autoCreate
49147      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49148      */
49149
49150     // private
49151     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49152
49153     // private
49154     onRender : function(ct, position){
49155         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49156         if(this.legend){
49157             this.setLegend(this.legend);
49158         }
49159     },
49160
49161     // private
49162     setLegend : function(text){
49163         if(this.rendered){
49164             this.el.child('legend').update(text);
49165         }
49166     }
49167 });/*
49168  * Based on:
49169  * Ext JS Library 1.1.1
49170  * Copyright(c) 2006-2007, Ext JS, LLC.
49171  *
49172  * Originally Released Under LGPL - original licence link has changed is not relivant.
49173  *
49174  * Fork - LGPL
49175  * <script type="text/javascript">
49176  */
49177 /**
49178  * @class Roo.form.VTypes
49179  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49180  * @singleton
49181  */
49182 Roo.form.VTypes = function(){
49183     // closure these in so they are only created once.
49184     var alpha = /^[a-zA-Z_]+$/;
49185     var alphanum = /^[a-zA-Z0-9_]+$/;
49186     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49187     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49188
49189     // All these messages and functions are configurable
49190     return {
49191         /**
49192          * The function used to validate email addresses
49193          * @param {String} value The email address
49194          */
49195         'email' : function(v){
49196             return email.test(v);
49197         },
49198         /**
49199          * The error text to display when the email validation function returns false
49200          * @type String
49201          */
49202         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49203         /**
49204          * The keystroke filter mask to be applied on email input
49205          * @type RegExp
49206          */
49207         'emailMask' : /[a-z0-9_\.\-@]/i,
49208
49209         /**
49210          * The function used to validate URLs
49211          * @param {String} value The URL
49212          */
49213         'url' : function(v){
49214             return url.test(v);
49215         },
49216         /**
49217          * The error text to display when the url validation function returns false
49218          * @type String
49219          */
49220         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49221         
49222         /**
49223          * The function used to validate alpha values
49224          * @param {String} value The value
49225          */
49226         'alpha' : function(v){
49227             return alpha.test(v);
49228         },
49229         /**
49230          * The error text to display when the alpha validation function returns false
49231          * @type String
49232          */
49233         'alphaText' : 'This field should only contain letters and _',
49234         /**
49235          * The keystroke filter mask to be applied on alpha input
49236          * @type RegExp
49237          */
49238         'alphaMask' : /[a-z_]/i,
49239
49240         /**
49241          * The function used to validate alphanumeric values
49242          * @param {String} value The value
49243          */
49244         'alphanum' : function(v){
49245             return alphanum.test(v);
49246         },
49247         /**
49248          * The error text to display when the alphanumeric validation function returns false
49249          * @type String
49250          */
49251         'alphanumText' : 'This field should only contain letters, numbers and _',
49252         /**
49253          * The keystroke filter mask to be applied on alphanumeric input
49254          * @type RegExp
49255          */
49256         'alphanumMask' : /[a-z0-9_]/i
49257     };
49258 }();//<script type="text/javascript">
49259
49260 /**
49261  * @class Roo.form.FCKeditor
49262  * @extends Roo.form.TextArea
49263  * Wrapper around the FCKEditor http://www.fckeditor.net
49264  * @constructor
49265  * Creates a new FCKeditor
49266  * @param {Object} config Configuration options
49267  */
49268 Roo.form.FCKeditor = function(config){
49269     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49270     this.addEvents({
49271          /**
49272          * @event editorinit
49273          * Fired when the editor is initialized - you can add extra handlers here..
49274          * @param {FCKeditor} this
49275          * @param {Object} the FCK object.
49276          */
49277         editorinit : true
49278     });
49279     
49280     
49281 };
49282 Roo.form.FCKeditor.editors = { };
49283 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49284 {
49285     //defaultAutoCreate : {
49286     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49287     //},
49288     // private
49289     /**
49290      * @cfg {Object} fck options - see fck manual for details.
49291      */
49292     fckconfig : false,
49293     
49294     /**
49295      * @cfg {Object} fck toolbar set (Basic or Default)
49296      */
49297     toolbarSet : 'Basic',
49298     /**
49299      * @cfg {Object} fck BasePath
49300      */ 
49301     basePath : '/fckeditor/',
49302     
49303     
49304     frame : false,
49305     
49306     value : '',
49307     
49308    
49309     onRender : function(ct, position)
49310     {
49311         if(!this.el){
49312             this.defaultAutoCreate = {
49313                 tag: "textarea",
49314                 style:"width:300px;height:60px;",
49315                 autocomplete: "new-password"
49316             };
49317         }
49318         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49319         /*
49320         if(this.grow){
49321             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49322             if(this.preventScrollbars){
49323                 this.el.setStyle("overflow", "hidden");
49324             }
49325             this.el.setHeight(this.growMin);
49326         }
49327         */
49328         //console.log('onrender' + this.getId() );
49329         Roo.form.FCKeditor.editors[this.getId()] = this;
49330          
49331
49332         this.replaceTextarea() ;
49333         
49334     },
49335     
49336     getEditor : function() {
49337         return this.fckEditor;
49338     },
49339     /**
49340      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49341      * @param {Mixed} value The value to set
49342      */
49343     
49344     
49345     setValue : function(value)
49346     {
49347         //console.log('setValue: ' + value);
49348         
49349         if(typeof(value) == 'undefined') { // not sure why this is happending...
49350             return;
49351         }
49352         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49353         
49354         //if(!this.el || !this.getEditor()) {
49355         //    this.value = value;
49356             //this.setValue.defer(100,this,[value]);    
49357         //    return;
49358         //} 
49359         
49360         if(!this.getEditor()) {
49361             return;
49362         }
49363         
49364         this.getEditor().SetData(value);
49365         
49366         //
49367
49368     },
49369
49370     /**
49371      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49372      * @return {Mixed} value The field value
49373      */
49374     getValue : function()
49375     {
49376         
49377         if (this.frame && this.frame.dom.style.display == 'none') {
49378             return Roo.form.FCKeditor.superclass.getValue.call(this);
49379         }
49380         
49381         if(!this.el || !this.getEditor()) {
49382            
49383            // this.getValue.defer(100,this); 
49384             return this.value;
49385         }
49386        
49387         
49388         var value=this.getEditor().GetData();
49389         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49390         return Roo.form.FCKeditor.superclass.getValue.call(this);
49391         
49392
49393     },
49394
49395     /**
49396      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49397      * @return {Mixed} value The field value
49398      */
49399     getRawValue : function()
49400     {
49401         if (this.frame && this.frame.dom.style.display == 'none') {
49402             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49403         }
49404         
49405         if(!this.el || !this.getEditor()) {
49406             //this.getRawValue.defer(100,this); 
49407             return this.value;
49408             return;
49409         }
49410         
49411         
49412         
49413         var value=this.getEditor().GetData();
49414         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49415         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49416          
49417     },
49418     
49419     setSize : function(w,h) {
49420         
49421         
49422         
49423         //if (this.frame && this.frame.dom.style.display == 'none') {
49424         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49425         //    return;
49426         //}
49427         //if(!this.el || !this.getEditor()) {
49428         //    this.setSize.defer(100,this, [w,h]); 
49429         //    return;
49430         //}
49431         
49432         
49433         
49434         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49435         
49436         this.frame.dom.setAttribute('width', w);
49437         this.frame.dom.setAttribute('height', h);
49438         this.frame.setSize(w,h);
49439         
49440     },
49441     
49442     toggleSourceEdit : function(value) {
49443         
49444       
49445          
49446         this.el.dom.style.display = value ? '' : 'none';
49447         this.frame.dom.style.display = value ?  'none' : '';
49448         
49449     },
49450     
49451     
49452     focus: function(tag)
49453     {
49454         if (this.frame.dom.style.display == 'none') {
49455             return Roo.form.FCKeditor.superclass.focus.call(this);
49456         }
49457         if(!this.el || !this.getEditor()) {
49458             this.focus.defer(100,this, [tag]); 
49459             return;
49460         }
49461         
49462         
49463         
49464         
49465         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49466         this.getEditor().Focus();
49467         if (tgs.length) {
49468             if (!this.getEditor().Selection.GetSelection()) {
49469                 this.focus.defer(100,this, [tag]); 
49470                 return;
49471             }
49472             
49473             
49474             var r = this.getEditor().EditorDocument.createRange();
49475             r.setStart(tgs[0],0);
49476             r.setEnd(tgs[0],0);
49477             this.getEditor().Selection.GetSelection().removeAllRanges();
49478             this.getEditor().Selection.GetSelection().addRange(r);
49479             this.getEditor().Focus();
49480         }
49481         
49482     },
49483     
49484     
49485     
49486     replaceTextarea : function()
49487     {
49488         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49489             return ;
49490         }
49491         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49492         //{
49493             // We must check the elements firstly using the Id and then the name.
49494         var oTextarea = document.getElementById( this.getId() );
49495         
49496         var colElementsByName = document.getElementsByName( this.getId() ) ;
49497          
49498         oTextarea.style.display = 'none' ;
49499
49500         if ( oTextarea.tabIndex ) {            
49501             this.TabIndex = oTextarea.tabIndex ;
49502         }
49503         
49504         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49505         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49506         this.frame = Roo.get(this.getId() + '___Frame')
49507     },
49508     
49509     _getConfigHtml : function()
49510     {
49511         var sConfig = '' ;
49512
49513         for ( var o in this.fckconfig ) {
49514             sConfig += sConfig.length > 0  ? '&amp;' : '';
49515             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49516         }
49517
49518         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49519     },
49520     
49521     
49522     _getIFrameHtml : function()
49523     {
49524         var sFile = 'fckeditor.html' ;
49525         /* no idea what this is about..
49526         try
49527         {
49528             if ( (/fcksource=true/i).test( window.top.location.search ) )
49529                 sFile = 'fckeditor.original.html' ;
49530         }
49531         catch (e) { 
49532         */
49533
49534         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49535         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49536         
49537         
49538         var html = '<iframe id="' + this.getId() +
49539             '___Frame" src="' + sLink +
49540             '" width="' + this.width +
49541             '" height="' + this.height + '"' +
49542             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49543             ' frameborder="0" scrolling="no"></iframe>' ;
49544
49545         return html ;
49546     },
49547     
49548     _insertHtmlBefore : function( html, element )
49549     {
49550         if ( element.insertAdjacentHTML )       {
49551             // IE
49552             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49553         } else { // Gecko
49554             var oRange = document.createRange() ;
49555             oRange.setStartBefore( element ) ;
49556             var oFragment = oRange.createContextualFragment( html );
49557             element.parentNode.insertBefore( oFragment, element ) ;
49558         }
49559     }
49560     
49561     
49562   
49563     
49564     
49565     
49566     
49567
49568 });
49569
49570 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49571
49572 function FCKeditor_OnComplete(editorInstance){
49573     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49574     f.fckEditor = editorInstance;
49575     //console.log("loaded");
49576     f.fireEvent('editorinit', f, editorInstance);
49577
49578   
49579
49580  
49581
49582
49583
49584
49585
49586
49587
49588
49589
49590
49591
49592
49593
49594
49595
49596 //<script type="text/javascript">
49597 /**
49598  * @class Roo.form.GridField
49599  * @extends Roo.form.Field
49600  * Embed a grid (or editable grid into a form)
49601  * STATUS ALPHA
49602  * 
49603  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49604  * it needs 
49605  * xgrid.store = Roo.data.Store
49606  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49607  * xgrid.store.reader = Roo.data.JsonReader 
49608  * 
49609  * 
49610  * @constructor
49611  * Creates a new GridField
49612  * @param {Object} config Configuration options
49613  */
49614 Roo.form.GridField = function(config){
49615     Roo.form.GridField.superclass.constructor.call(this, config);
49616      
49617 };
49618
49619 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49620     /**
49621      * @cfg {Number} width  - used to restrict width of grid..
49622      */
49623     width : 100,
49624     /**
49625      * @cfg {Number} height - used to restrict height of grid..
49626      */
49627     height : 50,
49628      /**
49629      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49630          * 
49631          *}
49632      */
49633     xgrid : false, 
49634     /**
49635      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49636      * {tag: "input", type: "checkbox", autocomplete: "off"})
49637      */
49638    // defaultAutoCreate : { tag: 'div' },
49639     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49640     /**
49641      * @cfg {String} addTitle Text to include for adding a title.
49642      */
49643     addTitle : false,
49644     //
49645     onResize : function(){
49646         Roo.form.Field.superclass.onResize.apply(this, arguments);
49647     },
49648
49649     initEvents : function(){
49650         // Roo.form.Checkbox.superclass.initEvents.call(this);
49651         // has no events...
49652        
49653     },
49654
49655
49656     getResizeEl : function(){
49657         return this.wrap;
49658     },
49659
49660     getPositionEl : function(){
49661         return this.wrap;
49662     },
49663
49664     // private
49665     onRender : function(ct, position){
49666         
49667         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49668         var style = this.style;
49669         delete this.style;
49670         
49671         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49672         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49673         this.viewEl = this.wrap.createChild({ tag: 'div' });
49674         if (style) {
49675             this.viewEl.applyStyles(style);
49676         }
49677         if (this.width) {
49678             this.viewEl.setWidth(this.width);
49679         }
49680         if (this.height) {
49681             this.viewEl.setHeight(this.height);
49682         }
49683         //if(this.inputValue !== undefined){
49684         //this.setValue(this.value);
49685         
49686         
49687         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49688         
49689         
49690         this.grid.render();
49691         this.grid.getDataSource().on('remove', this.refreshValue, this);
49692         this.grid.getDataSource().on('update', this.refreshValue, this);
49693         this.grid.on('afteredit', this.refreshValue, this);
49694  
49695     },
49696      
49697     
49698     /**
49699      * Sets the value of the item. 
49700      * @param {String} either an object  or a string..
49701      */
49702     setValue : function(v){
49703         //this.value = v;
49704         v = v || []; // empty set..
49705         // this does not seem smart - it really only affects memoryproxy grids..
49706         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49707             var ds = this.grid.getDataSource();
49708             // assumes a json reader..
49709             var data = {}
49710             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49711             ds.loadData( data);
49712         }
49713         // clear selection so it does not get stale.
49714         if (this.grid.sm) { 
49715             this.grid.sm.clearSelections();
49716         }
49717         
49718         Roo.form.GridField.superclass.setValue.call(this, v);
49719         this.refreshValue();
49720         // should load data in the grid really....
49721     },
49722     
49723     // private
49724     refreshValue: function() {
49725          var val = [];
49726         this.grid.getDataSource().each(function(r) {
49727             val.push(r.data);
49728         });
49729         this.el.dom.value = Roo.encode(val);
49730     }
49731     
49732      
49733     
49734     
49735 });/*
49736  * Based on:
49737  * Ext JS Library 1.1.1
49738  * Copyright(c) 2006-2007, Ext JS, LLC.
49739  *
49740  * Originally Released Under LGPL - original licence link has changed is not relivant.
49741  *
49742  * Fork - LGPL
49743  * <script type="text/javascript">
49744  */
49745 /**
49746  * @class Roo.form.DisplayField
49747  * @extends Roo.form.Field
49748  * A generic Field to display non-editable data.
49749  * @cfg {Boolean} closable (true|false) default false
49750  * @constructor
49751  * Creates a new Display Field item.
49752  * @param {Object} config Configuration options
49753  */
49754 Roo.form.DisplayField = function(config){
49755     Roo.form.DisplayField.superclass.constructor.call(this, config);
49756     
49757     this.addEvents({
49758         /**
49759          * @event close
49760          * Fires after the click the close btn
49761              * @param {Roo.form.DisplayField} this
49762              */
49763         close : true
49764     });
49765 };
49766
49767 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49768     inputType:      'hidden',
49769     allowBlank:     true,
49770     readOnly:         true,
49771     
49772  
49773     /**
49774      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49775      */
49776     focusClass : undefined,
49777     /**
49778      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49779      */
49780     fieldClass: 'x-form-field',
49781     
49782      /**
49783      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49784      */
49785     valueRenderer: undefined,
49786     
49787     width: 100,
49788     /**
49789      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49790      * {tag: "input", type: "checkbox", autocomplete: "off"})
49791      */
49792      
49793  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49794  
49795     closable : false,
49796     
49797     onResize : function(){
49798         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49799         
49800     },
49801
49802     initEvents : function(){
49803         // Roo.form.Checkbox.superclass.initEvents.call(this);
49804         // has no events...
49805         
49806         if(this.closable){
49807             this.closeEl.on('click', this.onClose, this);
49808         }
49809        
49810     },
49811
49812
49813     getResizeEl : function(){
49814         return this.wrap;
49815     },
49816
49817     getPositionEl : function(){
49818         return this.wrap;
49819     },
49820
49821     // private
49822     onRender : function(ct, position){
49823         
49824         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49825         //if(this.inputValue !== undefined){
49826         this.wrap = this.el.wrap();
49827         
49828         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49829         
49830         if(this.closable){
49831             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49832         }
49833         
49834         if (this.bodyStyle) {
49835             this.viewEl.applyStyles(this.bodyStyle);
49836         }
49837         //this.viewEl.setStyle('padding', '2px');
49838         
49839         this.setValue(this.value);
49840         
49841     },
49842 /*
49843     // private
49844     initValue : Roo.emptyFn,
49845
49846   */
49847
49848         // private
49849     onClick : function(){
49850         
49851     },
49852
49853     /**
49854      * Sets the checked state of the checkbox.
49855      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49856      */
49857     setValue : function(v){
49858         this.value = v;
49859         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49860         // this might be called before we have a dom element..
49861         if (!this.viewEl) {
49862             return;
49863         }
49864         this.viewEl.dom.innerHTML = html;
49865         Roo.form.DisplayField.superclass.setValue.call(this, v);
49866
49867     },
49868     
49869     onClose : function(e)
49870     {
49871         e.preventDefault();
49872         
49873         this.fireEvent('close', this);
49874     }
49875 });/*
49876  * 
49877  * Licence- LGPL
49878  * 
49879  */
49880
49881 /**
49882  * @class Roo.form.DayPicker
49883  * @extends Roo.form.Field
49884  * A Day picker show [M] [T] [W] ....
49885  * @constructor
49886  * Creates a new Day Picker
49887  * @param {Object} config Configuration options
49888  */
49889 Roo.form.DayPicker= function(config){
49890     Roo.form.DayPicker.superclass.constructor.call(this, config);
49891      
49892 };
49893
49894 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49895     /**
49896      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49897      */
49898     focusClass : undefined,
49899     /**
49900      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49901      */
49902     fieldClass: "x-form-field",
49903    
49904     /**
49905      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49906      * {tag: "input", type: "checkbox", autocomplete: "off"})
49907      */
49908     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49909     
49910    
49911     actionMode : 'viewEl', 
49912     //
49913     // private
49914  
49915     inputType : 'hidden',
49916     
49917      
49918     inputElement: false, // real input element?
49919     basedOn: false, // ????
49920     
49921     isFormField: true, // not sure where this is needed!!!!
49922
49923     onResize : function(){
49924         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49925         if(!this.boxLabel){
49926             this.el.alignTo(this.wrap, 'c-c');
49927         }
49928     },
49929
49930     initEvents : function(){
49931         Roo.form.Checkbox.superclass.initEvents.call(this);
49932         this.el.on("click", this.onClick,  this);
49933         this.el.on("change", this.onClick,  this);
49934     },
49935
49936
49937     getResizeEl : function(){
49938         return this.wrap;
49939     },
49940
49941     getPositionEl : function(){
49942         return this.wrap;
49943     },
49944
49945     
49946     // private
49947     onRender : function(ct, position){
49948         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
49949        
49950         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
49951         
49952         var r1 = '<table><tr>';
49953         var r2 = '<tr class="x-form-daypick-icons">';
49954         for (var i=0; i < 7; i++) {
49955             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
49956             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
49957         }
49958         
49959         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
49960         viewEl.select('img').on('click', this.onClick, this);
49961         this.viewEl = viewEl;   
49962         
49963         
49964         // this will not work on Chrome!!!
49965         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
49966         this.el.on('propertychange', this.setFromHidden,  this);  //ie
49967         
49968         
49969           
49970
49971     },
49972
49973     // private
49974     initValue : Roo.emptyFn,
49975
49976     /**
49977      * Returns the checked state of the checkbox.
49978      * @return {Boolean} True if checked, else false
49979      */
49980     getValue : function(){
49981         return this.el.dom.value;
49982         
49983     },
49984
49985         // private
49986     onClick : function(e){ 
49987         //this.setChecked(!this.checked);
49988         Roo.get(e.target).toggleClass('x-menu-item-checked');
49989         this.refreshValue();
49990         //if(this.el.dom.checked != this.checked){
49991         //    this.setValue(this.el.dom.checked);
49992        // }
49993     },
49994     
49995     // private
49996     refreshValue : function()
49997     {
49998         var val = '';
49999         this.viewEl.select('img',true).each(function(e,i,n)  {
50000             val += e.is(".x-menu-item-checked") ? String(n) : '';
50001         });
50002         this.setValue(val, true);
50003     },
50004
50005     /**
50006      * Sets the checked state of the checkbox.
50007      * On is always based on a string comparison between inputValue and the param.
50008      * @param {Boolean/String} value - the value to set 
50009      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50010      */
50011     setValue : function(v,suppressEvent){
50012         if (!this.el.dom) {
50013             return;
50014         }
50015         var old = this.el.dom.value ;
50016         this.el.dom.value = v;
50017         if (suppressEvent) {
50018             return ;
50019         }
50020          
50021         // update display..
50022         this.viewEl.select('img',true).each(function(e,i,n)  {
50023             
50024             var on = e.is(".x-menu-item-checked");
50025             var newv = v.indexOf(String(n)) > -1;
50026             if (on != newv) {
50027                 e.toggleClass('x-menu-item-checked');
50028             }
50029             
50030         });
50031         
50032         
50033         this.fireEvent('change', this, v, old);
50034         
50035         
50036     },
50037    
50038     // handle setting of hidden value by some other method!!?!?
50039     setFromHidden: function()
50040     {
50041         if(!this.el){
50042             return;
50043         }
50044         //console.log("SET FROM HIDDEN");
50045         //alert('setFrom hidden');
50046         this.setValue(this.el.dom.value);
50047     },
50048     
50049     onDestroy : function()
50050     {
50051         if(this.viewEl){
50052             Roo.get(this.viewEl).remove();
50053         }
50054          
50055         Roo.form.DayPicker.superclass.onDestroy.call(this);
50056     }
50057
50058 });/*
50059  * RooJS Library 1.1.1
50060  * Copyright(c) 2008-2011  Alan Knowles
50061  *
50062  * License - LGPL
50063  */
50064  
50065
50066 /**
50067  * @class Roo.form.ComboCheck
50068  * @extends Roo.form.ComboBox
50069  * A combobox for multiple select items.
50070  *
50071  * FIXME - could do with a reset button..
50072  * 
50073  * @constructor
50074  * Create a new ComboCheck
50075  * @param {Object} config Configuration options
50076  */
50077 Roo.form.ComboCheck = function(config){
50078     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50079     // should verify some data...
50080     // like
50081     // hiddenName = required..
50082     // displayField = required
50083     // valudField == required
50084     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50085     var _t = this;
50086     Roo.each(req, function(e) {
50087         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50088             throw "Roo.form.ComboCheck : missing value for: " + e;
50089         }
50090     });
50091     
50092     
50093 };
50094
50095 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50096      
50097      
50098     editable : false,
50099      
50100     selectedClass: 'x-menu-item-checked', 
50101     
50102     // private
50103     onRender : function(ct, position){
50104         var _t = this;
50105         
50106         
50107         
50108         if(!this.tpl){
50109             var cls = 'x-combo-list';
50110
50111             
50112             this.tpl =  new Roo.Template({
50113                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50114                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50115                    '<span>{' + this.displayField + '}</span>' +
50116                     '</div>' 
50117                 
50118             });
50119         }
50120  
50121         
50122         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50123         this.view.singleSelect = false;
50124         this.view.multiSelect = true;
50125         this.view.toggleSelect = true;
50126         this.pageTb.add(new Roo.Toolbar.Fill(), {
50127             
50128             text: 'Done',
50129             handler: function()
50130             {
50131                 _t.collapse();
50132             }
50133         });
50134     },
50135     
50136     onViewOver : function(e, t){
50137         // do nothing...
50138         return;
50139         
50140     },
50141     
50142     onViewClick : function(doFocus,index){
50143         return;
50144         
50145     },
50146     select: function () {
50147         //Roo.log("SELECT CALLED");
50148     },
50149      
50150     selectByValue : function(xv, scrollIntoView){
50151         var ar = this.getValueArray();
50152         var sels = [];
50153         
50154         Roo.each(ar, function(v) {
50155             if(v === undefined || v === null){
50156                 return;
50157             }
50158             var r = this.findRecord(this.valueField, v);
50159             if(r){
50160                 sels.push(this.store.indexOf(r))
50161                 
50162             }
50163         },this);
50164         this.view.select(sels);
50165         return false;
50166     },
50167     
50168     
50169     
50170     onSelect : function(record, index){
50171        // Roo.log("onselect Called");
50172        // this is only called by the clear button now..
50173         this.view.clearSelections();
50174         this.setValue('[]');
50175         if (this.value != this.valueBefore) {
50176             this.fireEvent('change', this, this.value, this.valueBefore);
50177             this.valueBefore = this.value;
50178         }
50179     },
50180     getValueArray : function()
50181     {
50182         var ar = [] ;
50183         
50184         try {
50185             //Roo.log(this.value);
50186             if (typeof(this.value) == 'undefined') {
50187                 return [];
50188             }
50189             var ar = Roo.decode(this.value);
50190             return  ar instanceof Array ? ar : []; //?? valid?
50191             
50192         } catch(e) {
50193             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50194             return [];
50195         }
50196          
50197     },
50198     expand : function ()
50199     {
50200         
50201         Roo.form.ComboCheck.superclass.expand.call(this);
50202         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50203         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50204         
50205
50206     },
50207     
50208     collapse : function(){
50209         Roo.form.ComboCheck.superclass.collapse.call(this);
50210         var sl = this.view.getSelectedIndexes();
50211         var st = this.store;
50212         var nv = [];
50213         var tv = [];
50214         var r;
50215         Roo.each(sl, function(i) {
50216             r = st.getAt(i);
50217             nv.push(r.get(this.valueField));
50218         },this);
50219         this.setValue(Roo.encode(nv));
50220         if (this.value != this.valueBefore) {
50221
50222             this.fireEvent('change', this, this.value, this.valueBefore);
50223             this.valueBefore = this.value;
50224         }
50225         
50226     },
50227     
50228     setValue : function(v){
50229         // Roo.log(v);
50230         this.value = v;
50231         
50232         var vals = this.getValueArray();
50233         var tv = [];
50234         Roo.each(vals, function(k) {
50235             var r = this.findRecord(this.valueField, k);
50236             if(r){
50237                 tv.push(r.data[this.displayField]);
50238             }else if(this.valueNotFoundText !== undefined){
50239                 tv.push( this.valueNotFoundText );
50240             }
50241         },this);
50242        // Roo.log(tv);
50243         
50244         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50245         this.hiddenField.value = v;
50246         this.value = v;
50247     }
50248     
50249 });/*
50250  * Based on:
50251  * Ext JS Library 1.1.1
50252  * Copyright(c) 2006-2007, Ext JS, LLC.
50253  *
50254  * Originally Released Under LGPL - original licence link has changed is not relivant.
50255  *
50256  * Fork - LGPL
50257  * <script type="text/javascript">
50258  */
50259  
50260 /**
50261  * @class Roo.form.Signature
50262  * @extends Roo.form.Field
50263  * Signature field.  
50264  * @constructor
50265  * 
50266  * @param {Object} config Configuration options
50267  */
50268
50269 Roo.form.Signature = function(config){
50270     Roo.form.Signature.superclass.constructor.call(this, config);
50271     
50272     this.addEvents({// not in used??
50273          /**
50274          * @event confirm
50275          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50276              * @param {Roo.form.Signature} combo This combo box
50277              */
50278         'confirm' : true,
50279         /**
50280          * @event reset
50281          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50282              * @param {Roo.form.ComboBox} combo This combo box
50283              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50284              */
50285         'reset' : true
50286     });
50287 };
50288
50289 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50290     /**
50291      * @cfg {Object} labels Label to use when rendering a form.
50292      * defaults to 
50293      * labels : { 
50294      *      clear : "Clear",
50295      *      confirm : "Confirm"
50296      *  }
50297      */
50298     labels : { 
50299         clear : "Clear",
50300         confirm : "Confirm"
50301     },
50302     /**
50303      * @cfg {Number} width The signature panel width (defaults to 300)
50304      */
50305     width: 300,
50306     /**
50307      * @cfg {Number} height The signature panel height (defaults to 100)
50308      */
50309     height : 100,
50310     /**
50311      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50312      */
50313     allowBlank : false,
50314     
50315     //private
50316     // {Object} signPanel The signature SVG panel element (defaults to {})
50317     signPanel : {},
50318     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50319     isMouseDown : false,
50320     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50321     isConfirmed : false,
50322     // {String} signatureTmp SVG mapping string (defaults to empty string)
50323     signatureTmp : '',
50324     
50325     
50326     defaultAutoCreate : { // modified by initCompnoent..
50327         tag: "input",
50328         type:"hidden"
50329     },
50330
50331     // private
50332     onRender : function(ct, position){
50333         
50334         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50335         
50336         this.wrap = this.el.wrap({
50337             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50338         });
50339         
50340         this.createToolbar(this);
50341         this.signPanel = this.wrap.createChild({
50342                 tag: 'div',
50343                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50344             }, this.el
50345         );
50346             
50347         this.svgID = Roo.id();
50348         this.svgEl = this.signPanel.createChild({
50349               xmlns : 'http://www.w3.org/2000/svg',
50350               tag : 'svg',
50351               id : this.svgID + "-svg",
50352               width: this.width,
50353               height: this.height,
50354               viewBox: '0 0 '+this.width+' '+this.height,
50355               cn : [
50356                 {
50357                     tag: "rect",
50358                     id: this.svgID + "-svg-r",
50359                     width: this.width,
50360                     height: this.height,
50361                     fill: "#ffa"
50362                 },
50363                 {
50364                     tag: "line",
50365                     id: this.svgID + "-svg-l",
50366                     x1: "0", // start
50367                     y1: (this.height*0.8), // start set the line in 80% of height
50368                     x2: this.width, // end
50369                     y2: (this.height*0.8), // end set the line in 80% of height
50370                     'stroke': "#666",
50371                     'stroke-width': "1",
50372                     'stroke-dasharray': "3",
50373                     'shape-rendering': "crispEdges",
50374                     'pointer-events': "none"
50375                 },
50376                 {
50377                     tag: "path",
50378                     id: this.svgID + "-svg-p",
50379                     'stroke': "navy",
50380                     'stroke-width': "3",
50381                     'fill': "none",
50382                     'pointer-events': 'none'
50383                 }
50384               ]
50385         });
50386         this.createSVG();
50387         this.svgBox = this.svgEl.dom.getScreenCTM();
50388     },
50389     createSVG : function(){ 
50390         var svg = this.signPanel;
50391         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50392         var t = this;
50393
50394         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50395         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50396         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50397         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50398         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50399         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50400         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50401         
50402     },
50403     isTouchEvent : function(e){
50404         return e.type.match(/^touch/);
50405     },
50406     getCoords : function (e) {
50407         var pt    = this.svgEl.dom.createSVGPoint();
50408         pt.x = e.clientX; 
50409         pt.y = e.clientY;
50410         if (this.isTouchEvent(e)) {
50411             pt.x =  e.targetTouches[0].clientX;
50412             pt.y = e.targetTouches[0].clientY;
50413         }
50414         var a = this.svgEl.dom.getScreenCTM();
50415         var b = a.inverse();
50416         var mx = pt.matrixTransform(b);
50417         return mx.x + ',' + mx.y;
50418     },
50419     //mouse event headler 
50420     down : function (e) {
50421         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50422         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50423         
50424         this.isMouseDown = true;
50425         
50426         e.preventDefault();
50427     },
50428     move : function (e) {
50429         if (this.isMouseDown) {
50430             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50431             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50432         }
50433         
50434         e.preventDefault();
50435     },
50436     up : function (e) {
50437         this.isMouseDown = false;
50438         var sp = this.signatureTmp.split(' ');
50439         
50440         if(sp.length > 1){
50441             if(!sp[sp.length-2].match(/^L/)){
50442                 sp.pop();
50443                 sp.pop();
50444                 sp.push("");
50445                 this.signatureTmp = sp.join(" ");
50446             }
50447         }
50448         if(this.getValue() != this.signatureTmp){
50449             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50450             this.isConfirmed = false;
50451         }
50452         e.preventDefault();
50453     },
50454     
50455     /**
50456      * Protected method that will not generally be called directly. It
50457      * is called when the editor creates its toolbar. Override this method if you need to
50458      * add custom toolbar buttons.
50459      * @param {HtmlEditor} editor
50460      */
50461     createToolbar : function(editor){
50462          function btn(id, toggle, handler){
50463             var xid = fid + '-'+ id ;
50464             return {
50465                 id : xid,
50466                 cmd : id,
50467                 cls : 'x-btn-icon x-edit-'+id,
50468                 enableToggle:toggle !== false,
50469                 scope: editor, // was editor...
50470                 handler:handler||editor.relayBtnCmd,
50471                 clickEvent:'mousedown',
50472                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50473                 tabIndex:-1
50474             };
50475         }
50476         
50477         
50478         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50479         this.tb = tb;
50480         this.tb.add(
50481            {
50482                 cls : ' x-signature-btn x-signature-'+id,
50483                 scope: editor, // was editor...
50484                 handler: this.reset,
50485                 clickEvent:'mousedown',
50486                 text: this.labels.clear
50487             },
50488             {
50489                  xtype : 'Fill',
50490                  xns: Roo.Toolbar
50491             }, 
50492             {
50493                 cls : '  x-signature-btn x-signature-'+id,
50494                 scope: editor, // was editor...
50495                 handler: this.confirmHandler,
50496                 clickEvent:'mousedown',
50497                 text: this.labels.confirm
50498             }
50499         );
50500     
50501     },
50502     //public
50503     /**
50504      * when user is clicked confirm then show this image.....
50505      * 
50506      * @return {String} Image Data URI
50507      */
50508     getImageDataURI : function(){
50509         var svg = this.svgEl.dom.parentNode.innerHTML;
50510         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50511         return src; 
50512     },
50513     /**
50514      * 
50515      * @return {Boolean} this.isConfirmed
50516      */
50517     getConfirmed : function(){
50518         return this.isConfirmed;
50519     },
50520     /**
50521      * 
50522      * @return {Number} this.width
50523      */
50524     getWidth : function(){
50525         return this.width;
50526     },
50527     /**
50528      * 
50529      * @return {Number} this.height
50530      */
50531     getHeight : function(){
50532         return this.height;
50533     },
50534     // private
50535     getSignature : function(){
50536         return this.signatureTmp;
50537     },
50538     // private
50539     reset : function(){
50540         this.signatureTmp = '';
50541         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50542         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50543         this.isConfirmed = false;
50544         Roo.form.Signature.superclass.reset.call(this);
50545     },
50546     setSignature : function(s){
50547         this.signatureTmp = s;
50548         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50549         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50550         this.setValue(s);
50551         this.isConfirmed = false;
50552         Roo.form.Signature.superclass.reset.call(this);
50553     }, 
50554     test : function(){
50555 //        Roo.log(this.signPanel.dom.contentWindow.up())
50556     },
50557     //private
50558     setConfirmed : function(){
50559         
50560         
50561         
50562 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50563     },
50564     // private
50565     confirmHandler : function(){
50566         if(!this.getSignature()){
50567             return;
50568         }
50569         
50570         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50571         this.setValue(this.getSignature());
50572         this.isConfirmed = true;
50573         
50574         this.fireEvent('confirm', this);
50575     },
50576     // private
50577     // Subclasses should provide the validation implementation by overriding this
50578     validateValue : function(value){
50579         if(this.allowBlank){
50580             return true;
50581         }
50582         
50583         if(this.isConfirmed){
50584             return true;
50585         }
50586         return false;
50587     }
50588 });/*
50589  * Based on:
50590  * Ext JS Library 1.1.1
50591  * Copyright(c) 2006-2007, Ext JS, LLC.
50592  *
50593  * Originally Released Under LGPL - original licence link has changed is not relivant.
50594  *
50595  * Fork - LGPL
50596  * <script type="text/javascript">
50597  */
50598  
50599
50600 /**
50601  * @class Roo.form.ComboBox
50602  * @extends Roo.form.TriggerField
50603  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50604  * @constructor
50605  * Create a new ComboBox.
50606  * @param {Object} config Configuration options
50607  */
50608 Roo.form.Select = function(config){
50609     Roo.form.Select.superclass.constructor.call(this, config);
50610      
50611 };
50612
50613 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50614     /**
50615      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50616      */
50617     /**
50618      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50619      * rendering into an Roo.Editor, defaults to false)
50620      */
50621     /**
50622      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50623      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50624      */
50625     /**
50626      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50627      */
50628     /**
50629      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50630      * the dropdown list (defaults to undefined, with no header element)
50631      */
50632
50633      /**
50634      * @cfg {String/Roo.Template} tpl The template to use to render the output
50635      */
50636      
50637     // private
50638     defaultAutoCreate : {tag: "select"  },
50639     /**
50640      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50641      */
50642     listWidth: undefined,
50643     /**
50644      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50645      * mode = 'remote' or 'text' if mode = 'local')
50646      */
50647     displayField: undefined,
50648     /**
50649      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50650      * mode = 'remote' or 'value' if mode = 'local'). 
50651      * Note: use of a valueField requires the user make a selection
50652      * in order for a value to be mapped.
50653      */
50654     valueField: undefined,
50655     
50656     
50657     /**
50658      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50659      * field's data value (defaults to the underlying DOM element's name)
50660      */
50661     hiddenName: undefined,
50662     /**
50663      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50664      */
50665     listClass: '',
50666     /**
50667      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50668      */
50669     selectedClass: 'x-combo-selected',
50670     /**
50671      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50672      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50673      * which displays a downward arrow icon).
50674      */
50675     triggerClass : 'x-form-arrow-trigger',
50676     /**
50677      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50678      */
50679     shadow:'sides',
50680     /**
50681      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50682      * anchor positions (defaults to 'tl-bl')
50683      */
50684     listAlign: 'tl-bl?',
50685     /**
50686      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50687      */
50688     maxHeight: 300,
50689     /**
50690      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50691      * query specified by the allQuery config option (defaults to 'query')
50692      */
50693     triggerAction: 'query',
50694     /**
50695      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50696      * (defaults to 4, does not apply if editable = false)
50697      */
50698     minChars : 4,
50699     /**
50700      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50701      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50702      */
50703     typeAhead: false,
50704     /**
50705      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50706      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50707      */
50708     queryDelay: 500,
50709     /**
50710      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50711      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50712      */
50713     pageSize: 0,
50714     /**
50715      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50716      * when editable = true (defaults to false)
50717      */
50718     selectOnFocus:false,
50719     /**
50720      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50721      */
50722     queryParam: 'query',
50723     /**
50724      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50725      * when mode = 'remote' (defaults to 'Loading...')
50726      */
50727     loadingText: 'Loading...',
50728     /**
50729      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50730      */
50731     resizable: false,
50732     /**
50733      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50734      */
50735     handleHeight : 8,
50736     /**
50737      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50738      * traditional select (defaults to true)
50739      */
50740     editable: true,
50741     /**
50742      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50743      */
50744     allQuery: '',
50745     /**
50746      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50747      */
50748     mode: 'remote',
50749     /**
50750      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50751      * listWidth has a higher value)
50752      */
50753     minListWidth : 70,
50754     /**
50755      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50756      * allow the user to set arbitrary text into the field (defaults to false)
50757      */
50758     forceSelection:false,
50759     /**
50760      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50761      * if typeAhead = true (defaults to 250)
50762      */
50763     typeAheadDelay : 250,
50764     /**
50765      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50766      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50767      */
50768     valueNotFoundText : undefined,
50769     
50770     /**
50771      * @cfg {String} defaultValue The value displayed after loading the store.
50772      */
50773     defaultValue: '',
50774     
50775     /**
50776      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50777      */
50778     blockFocus : false,
50779     
50780     /**
50781      * @cfg {Boolean} disableClear Disable showing of clear button.
50782      */
50783     disableClear : false,
50784     /**
50785      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50786      */
50787     alwaysQuery : false,
50788     
50789     //private
50790     addicon : false,
50791     editicon: false,
50792     
50793     // element that contains real text value.. (when hidden is used..)
50794      
50795     // private
50796     onRender : function(ct, position){
50797         Roo.form.Field.prototype.onRender.call(this, ct, position);
50798         
50799         if(this.store){
50800             this.store.on('beforeload', this.onBeforeLoad, this);
50801             this.store.on('load', this.onLoad, this);
50802             this.store.on('loadexception', this.onLoadException, this);
50803             this.store.load({});
50804         }
50805         
50806         
50807         
50808     },
50809
50810     // private
50811     initEvents : function(){
50812         //Roo.form.ComboBox.superclass.initEvents.call(this);
50813  
50814     },
50815
50816     onDestroy : function(){
50817        
50818         if(this.store){
50819             this.store.un('beforeload', this.onBeforeLoad, this);
50820             this.store.un('load', this.onLoad, this);
50821             this.store.un('loadexception', this.onLoadException, this);
50822         }
50823         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50824     },
50825
50826     // private
50827     fireKey : function(e){
50828         if(e.isNavKeyPress() && !this.list.isVisible()){
50829             this.fireEvent("specialkey", this, e);
50830         }
50831     },
50832
50833     // private
50834     onResize: function(w, h){
50835         
50836         return; 
50837     
50838         
50839     },
50840
50841     /**
50842      * Allow or prevent the user from directly editing the field text.  If false is passed,
50843      * the user will only be able to select from the items defined in the dropdown list.  This method
50844      * is the runtime equivalent of setting the 'editable' config option at config time.
50845      * @param {Boolean} value True to allow the user to directly edit the field text
50846      */
50847     setEditable : function(value){
50848          
50849     },
50850
50851     // private
50852     onBeforeLoad : function(){
50853         
50854         Roo.log("Select before load");
50855         return;
50856     
50857         this.innerList.update(this.loadingText ?
50858                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50859         //this.restrictHeight();
50860         this.selectedIndex = -1;
50861     },
50862
50863     // private
50864     onLoad : function(){
50865
50866     
50867         var dom = this.el.dom;
50868         dom.innerHTML = '';
50869          var od = dom.ownerDocument;
50870          
50871         if (this.emptyText) {
50872             var op = od.createElement('option');
50873             op.setAttribute('value', '');
50874             op.innerHTML = String.format('{0}', this.emptyText);
50875             dom.appendChild(op);
50876         }
50877         if(this.store.getCount() > 0){
50878            
50879             var vf = this.valueField;
50880             var df = this.displayField;
50881             this.store.data.each(function(r) {
50882                 // which colmsn to use... testing - cdoe / title..
50883                 var op = od.createElement('option');
50884                 op.setAttribute('value', r.data[vf]);
50885                 op.innerHTML = String.format('{0}', r.data[df]);
50886                 dom.appendChild(op);
50887             });
50888             if (typeof(this.defaultValue != 'undefined')) {
50889                 this.setValue(this.defaultValue);
50890             }
50891             
50892              
50893         }else{
50894             //this.onEmptyResults();
50895         }
50896         //this.el.focus();
50897     },
50898     // private
50899     onLoadException : function()
50900     {
50901         dom.innerHTML = '';
50902             
50903         Roo.log("Select on load exception");
50904         return;
50905     
50906         this.collapse();
50907         Roo.log(this.store.reader.jsonData);
50908         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50909             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50910         }
50911         
50912         
50913     },
50914     // private
50915     onTypeAhead : function(){
50916          
50917     },
50918
50919     // private
50920     onSelect : function(record, index){
50921         Roo.log('on select?');
50922         return;
50923         if(this.fireEvent('beforeselect', this, record, index) !== false){
50924             this.setFromData(index > -1 ? record.data : false);
50925             this.collapse();
50926             this.fireEvent('select', this, record, index);
50927         }
50928     },
50929
50930     /**
50931      * Returns the currently selected field value or empty string if no value is set.
50932      * @return {String} value The selected value
50933      */
50934     getValue : function(){
50935         var dom = this.el.dom;
50936         this.value = dom.options[dom.selectedIndex].value;
50937         return this.value;
50938         
50939     },
50940
50941     /**
50942      * Clears any text/value currently set in the field
50943      */
50944     clearValue : function(){
50945         this.value = '';
50946         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
50947         
50948     },
50949
50950     /**
50951      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
50952      * will be displayed in the field.  If the value does not match the data value of an existing item,
50953      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
50954      * Otherwise the field will be blank (although the value will still be set).
50955      * @param {String} value The value to match
50956      */
50957     setValue : function(v){
50958         var d = this.el.dom;
50959         for (var i =0; i < d.options.length;i++) {
50960             if (v == d.options[i].value) {
50961                 d.selectedIndex = i;
50962                 this.value = v;
50963                 return;
50964             }
50965         }
50966         this.clearValue();
50967     },
50968     /**
50969      * @property {Object} the last set data for the element
50970      */
50971     
50972     lastData : false,
50973     /**
50974      * Sets the value of the field based on a object which is related to the record format for the store.
50975      * @param {Object} value the value to set as. or false on reset?
50976      */
50977     setFromData : function(o){
50978         Roo.log('setfrom data?');
50979          
50980         
50981         
50982     },
50983     // private
50984     reset : function(){
50985         this.clearValue();
50986     },
50987     // private
50988     findRecord : function(prop, value){
50989         
50990         return false;
50991     
50992         var record;
50993         if(this.store.getCount() > 0){
50994             this.store.each(function(r){
50995                 if(r.data[prop] == value){
50996                     record = r;
50997                     return false;
50998                 }
50999                 return true;
51000             });
51001         }
51002         return record;
51003     },
51004     
51005     getName: function()
51006     {
51007         // returns hidden if it's set..
51008         if (!this.rendered) {return ''};
51009         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51010         
51011     },
51012      
51013
51014     
51015
51016     // private
51017     onEmptyResults : function(){
51018         Roo.log('empty results');
51019         //this.collapse();
51020     },
51021
51022     /**
51023      * Returns true if the dropdown list is expanded, else false.
51024      */
51025     isExpanded : function(){
51026         return false;
51027     },
51028
51029     /**
51030      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51031      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51032      * @param {String} value The data value of the item to select
51033      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51034      * selected item if it is not currently in view (defaults to true)
51035      * @return {Boolean} True if the value matched an item in the list, else false
51036      */
51037     selectByValue : function(v, scrollIntoView){
51038         Roo.log('select By Value');
51039         return false;
51040     
51041         if(v !== undefined && v !== null){
51042             var r = this.findRecord(this.valueField || this.displayField, v);
51043             if(r){
51044                 this.select(this.store.indexOf(r), scrollIntoView);
51045                 return true;
51046             }
51047         }
51048         return false;
51049     },
51050
51051     /**
51052      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51053      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51054      * @param {Number} index The zero-based index of the list item to select
51055      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51056      * selected item if it is not currently in view (defaults to true)
51057      */
51058     select : function(index, scrollIntoView){
51059         Roo.log('select ');
51060         return  ;
51061         
51062         this.selectedIndex = index;
51063         this.view.select(index);
51064         if(scrollIntoView !== false){
51065             var el = this.view.getNode(index);
51066             if(el){
51067                 this.innerList.scrollChildIntoView(el, false);
51068             }
51069         }
51070     },
51071
51072       
51073
51074     // private
51075     validateBlur : function(){
51076         
51077         return;
51078         
51079     },
51080
51081     // private
51082     initQuery : function(){
51083         this.doQuery(this.getRawValue());
51084     },
51085
51086     // private
51087     doForce : function(){
51088         if(this.el.dom.value.length > 0){
51089             this.el.dom.value =
51090                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51091              
51092         }
51093     },
51094
51095     /**
51096      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51097      * query allowing the query action to be canceled if needed.
51098      * @param {String} query The SQL query to execute
51099      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51100      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51101      * saved in the current store (defaults to false)
51102      */
51103     doQuery : function(q, forceAll){
51104         
51105         Roo.log('doQuery?');
51106         if(q === undefined || q === null){
51107             q = '';
51108         }
51109         var qe = {
51110             query: q,
51111             forceAll: forceAll,
51112             combo: this,
51113             cancel:false
51114         };
51115         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51116             return false;
51117         }
51118         q = qe.query;
51119         forceAll = qe.forceAll;
51120         if(forceAll === true || (q.length >= this.minChars)){
51121             if(this.lastQuery != q || this.alwaysQuery){
51122                 this.lastQuery = q;
51123                 if(this.mode == 'local'){
51124                     this.selectedIndex = -1;
51125                     if(forceAll){
51126                         this.store.clearFilter();
51127                     }else{
51128                         this.store.filter(this.displayField, q);
51129                     }
51130                     this.onLoad();
51131                 }else{
51132                     this.store.baseParams[this.queryParam] = q;
51133                     this.store.load({
51134                         params: this.getParams(q)
51135                     });
51136                     this.expand();
51137                 }
51138             }else{
51139                 this.selectedIndex = -1;
51140                 this.onLoad();   
51141             }
51142         }
51143     },
51144
51145     // private
51146     getParams : function(q){
51147         var p = {};
51148         //p[this.queryParam] = q;
51149         if(this.pageSize){
51150             p.start = 0;
51151             p.limit = this.pageSize;
51152         }
51153         return p;
51154     },
51155
51156     /**
51157      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51158      */
51159     collapse : function(){
51160         
51161     },
51162
51163     // private
51164     collapseIf : function(e){
51165         
51166     },
51167
51168     /**
51169      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51170      */
51171     expand : function(){
51172         
51173     } ,
51174
51175     // private
51176      
51177
51178     /** 
51179     * @cfg {Boolean} grow 
51180     * @hide 
51181     */
51182     /** 
51183     * @cfg {Number} growMin 
51184     * @hide 
51185     */
51186     /** 
51187     * @cfg {Number} growMax 
51188     * @hide 
51189     */
51190     /**
51191      * @hide
51192      * @method autoSize
51193      */
51194     
51195     setWidth : function()
51196     {
51197         
51198     },
51199     getResizeEl : function(){
51200         return this.el;
51201     }
51202 });//<script type="text/javasscript">
51203  
51204
51205 /**
51206  * @class Roo.DDView
51207  * A DnD enabled version of Roo.View.
51208  * @param {Element/String} container The Element in which to create the View.
51209  * @param {String} tpl The template string used to create the markup for each element of the View
51210  * @param {Object} config The configuration properties. These include all the config options of
51211  * {@link Roo.View} plus some specific to this class.<br>
51212  * <p>
51213  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51214  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51215  * <p>
51216  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51217 .x-view-drag-insert-above {
51218         border-top:1px dotted #3366cc;
51219 }
51220 .x-view-drag-insert-below {
51221         border-bottom:1px dotted #3366cc;
51222 }
51223 </code></pre>
51224  * 
51225  */
51226  
51227 Roo.DDView = function(container, tpl, config) {
51228     Roo.DDView.superclass.constructor.apply(this, arguments);
51229     this.getEl().setStyle("outline", "0px none");
51230     this.getEl().unselectable();
51231     if (this.dragGroup) {
51232                 this.setDraggable(this.dragGroup.split(","));
51233     }
51234     if (this.dropGroup) {
51235                 this.setDroppable(this.dropGroup.split(","));
51236     }
51237     if (this.deletable) {
51238         this.setDeletable();
51239     }
51240     this.isDirtyFlag = false;
51241         this.addEvents({
51242                 "drop" : true
51243         });
51244 };
51245
51246 Roo.extend(Roo.DDView, Roo.View, {
51247 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51248 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51249 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51250 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51251
51252         isFormField: true,
51253
51254         reset: Roo.emptyFn,
51255         
51256         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51257
51258         validate: function() {
51259                 return true;
51260         },
51261         
51262         destroy: function() {
51263                 this.purgeListeners();
51264                 this.getEl.removeAllListeners();
51265                 this.getEl().remove();
51266                 if (this.dragZone) {
51267                         if (this.dragZone.destroy) {
51268                                 this.dragZone.destroy();
51269                         }
51270                 }
51271                 if (this.dropZone) {
51272                         if (this.dropZone.destroy) {
51273                                 this.dropZone.destroy();
51274                         }
51275                 }
51276         },
51277
51278 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51279         getName: function() {
51280                 return this.name;
51281         },
51282
51283 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51284         setValue: function(v) {
51285                 if (!this.store) {
51286                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51287                 }
51288                 var data = {};
51289                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51290                 this.store.proxy = new Roo.data.MemoryProxy(data);
51291                 this.store.load();
51292         },
51293
51294 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51295         getValue: function() {
51296                 var result = '(';
51297                 this.store.each(function(rec) {
51298                         result += rec.id + ',';
51299                 });
51300                 return result.substr(0, result.length - 1) + ')';
51301         },
51302         
51303         getIds: function() {
51304                 var i = 0, result = new Array(this.store.getCount());
51305                 this.store.each(function(rec) {
51306                         result[i++] = rec.id;
51307                 });
51308                 return result;
51309         },
51310         
51311         isDirty: function() {
51312                 return this.isDirtyFlag;
51313         },
51314
51315 /**
51316  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51317  *      whole Element becomes the target, and this causes the drop gesture to append.
51318  */
51319     getTargetFromEvent : function(e) {
51320                 var target = e.getTarget();
51321                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51322                 target = target.parentNode;
51323                 }
51324                 if (!target) {
51325                         target = this.el.dom.lastChild || this.el.dom;
51326                 }
51327                 return target;
51328     },
51329
51330 /**
51331  *      Create the drag data which consists of an object which has the property "ddel" as
51332  *      the drag proxy element. 
51333  */
51334     getDragData : function(e) {
51335         var target = this.findItemFromChild(e.getTarget());
51336                 if(target) {
51337                         this.handleSelection(e);
51338                         var selNodes = this.getSelectedNodes();
51339             var dragData = {
51340                 source: this,
51341                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51342                 nodes: selNodes,
51343                 records: []
51344                         };
51345                         var selectedIndices = this.getSelectedIndexes();
51346                         for (var i = 0; i < selectedIndices.length; i++) {
51347                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51348                         }
51349                         if (selNodes.length == 1) {
51350                                 dragData.ddel = target.cloneNode(true); // the div element
51351                         } else {
51352                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51353                                 div.className = 'multi-proxy';
51354                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51355                                         div.appendChild(selNodes[i].cloneNode(true));
51356                                 }
51357                                 dragData.ddel = div;
51358                         }
51359             //console.log(dragData)
51360             //console.log(dragData.ddel.innerHTML)
51361                         return dragData;
51362                 }
51363         //console.log('nodragData')
51364                 return false;
51365     },
51366     
51367 /**     Specify to which ddGroup items in this DDView may be dragged. */
51368     setDraggable: function(ddGroup) {
51369         if (ddGroup instanceof Array) {
51370                 Roo.each(ddGroup, this.setDraggable, this);
51371                 return;
51372         }
51373         if (this.dragZone) {
51374                 this.dragZone.addToGroup(ddGroup);
51375         } else {
51376                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51377                                 containerScroll: true,
51378                                 ddGroup: ddGroup 
51379
51380                         });
51381 //                      Draggability implies selection. DragZone's mousedown selects the element.
51382                         if (!this.multiSelect) { this.singleSelect = true; }
51383
51384 //                      Wire the DragZone's handlers up to methods in *this*
51385                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51386                 }
51387     },
51388
51389 /**     Specify from which ddGroup this DDView accepts drops. */
51390     setDroppable: function(ddGroup) {
51391         if (ddGroup instanceof Array) {
51392                 Roo.each(ddGroup, this.setDroppable, this);
51393                 return;
51394         }
51395         if (this.dropZone) {
51396                 this.dropZone.addToGroup(ddGroup);
51397         } else {
51398                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51399                                 containerScroll: true,
51400                                 ddGroup: ddGroup
51401                         });
51402
51403 //                      Wire the DropZone's handlers up to methods in *this*
51404                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51405                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51406                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51407                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51408                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51409                 }
51410     },
51411
51412 /**     Decide whether to drop above or below a View node. */
51413     getDropPoint : function(e, n, dd){
51414         if (n == this.el.dom) { return "above"; }
51415                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51416                 var c = t + (b - t) / 2;
51417                 var y = Roo.lib.Event.getPageY(e);
51418                 if(y <= c) {
51419                         return "above";
51420                 }else{
51421                         return "below";
51422                 }
51423     },
51424
51425     onNodeEnter : function(n, dd, e, data){
51426                 return false;
51427     },
51428     
51429     onNodeOver : function(n, dd, e, data){
51430                 var pt = this.getDropPoint(e, n, dd);
51431                 // set the insert point style on the target node
51432                 var dragElClass = this.dropNotAllowed;
51433                 if (pt) {
51434                         var targetElClass;
51435                         if (pt == "above"){
51436                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51437                                 targetElClass = "x-view-drag-insert-above";
51438                         } else {
51439                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51440                                 targetElClass = "x-view-drag-insert-below";
51441                         }
51442                         if (this.lastInsertClass != targetElClass){
51443                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51444                                 this.lastInsertClass = targetElClass;
51445                         }
51446                 }
51447                 return dragElClass;
51448         },
51449
51450     onNodeOut : function(n, dd, e, data){
51451                 this.removeDropIndicators(n);
51452     },
51453
51454     onNodeDrop : function(n, dd, e, data){
51455         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51456                 return false;
51457         }
51458         var pt = this.getDropPoint(e, n, dd);
51459                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51460                 if (pt == "below") { insertAt++; }
51461                 for (var i = 0; i < data.records.length; i++) {
51462                         var r = data.records[i];
51463                         var dup = this.store.getById(r.id);
51464                         if (dup && (dd != this.dragZone)) {
51465                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51466                         } else {
51467                                 if (data.copy) {
51468                                         this.store.insert(insertAt++, r.copy());
51469                                 } else {
51470                                         data.source.isDirtyFlag = true;
51471                                         r.store.remove(r);
51472                                         this.store.insert(insertAt++, r);
51473                                 }
51474                                 this.isDirtyFlag = true;
51475                         }
51476                 }
51477                 this.dragZone.cachedTarget = null;
51478                 return true;
51479     },
51480
51481     removeDropIndicators : function(n){
51482                 if(n){
51483                         Roo.fly(n).removeClass([
51484                                 "x-view-drag-insert-above",
51485                                 "x-view-drag-insert-below"]);
51486                         this.lastInsertClass = "_noclass";
51487                 }
51488     },
51489
51490 /**
51491  *      Utility method. Add a delete option to the DDView's context menu.
51492  *      @param {String} imageUrl The URL of the "delete" icon image.
51493  */
51494         setDeletable: function(imageUrl) {
51495                 if (!this.singleSelect && !this.multiSelect) {
51496                         this.singleSelect = true;
51497                 }
51498                 var c = this.getContextMenu();
51499                 this.contextMenu.on("itemclick", function(item) {
51500                         switch (item.id) {
51501                                 case "delete":
51502                                         this.remove(this.getSelectedIndexes());
51503                                         break;
51504                         }
51505                 }, this);
51506                 this.contextMenu.add({
51507                         icon: imageUrl,
51508                         id: "delete",
51509                         text: 'Delete'
51510                 });
51511         },
51512         
51513 /**     Return the context menu for this DDView. */
51514         getContextMenu: function() {
51515                 if (!this.contextMenu) {
51516 //                      Create the View's context menu
51517                         this.contextMenu = new Roo.menu.Menu({
51518                                 id: this.id + "-contextmenu"
51519                         });
51520                         this.el.on("contextmenu", this.showContextMenu, this);
51521                 }
51522                 return this.contextMenu;
51523         },
51524         
51525         disableContextMenu: function() {
51526                 if (this.contextMenu) {
51527                         this.el.un("contextmenu", this.showContextMenu, this);
51528                 }
51529         },
51530
51531         showContextMenu: function(e, item) {
51532         item = this.findItemFromChild(e.getTarget());
51533                 if (item) {
51534                         e.stopEvent();
51535                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51536                         this.contextMenu.showAt(e.getXY());
51537             }
51538     },
51539
51540 /**
51541  *      Remove {@link Roo.data.Record}s at the specified indices.
51542  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51543  */
51544     remove: function(selectedIndices) {
51545                 selectedIndices = [].concat(selectedIndices);
51546                 for (var i = 0; i < selectedIndices.length; i++) {
51547                         var rec = this.store.getAt(selectedIndices[i]);
51548                         this.store.remove(rec);
51549                 }
51550     },
51551
51552 /**
51553  *      Double click fires the event, but also, if this is draggable, and there is only one other
51554  *      related DropZone, it transfers the selected node.
51555  */
51556     onDblClick : function(e){
51557         var item = this.findItemFromChild(e.getTarget());
51558         if(item){
51559             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51560                 return false;
51561             }
51562             if (this.dragGroup) {
51563                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51564                     while (targets.indexOf(this.dropZone) > -1) {
51565                             targets.remove(this.dropZone);
51566                                 }
51567                     if (targets.length == 1) {
51568                                         this.dragZone.cachedTarget = null;
51569                         var el = Roo.get(targets[0].getEl());
51570                         var box = el.getBox(true);
51571                         targets[0].onNodeDrop(el.dom, {
51572                                 target: el.dom,
51573                                 xy: [box.x, box.y + box.height - 1]
51574                         }, null, this.getDragData(e));
51575                     }
51576                 }
51577         }
51578     },
51579     
51580     handleSelection: function(e) {
51581                 this.dragZone.cachedTarget = null;
51582         var item = this.findItemFromChild(e.getTarget());
51583         if (!item) {
51584                 this.clearSelections(true);
51585                 return;
51586         }
51587                 if (item && (this.multiSelect || this.singleSelect)){
51588                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51589                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51590                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51591                                 this.unselect(item);
51592                         } else {
51593                                 this.select(item, this.multiSelect && e.ctrlKey);
51594                                 this.lastSelection = item;
51595                         }
51596                 }
51597     },
51598
51599     onItemClick : function(item, index, e){
51600                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51601                         return false;
51602                 }
51603                 return true;
51604     },
51605
51606     unselect : function(nodeInfo, suppressEvent){
51607                 var node = this.getNode(nodeInfo);
51608                 if(node && this.isSelected(node)){
51609                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51610                                 Roo.fly(node).removeClass(this.selectedClass);
51611                                 this.selections.remove(node);
51612                                 if(!suppressEvent){
51613                                         this.fireEvent("selectionchange", this, this.selections);
51614                                 }
51615                         }
51616                 }
51617     }
51618 });
51619 /*
51620  * Based on:
51621  * Ext JS Library 1.1.1
51622  * Copyright(c) 2006-2007, Ext JS, LLC.
51623  *
51624  * Originally Released Under LGPL - original licence link has changed is not relivant.
51625  *
51626  * Fork - LGPL
51627  * <script type="text/javascript">
51628  */
51629  
51630 /**
51631  * @class Roo.LayoutManager
51632  * @extends Roo.util.Observable
51633  * Base class for layout managers.
51634  */
51635 Roo.LayoutManager = function(container, config){
51636     Roo.LayoutManager.superclass.constructor.call(this);
51637     this.el = Roo.get(container);
51638     // ie scrollbar fix
51639     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51640         document.body.scroll = "no";
51641     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51642         this.el.position('relative');
51643     }
51644     this.id = this.el.id;
51645     this.el.addClass("x-layout-container");
51646     /** false to disable window resize monitoring @type Boolean */
51647     this.monitorWindowResize = true;
51648     this.regions = {};
51649     this.addEvents({
51650         /**
51651          * @event layout
51652          * Fires when a layout is performed. 
51653          * @param {Roo.LayoutManager} this
51654          */
51655         "layout" : true,
51656         /**
51657          * @event regionresized
51658          * Fires when the user resizes a region. 
51659          * @param {Roo.LayoutRegion} region The resized region
51660          * @param {Number} newSize The new size (width for east/west, height for north/south)
51661          */
51662         "regionresized" : true,
51663         /**
51664          * @event regioncollapsed
51665          * Fires when a region is collapsed. 
51666          * @param {Roo.LayoutRegion} region The collapsed region
51667          */
51668         "regioncollapsed" : true,
51669         /**
51670          * @event regionexpanded
51671          * Fires when a region is expanded.  
51672          * @param {Roo.LayoutRegion} region The expanded region
51673          */
51674         "regionexpanded" : true
51675     });
51676     this.updating = false;
51677     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51678 };
51679
51680 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51681     /**
51682      * Returns true if this layout is currently being updated
51683      * @return {Boolean}
51684      */
51685     isUpdating : function(){
51686         return this.updating; 
51687     },
51688     
51689     /**
51690      * Suspend the LayoutManager from doing auto-layouts while
51691      * making multiple add or remove calls
51692      */
51693     beginUpdate : function(){
51694         this.updating = true;    
51695     },
51696     
51697     /**
51698      * Restore auto-layouts and optionally disable the manager from performing a layout
51699      * @param {Boolean} noLayout true to disable a layout update 
51700      */
51701     endUpdate : function(noLayout){
51702         this.updating = false;
51703         if(!noLayout){
51704             this.layout();
51705         }    
51706     },
51707     
51708     layout: function(){
51709         
51710     },
51711     
51712     onRegionResized : function(region, newSize){
51713         this.fireEvent("regionresized", region, newSize);
51714         this.layout();
51715     },
51716     
51717     onRegionCollapsed : function(region){
51718         this.fireEvent("regioncollapsed", region);
51719     },
51720     
51721     onRegionExpanded : function(region){
51722         this.fireEvent("regionexpanded", region);
51723     },
51724         
51725     /**
51726      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51727      * performs box-model adjustments.
51728      * @return {Object} The size as an object {width: (the width), height: (the height)}
51729      */
51730     getViewSize : function(){
51731         var size;
51732         if(this.el.dom != document.body){
51733             size = this.el.getSize();
51734         }else{
51735             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51736         }
51737         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51738         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51739         return size;
51740     },
51741     
51742     /**
51743      * Returns the Element this layout is bound to.
51744      * @return {Roo.Element}
51745      */
51746     getEl : function(){
51747         return this.el;
51748     },
51749     
51750     /**
51751      * Returns the specified region.
51752      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51753      * @return {Roo.LayoutRegion}
51754      */
51755     getRegion : function(target){
51756         return this.regions[target.toLowerCase()];
51757     },
51758     
51759     onWindowResize : function(){
51760         if(this.monitorWindowResize){
51761             this.layout();
51762         }
51763     }
51764 });/*
51765  * Based on:
51766  * Ext JS Library 1.1.1
51767  * Copyright(c) 2006-2007, Ext JS, LLC.
51768  *
51769  * Originally Released Under LGPL - original licence link has changed is not relivant.
51770  *
51771  * Fork - LGPL
51772  * <script type="text/javascript">
51773  */
51774 /**
51775  * @class Roo.BorderLayout
51776  * @extends Roo.LayoutManager
51777  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51778  * please see: <br><br>
51779  * <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>
51780  * <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>
51781  * Example:
51782  <pre><code>
51783  var layout = new Roo.BorderLayout(document.body, {
51784     north: {
51785         initialSize: 25,
51786         titlebar: false
51787     },
51788     west: {
51789         split:true,
51790         initialSize: 200,
51791         minSize: 175,
51792         maxSize: 400,
51793         titlebar: true,
51794         collapsible: true
51795     },
51796     east: {
51797         split:true,
51798         initialSize: 202,
51799         minSize: 175,
51800         maxSize: 400,
51801         titlebar: true,
51802         collapsible: true
51803     },
51804     south: {
51805         split:true,
51806         initialSize: 100,
51807         minSize: 100,
51808         maxSize: 200,
51809         titlebar: true,
51810         collapsible: true
51811     },
51812     center: {
51813         titlebar: true,
51814         autoScroll:true,
51815         resizeTabs: true,
51816         minTabWidth: 50,
51817         preferredTabWidth: 150
51818     }
51819 });
51820
51821 // shorthand
51822 var CP = Roo.ContentPanel;
51823
51824 layout.beginUpdate();
51825 layout.add("north", new CP("north", "North"));
51826 layout.add("south", new CP("south", {title: "South", closable: true}));
51827 layout.add("west", new CP("west", {title: "West"}));
51828 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51829 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51830 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51831 layout.getRegion("center").showPanel("center1");
51832 layout.endUpdate();
51833 </code></pre>
51834
51835 <b>The container the layout is rendered into can be either the body element or any other element.
51836 If it is not the body element, the container needs to either be an absolute positioned element,
51837 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51838 the container size if it is not the body element.</b>
51839
51840 * @constructor
51841 * Create a new BorderLayout
51842 * @param {String/HTMLElement/Element} container The container this layout is bound to
51843 * @param {Object} config Configuration options
51844  */
51845 Roo.BorderLayout = function(container, config){
51846     config = config || {};
51847     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51848     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51849     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51850         var target = this.factory.validRegions[i];
51851         if(config[target]){
51852             this.addRegion(target, config[target]);
51853         }
51854     }
51855 };
51856
51857 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51858     /**
51859      * Creates and adds a new region if it doesn't already exist.
51860      * @param {String} target The target region key (north, south, east, west or center).
51861      * @param {Object} config The regions config object
51862      * @return {BorderLayoutRegion} The new region
51863      */
51864     addRegion : function(target, config){
51865         if(!this.regions[target]){
51866             var r = this.factory.create(target, this, config);
51867             this.bindRegion(target, r);
51868         }
51869         return this.regions[target];
51870     },
51871
51872     // private (kinda)
51873     bindRegion : function(name, r){
51874         this.regions[name] = r;
51875         r.on("visibilitychange", this.layout, this);
51876         r.on("paneladded", this.layout, this);
51877         r.on("panelremoved", this.layout, this);
51878         r.on("invalidated", this.layout, this);
51879         r.on("resized", this.onRegionResized, this);
51880         r.on("collapsed", this.onRegionCollapsed, this);
51881         r.on("expanded", this.onRegionExpanded, this);
51882     },
51883
51884     /**
51885      * Performs a layout update.
51886      */
51887     layout : function(){
51888         if(this.updating) {
51889             return;
51890         }
51891         var size = this.getViewSize();
51892         var w = size.width;
51893         var h = size.height;
51894         var centerW = w;
51895         var centerH = h;
51896         var centerY = 0;
51897         var centerX = 0;
51898         //var x = 0, y = 0;
51899
51900         var rs = this.regions;
51901         var north = rs["north"];
51902         var south = rs["south"]; 
51903         var west = rs["west"];
51904         var east = rs["east"];
51905         var center = rs["center"];
51906         //if(this.hideOnLayout){ // not supported anymore
51907             //c.el.setStyle("display", "none");
51908         //}
51909         if(north && north.isVisible()){
51910             var b = north.getBox();
51911             var m = north.getMargins();
51912             b.width = w - (m.left+m.right);
51913             b.x = m.left;
51914             b.y = m.top;
51915             centerY = b.height + b.y + m.bottom;
51916             centerH -= centerY;
51917             north.updateBox(this.safeBox(b));
51918         }
51919         if(south && south.isVisible()){
51920             var b = south.getBox();
51921             var m = south.getMargins();
51922             b.width = w - (m.left+m.right);
51923             b.x = m.left;
51924             var totalHeight = (b.height + m.top + m.bottom);
51925             b.y = h - totalHeight + m.top;
51926             centerH -= totalHeight;
51927             south.updateBox(this.safeBox(b));
51928         }
51929         if(west && west.isVisible()){
51930             var b = west.getBox();
51931             var m = west.getMargins();
51932             b.height = centerH - (m.top+m.bottom);
51933             b.x = m.left;
51934             b.y = centerY + m.top;
51935             var totalWidth = (b.width + m.left + m.right);
51936             centerX += totalWidth;
51937             centerW -= totalWidth;
51938             west.updateBox(this.safeBox(b));
51939         }
51940         if(east && east.isVisible()){
51941             var b = east.getBox();
51942             var m = east.getMargins();
51943             b.height = centerH - (m.top+m.bottom);
51944             var totalWidth = (b.width + m.left + m.right);
51945             b.x = w - totalWidth + m.left;
51946             b.y = centerY + m.top;
51947             centerW -= totalWidth;
51948             east.updateBox(this.safeBox(b));
51949         }
51950         if(center){
51951             var m = center.getMargins();
51952             var centerBox = {
51953                 x: centerX + m.left,
51954                 y: centerY + m.top,
51955                 width: centerW - (m.left+m.right),
51956                 height: centerH - (m.top+m.bottom)
51957             };
51958             //if(this.hideOnLayout){
51959                 //center.el.setStyle("display", "block");
51960             //}
51961             center.updateBox(this.safeBox(centerBox));
51962         }
51963         this.el.repaint();
51964         this.fireEvent("layout", this);
51965     },
51966
51967     // private
51968     safeBox : function(box){
51969         box.width = Math.max(0, box.width);
51970         box.height = Math.max(0, box.height);
51971         return box;
51972     },
51973
51974     /**
51975      * Adds a ContentPanel (or subclass) to this layout.
51976      * @param {String} target The target region key (north, south, east, west or center).
51977      * @param {Roo.ContentPanel} panel The panel to add
51978      * @return {Roo.ContentPanel} The added panel
51979      */
51980     add : function(target, panel){
51981          
51982         target = target.toLowerCase();
51983         return this.regions[target].add(panel);
51984     },
51985
51986     /**
51987      * Remove a ContentPanel (or subclass) to this layout.
51988      * @param {String} target The target region key (north, south, east, west or center).
51989      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
51990      * @return {Roo.ContentPanel} The removed panel
51991      */
51992     remove : function(target, panel){
51993         target = target.toLowerCase();
51994         return this.regions[target].remove(panel);
51995     },
51996
51997     /**
51998      * Searches all regions for a panel with the specified id
51999      * @param {String} panelId
52000      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52001      */
52002     findPanel : function(panelId){
52003         var rs = this.regions;
52004         for(var target in rs){
52005             if(typeof rs[target] != "function"){
52006                 var p = rs[target].getPanel(panelId);
52007                 if(p){
52008                     return p;
52009                 }
52010             }
52011         }
52012         return null;
52013     },
52014
52015     /**
52016      * Searches all regions for a panel with the specified id and activates (shows) it.
52017      * @param {String/ContentPanel} panelId The panels id or the panel itself
52018      * @return {Roo.ContentPanel} The shown panel or null
52019      */
52020     showPanel : function(panelId) {
52021       var rs = this.regions;
52022       for(var target in rs){
52023          var r = rs[target];
52024          if(typeof r != "function"){
52025             if(r.hasPanel(panelId)){
52026                return r.showPanel(panelId);
52027             }
52028          }
52029       }
52030       return null;
52031    },
52032
52033    /**
52034      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52035      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52036      */
52037     restoreState : function(provider){
52038         if(!provider){
52039             provider = Roo.state.Manager;
52040         }
52041         var sm = new Roo.LayoutStateManager();
52042         sm.init(this, provider);
52043     },
52044
52045     /**
52046      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52047      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52048      * a valid ContentPanel config object.  Example:
52049      * <pre><code>
52050 // Create the main layout
52051 var layout = new Roo.BorderLayout('main-ct', {
52052     west: {
52053         split:true,
52054         minSize: 175,
52055         titlebar: true
52056     },
52057     center: {
52058         title:'Components'
52059     }
52060 }, 'main-ct');
52061
52062 // Create and add multiple ContentPanels at once via configs
52063 layout.batchAdd({
52064    west: {
52065        id: 'source-files',
52066        autoCreate:true,
52067        title:'Ext Source Files',
52068        autoScroll:true,
52069        fitToFrame:true
52070    },
52071    center : {
52072        el: cview,
52073        autoScroll:true,
52074        fitToFrame:true,
52075        toolbar: tb,
52076        resizeEl:'cbody'
52077    }
52078 });
52079 </code></pre>
52080      * @param {Object} regions An object containing ContentPanel configs by region name
52081      */
52082     batchAdd : function(regions){
52083         this.beginUpdate();
52084         for(var rname in regions){
52085             var lr = this.regions[rname];
52086             if(lr){
52087                 this.addTypedPanels(lr, regions[rname]);
52088             }
52089         }
52090         this.endUpdate();
52091     },
52092
52093     // private
52094     addTypedPanels : function(lr, ps){
52095         if(typeof ps == 'string'){
52096             lr.add(new Roo.ContentPanel(ps));
52097         }
52098         else if(ps instanceof Array){
52099             for(var i =0, len = ps.length; i < len; i++){
52100                 this.addTypedPanels(lr, ps[i]);
52101             }
52102         }
52103         else if(!ps.events){ // raw config?
52104             var el = ps.el;
52105             delete ps.el; // prevent conflict
52106             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52107         }
52108         else {  // panel object assumed!
52109             lr.add(ps);
52110         }
52111     },
52112     /**
52113      * Adds a xtype elements to the layout.
52114      * <pre><code>
52115
52116 layout.addxtype({
52117        xtype : 'ContentPanel',
52118        region: 'west',
52119        items: [ .... ]
52120    }
52121 );
52122
52123 layout.addxtype({
52124         xtype : 'NestedLayoutPanel',
52125         region: 'west',
52126         layout: {
52127            center: { },
52128            west: { }   
52129         },
52130         items : [ ... list of content panels or nested layout panels.. ]
52131    }
52132 );
52133 </code></pre>
52134      * @param {Object} cfg Xtype definition of item to add.
52135      */
52136     addxtype : function(cfg)
52137     {
52138         // basically accepts a pannel...
52139         // can accept a layout region..!?!?
52140         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52141         
52142         if (!cfg.xtype.match(/Panel$/)) {
52143             return false;
52144         }
52145         var ret = false;
52146         
52147         if (typeof(cfg.region) == 'undefined') {
52148             Roo.log("Failed to add Panel, region was not set");
52149             Roo.log(cfg);
52150             return false;
52151         }
52152         var region = cfg.region;
52153         delete cfg.region;
52154         
52155           
52156         var xitems = [];
52157         if (cfg.items) {
52158             xitems = cfg.items;
52159             delete cfg.items;
52160         }
52161         var nb = false;
52162         
52163         switch(cfg.xtype) 
52164         {
52165             case 'ContentPanel':  // ContentPanel (el, cfg)
52166             case 'ScrollPanel':  // ContentPanel (el, cfg)
52167             case 'ViewPanel': 
52168                 if(cfg.autoCreate) {
52169                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52170                 } else {
52171                     var el = this.el.createChild();
52172                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52173                 }
52174                 
52175                 this.add(region, ret);
52176                 break;
52177             
52178             
52179             case 'TreePanel': // our new panel!
52180                 cfg.el = this.el.createChild();
52181                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52182                 this.add(region, ret);
52183                 break;
52184             
52185             case 'NestedLayoutPanel': 
52186                 // create a new Layout (which is  a Border Layout...
52187                 var el = this.el.createChild();
52188                 var clayout = cfg.layout;
52189                 delete cfg.layout;
52190                 clayout.items   = clayout.items  || [];
52191                 // replace this exitems with the clayout ones..
52192                 xitems = clayout.items;
52193                  
52194                 
52195                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52196                     cfg.background = false;
52197                 }
52198                 var layout = new Roo.BorderLayout(el, clayout);
52199                 
52200                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52201                 //console.log('adding nested layout panel '  + cfg.toSource());
52202                 this.add(region, ret);
52203                 nb = {}; /// find first...
52204                 break;
52205                 
52206             case 'GridPanel': 
52207             
52208                 // needs grid and region
52209                 
52210                 //var el = this.getRegion(region).el.createChild();
52211                 var el = this.el.createChild();
52212                 // create the grid first...
52213                 
52214                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52215                 delete cfg.grid;
52216                 if (region == 'center' && this.active ) {
52217                     cfg.background = false;
52218                 }
52219                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52220                 
52221                 this.add(region, ret);
52222                 if (cfg.background) {
52223                     ret.on('activate', function(gp) {
52224                         if (!gp.grid.rendered) {
52225                             gp.grid.render();
52226                         }
52227                     });
52228                 } else {
52229                     grid.render();
52230                 }
52231                 break;
52232            
52233            
52234            
52235                 
52236                 
52237                 
52238             default:
52239                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52240                     
52241                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52242                     this.add(region, ret);
52243                 } else {
52244                 
52245                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52246                     return null;
52247                 }
52248                 
52249              // GridPanel (grid, cfg)
52250             
52251         }
52252         this.beginUpdate();
52253         // add children..
52254         var region = '';
52255         var abn = {};
52256         Roo.each(xitems, function(i)  {
52257             region = nb && i.region ? i.region : false;
52258             
52259             var add = ret.addxtype(i);
52260            
52261             if (region) {
52262                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52263                 if (!i.background) {
52264                     abn[region] = nb[region] ;
52265                 }
52266             }
52267             
52268         });
52269         this.endUpdate();
52270
52271         // make the last non-background panel active..
52272         //if (nb) { Roo.log(abn); }
52273         if (nb) {
52274             
52275             for(var r in abn) {
52276                 region = this.getRegion(r);
52277                 if (region) {
52278                     // tried using nb[r], but it does not work..
52279                      
52280                     region.showPanel(abn[r]);
52281                    
52282                 }
52283             }
52284         }
52285         return ret;
52286         
52287     }
52288 });
52289
52290 /**
52291  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52292  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52293  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52294  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52295  * <pre><code>
52296 // shorthand
52297 var CP = Roo.ContentPanel;
52298
52299 var layout = Roo.BorderLayout.create({
52300     north: {
52301         initialSize: 25,
52302         titlebar: false,
52303         panels: [new CP("north", "North")]
52304     },
52305     west: {
52306         split:true,
52307         initialSize: 200,
52308         minSize: 175,
52309         maxSize: 400,
52310         titlebar: true,
52311         collapsible: true,
52312         panels: [new CP("west", {title: "West"})]
52313     },
52314     east: {
52315         split:true,
52316         initialSize: 202,
52317         minSize: 175,
52318         maxSize: 400,
52319         titlebar: true,
52320         collapsible: true,
52321         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52322     },
52323     south: {
52324         split:true,
52325         initialSize: 100,
52326         minSize: 100,
52327         maxSize: 200,
52328         titlebar: true,
52329         collapsible: true,
52330         panels: [new CP("south", {title: "South", closable: true})]
52331     },
52332     center: {
52333         titlebar: true,
52334         autoScroll:true,
52335         resizeTabs: true,
52336         minTabWidth: 50,
52337         preferredTabWidth: 150,
52338         panels: [
52339             new CP("center1", {title: "Close Me", closable: true}),
52340             new CP("center2", {title: "Center Panel", closable: false})
52341         ]
52342     }
52343 }, document.body);
52344
52345 layout.getRegion("center").showPanel("center1");
52346 </code></pre>
52347  * @param config
52348  * @param targetEl
52349  */
52350 Roo.BorderLayout.create = function(config, targetEl){
52351     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52352     layout.beginUpdate();
52353     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52354     for(var j = 0, jlen = regions.length; j < jlen; j++){
52355         var lr = regions[j];
52356         if(layout.regions[lr] && config[lr].panels){
52357             var r = layout.regions[lr];
52358             var ps = config[lr].panels;
52359             layout.addTypedPanels(r, ps);
52360         }
52361     }
52362     layout.endUpdate();
52363     return layout;
52364 };
52365
52366 // private
52367 Roo.BorderLayout.RegionFactory = {
52368     // private
52369     validRegions : ["north","south","east","west","center"],
52370
52371     // private
52372     create : function(target, mgr, config){
52373         target = target.toLowerCase();
52374         if(config.lightweight || config.basic){
52375             return new Roo.BasicLayoutRegion(mgr, config, target);
52376         }
52377         switch(target){
52378             case "north":
52379                 return new Roo.NorthLayoutRegion(mgr, config);
52380             case "south":
52381                 return new Roo.SouthLayoutRegion(mgr, config);
52382             case "east":
52383                 return new Roo.EastLayoutRegion(mgr, config);
52384             case "west":
52385                 return new Roo.WestLayoutRegion(mgr, config);
52386             case "center":
52387                 return new Roo.CenterLayoutRegion(mgr, config);
52388         }
52389         throw 'Layout region "'+target+'" not supported.';
52390     }
52391 };/*
52392  * Based on:
52393  * Ext JS Library 1.1.1
52394  * Copyright(c) 2006-2007, Ext JS, LLC.
52395  *
52396  * Originally Released Under LGPL - original licence link has changed is not relivant.
52397  *
52398  * Fork - LGPL
52399  * <script type="text/javascript">
52400  */
52401  
52402 /**
52403  * @class Roo.BasicLayoutRegion
52404  * @extends Roo.util.Observable
52405  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52406  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52407  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52408  */
52409 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52410     this.mgr = mgr;
52411     this.position  = pos;
52412     this.events = {
52413         /**
52414          * @scope Roo.BasicLayoutRegion
52415          */
52416         
52417         /**
52418          * @event beforeremove
52419          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52420          * @param {Roo.LayoutRegion} this
52421          * @param {Roo.ContentPanel} panel The panel
52422          * @param {Object} e The cancel event object
52423          */
52424         "beforeremove" : true,
52425         /**
52426          * @event invalidated
52427          * Fires when the layout for this region is changed.
52428          * @param {Roo.LayoutRegion} this
52429          */
52430         "invalidated" : true,
52431         /**
52432          * @event visibilitychange
52433          * Fires when this region is shown or hidden 
52434          * @param {Roo.LayoutRegion} this
52435          * @param {Boolean} visibility true or false
52436          */
52437         "visibilitychange" : true,
52438         /**
52439          * @event paneladded
52440          * Fires when a panel is added. 
52441          * @param {Roo.LayoutRegion} this
52442          * @param {Roo.ContentPanel} panel The panel
52443          */
52444         "paneladded" : true,
52445         /**
52446          * @event panelremoved
52447          * Fires when a panel is removed. 
52448          * @param {Roo.LayoutRegion} this
52449          * @param {Roo.ContentPanel} panel The panel
52450          */
52451         "panelremoved" : true,
52452         /**
52453          * @event beforecollapse
52454          * Fires when this region before collapse.
52455          * @param {Roo.LayoutRegion} this
52456          */
52457         "beforecollapse" : true,
52458         /**
52459          * @event collapsed
52460          * Fires when this region is collapsed.
52461          * @param {Roo.LayoutRegion} this
52462          */
52463         "collapsed" : true,
52464         /**
52465          * @event expanded
52466          * Fires when this region is expanded.
52467          * @param {Roo.LayoutRegion} this
52468          */
52469         "expanded" : true,
52470         /**
52471          * @event slideshow
52472          * Fires when this region is slid into view.
52473          * @param {Roo.LayoutRegion} this
52474          */
52475         "slideshow" : true,
52476         /**
52477          * @event slidehide
52478          * Fires when this region slides out of view. 
52479          * @param {Roo.LayoutRegion} this
52480          */
52481         "slidehide" : true,
52482         /**
52483          * @event panelactivated
52484          * Fires when a panel is activated. 
52485          * @param {Roo.LayoutRegion} this
52486          * @param {Roo.ContentPanel} panel The activated panel
52487          */
52488         "panelactivated" : true,
52489         /**
52490          * @event resized
52491          * Fires when the user resizes this region. 
52492          * @param {Roo.LayoutRegion} this
52493          * @param {Number} newSize The new size (width for east/west, height for north/south)
52494          */
52495         "resized" : true
52496     };
52497     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52498     this.panels = new Roo.util.MixedCollection();
52499     this.panels.getKey = this.getPanelId.createDelegate(this);
52500     this.box = null;
52501     this.activePanel = null;
52502     // ensure listeners are added...
52503     
52504     if (config.listeners || config.events) {
52505         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52506             listeners : config.listeners || {},
52507             events : config.events || {}
52508         });
52509     }
52510     
52511     if(skipConfig !== true){
52512         this.applyConfig(config);
52513     }
52514 };
52515
52516 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52517     getPanelId : function(p){
52518         return p.getId();
52519     },
52520     
52521     applyConfig : function(config){
52522         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52523         this.config = config;
52524         
52525     },
52526     
52527     /**
52528      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52529      * the width, for horizontal (north, south) the height.
52530      * @param {Number} newSize The new width or height
52531      */
52532     resizeTo : function(newSize){
52533         var el = this.el ? this.el :
52534                  (this.activePanel ? this.activePanel.getEl() : null);
52535         if(el){
52536             switch(this.position){
52537                 case "east":
52538                 case "west":
52539                     el.setWidth(newSize);
52540                     this.fireEvent("resized", this, newSize);
52541                 break;
52542                 case "north":
52543                 case "south":
52544                     el.setHeight(newSize);
52545                     this.fireEvent("resized", this, newSize);
52546                 break;                
52547             }
52548         }
52549     },
52550     
52551     getBox : function(){
52552         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52553     },
52554     
52555     getMargins : function(){
52556         return this.margins;
52557     },
52558     
52559     updateBox : function(box){
52560         this.box = box;
52561         var el = this.activePanel.getEl();
52562         el.dom.style.left = box.x + "px";
52563         el.dom.style.top = box.y + "px";
52564         this.activePanel.setSize(box.width, box.height);
52565     },
52566     
52567     /**
52568      * Returns the container element for this region.
52569      * @return {Roo.Element}
52570      */
52571     getEl : function(){
52572         return this.activePanel;
52573     },
52574     
52575     /**
52576      * Returns true if this region is currently visible.
52577      * @return {Boolean}
52578      */
52579     isVisible : function(){
52580         return this.activePanel ? true : false;
52581     },
52582     
52583     setActivePanel : function(panel){
52584         panel = this.getPanel(panel);
52585         if(this.activePanel && this.activePanel != panel){
52586             this.activePanel.setActiveState(false);
52587             this.activePanel.getEl().setLeftTop(-10000,-10000);
52588         }
52589         this.activePanel = panel;
52590         panel.setActiveState(true);
52591         if(this.box){
52592             panel.setSize(this.box.width, this.box.height);
52593         }
52594         this.fireEvent("panelactivated", this, panel);
52595         this.fireEvent("invalidated");
52596     },
52597     
52598     /**
52599      * Show the specified panel.
52600      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52601      * @return {Roo.ContentPanel} The shown panel or null
52602      */
52603     showPanel : function(panel){
52604         if(panel = this.getPanel(panel)){
52605             this.setActivePanel(panel);
52606         }
52607         return panel;
52608     },
52609     
52610     /**
52611      * Get the active panel for this region.
52612      * @return {Roo.ContentPanel} The active panel or null
52613      */
52614     getActivePanel : function(){
52615         return this.activePanel;
52616     },
52617     
52618     /**
52619      * Add the passed ContentPanel(s)
52620      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52621      * @return {Roo.ContentPanel} The panel added (if only one was added)
52622      */
52623     add : function(panel){
52624         if(arguments.length > 1){
52625             for(var i = 0, len = arguments.length; i < len; i++) {
52626                 this.add(arguments[i]);
52627             }
52628             return null;
52629         }
52630         if(this.hasPanel(panel)){
52631             this.showPanel(panel);
52632             return panel;
52633         }
52634         var el = panel.getEl();
52635         if(el.dom.parentNode != this.mgr.el.dom){
52636             this.mgr.el.dom.appendChild(el.dom);
52637         }
52638         if(panel.setRegion){
52639             panel.setRegion(this);
52640         }
52641         this.panels.add(panel);
52642         el.setStyle("position", "absolute");
52643         if(!panel.background){
52644             this.setActivePanel(panel);
52645             if(this.config.initialSize && this.panels.getCount()==1){
52646                 this.resizeTo(this.config.initialSize);
52647             }
52648         }
52649         this.fireEvent("paneladded", this, panel);
52650         return panel;
52651     },
52652     
52653     /**
52654      * Returns true if the panel is in this region.
52655      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52656      * @return {Boolean}
52657      */
52658     hasPanel : function(panel){
52659         if(typeof panel == "object"){ // must be panel obj
52660             panel = panel.getId();
52661         }
52662         return this.getPanel(panel) ? true : false;
52663     },
52664     
52665     /**
52666      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52667      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52668      * @param {Boolean} preservePanel Overrides the config preservePanel option
52669      * @return {Roo.ContentPanel} The panel that was removed
52670      */
52671     remove : function(panel, preservePanel){
52672         panel = this.getPanel(panel);
52673         if(!panel){
52674             return null;
52675         }
52676         var e = {};
52677         this.fireEvent("beforeremove", this, panel, e);
52678         if(e.cancel === true){
52679             return null;
52680         }
52681         var panelId = panel.getId();
52682         this.panels.removeKey(panelId);
52683         return panel;
52684     },
52685     
52686     /**
52687      * Returns the panel specified or null if it's not in this region.
52688      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52689      * @return {Roo.ContentPanel}
52690      */
52691     getPanel : function(id){
52692         if(typeof id == "object"){ // must be panel obj
52693             return id;
52694         }
52695         return this.panels.get(id);
52696     },
52697     
52698     /**
52699      * Returns this regions position (north/south/east/west/center).
52700      * @return {String} 
52701      */
52702     getPosition: function(){
52703         return this.position;    
52704     }
52705 });/*
52706  * Based on:
52707  * Ext JS Library 1.1.1
52708  * Copyright(c) 2006-2007, Ext JS, LLC.
52709  *
52710  * Originally Released Under LGPL - original licence link has changed is not relivant.
52711  *
52712  * Fork - LGPL
52713  * <script type="text/javascript">
52714  */
52715  
52716 /**
52717  * @class Roo.LayoutRegion
52718  * @extends Roo.BasicLayoutRegion
52719  * This class represents a region in a layout manager.
52720  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52721  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52722  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52723  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52724  * @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})
52725  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52726  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52727  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52728  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52729  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52730  * @cfg {String}    title           The title for the region (overrides panel titles)
52731  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52732  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52733  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52734  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52735  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52736  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52737  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52738  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52739  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52740  * @cfg {Boolean}   showPin         True to show a pin button
52741  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52742  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52743  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52744  * @cfg {Number}    width           For East/West panels
52745  * @cfg {Number}    height          For North/South panels
52746  * @cfg {Boolean}   split           To show the splitter
52747  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52748  */
52749 Roo.LayoutRegion = function(mgr, config, pos){
52750     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52751     var dh = Roo.DomHelper;
52752     /** This region's container element 
52753     * @type Roo.Element */
52754     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52755     /** This region's title element 
52756     * @type Roo.Element */
52757
52758     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52759         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52760         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52761     ]}, true);
52762     this.titleEl.enableDisplayMode();
52763     /** This region's title text element 
52764     * @type HTMLElement */
52765     this.titleTextEl = this.titleEl.dom.firstChild;
52766     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52767     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52768     this.closeBtn.enableDisplayMode();
52769     this.closeBtn.on("click", this.closeClicked, this);
52770     this.closeBtn.hide();
52771
52772     this.createBody(config);
52773     this.visible = true;
52774     this.collapsed = false;
52775
52776     if(config.hideWhenEmpty){
52777         this.hide();
52778         this.on("paneladded", this.validateVisibility, this);
52779         this.on("panelremoved", this.validateVisibility, this);
52780     }
52781     this.applyConfig(config);
52782 };
52783
52784 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52785
52786     createBody : function(){
52787         /** This region's body element 
52788         * @type Roo.Element */
52789         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52790     },
52791
52792     applyConfig : function(c){
52793         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52794             var dh = Roo.DomHelper;
52795             if(c.titlebar !== false){
52796                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52797                 this.collapseBtn.on("click", this.collapse, this);
52798                 this.collapseBtn.enableDisplayMode();
52799
52800                 if(c.showPin === true || this.showPin){
52801                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52802                     this.stickBtn.enableDisplayMode();
52803                     this.stickBtn.on("click", this.expand, this);
52804                     this.stickBtn.hide();
52805                 }
52806             }
52807             /** This region's collapsed element
52808             * @type Roo.Element */
52809             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52810                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52811             ]}, true);
52812             if(c.floatable !== false){
52813                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52814                this.collapsedEl.on("click", this.collapseClick, this);
52815             }
52816
52817             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52818                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52819                    id: "message", unselectable: "on", style:{"float":"left"}});
52820                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52821              }
52822             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52823             this.expandBtn.on("click", this.expand, this);
52824         }
52825         if(this.collapseBtn){
52826             this.collapseBtn.setVisible(c.collapsible == true);
52827         }
52828         this.cmargins = c.cmargins || this.cmargins ||
52829                          (this.position == "west" || this.position == "east" ?
52830                              {top: 0, left: 2, right:2, bottom: 0} :
52831                              {top: 2, left: 0, right:0, bottom: 2});
52832         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52833         this.bottomTabs = c.tabPosition != "top";
52834         this.autoScroll = c.autoScroll || false;
52835         if(this.autoScroll){
52836             this.bodyEl.setStyle("overflow", "auto");
52837         }else{
52838             this.bodyEl.setStyle("overflow", "hidden");
52839         }
52840         //if(c.titlebar !== false){
52841             if((!c.titlebar && !c.title) || c.titlebar === false){
52842                 this.titleEl.hide();
52843             }else{
52844                 this.titleEl.show();
52845                 if(c.title){
52846                     this.titleTextEl.innerHTML = c.title;
52847                 }
52848             }
52849         //}
52850         this.duration = c.duration || .30;
52851         this.slideDuration = c.slideDuration || .45;
52852         this.config = c;
52853         if(c.collapsed){
52854             this.collapse(true);
52855         }
52856         if(c.hidden){
52857             this.hide();
52858         }
52859     },
52860     /**
52861      * Returns true if this region is currently visible.
52862      * @return {Boolean}
52863      */
52864     isVisible : function(){
52865         return this.visible;
52866     },
52867
52868     /**
52869      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52870      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52871      */
52872     setCollapsedTitle : function(title){
52873         title = title || "&#160;";
52874         if(this.collapsedTitleTextEl){
52875             this.collapsedTitleTextEl.innerHTML = title;
52876         }
52877     },
52878
52879     getBox : function(){
52880         var b;
52881         if(!this.collapsed){
52882             b = this.el.getBox(false, true);
52883         }else{
52884             b = this.collapsedEl.getBox(false, true);
52885         }
52886         return b;
52887     },
52888
52889     getMargins : function(){
52890         return this.collapsed ? this.cmargins : this.margins;
52891     },
52892
52893     highlight : function(){
52894         this.el.addClass("x-layout-panel-dragover");
52895     },
52896
52897     unhighlight : function(){
52898         this.el.removeClass("x-layout-panel-dragover");
52899     },
52900
52901     updateBox : function(box){
52902         this.box = box;
52903         if(!this.collapsed){
52904             this.el.dom.style.left = box.x + "px";
52905             this.el.dom.style.top = box.y + "px";
52906             this.updateBody(box.width, box.height);
52907         }else{
52908             this.collapsedEl.dom.style.left = box.x + "px";
52909             this.collapsedEl.dom.style.top = box.y + "px";
52910             this.collapsedEl.setSize(box.width, box.height);
52911         }
52912         if(this.tabs){
52913             this.tabs.autoSizeTabs();
52914         }
52915     },
52916
52917     updateBody : function(w, h){
52918         if(w !== null){
52919             this.el.setWidth(w);
52920             w -= this.el.getBorderWidth("rl");
52921             if(this.config.adjustments){
52922                 w += this.config.adjustments[0];
52923             }
52924         }
52925         if(h !== null){
52926             this.el.setHeight(h);
52927             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52928             h -= this.el.getBorderWidth("tb");
52929             if(this.config.adjustments){
52930                 h += this.config.adjustments[1];
52931             }
52932             this.bodyEl.setHeight(h);
52933             if(this.tabs){
52934                 h = this.tabs.syncHeight(h);
52935             }
52936         }
52937         if(this.panelSize){
52938             w = w !== null ? w : this.panelSize.width;
52939             h = h !== null ? h : this.panelSize.height;
52940         }
52941         if(this.activePanel){
52942             var el = this.activePanel.getEl();
52943             w = w !== null ? w : el.getWidth();
52944             h = h !== null ? h : el.getHeight();
52945             this.panelSize = {width: w, height: h};
52946             this.activePanel.setSize(w, h);
52947         }
52948         if(Roo.isIE && this.tabs){
52949             this.tabs.el.repaint();
52950         }
52951     },
52952
52953     /**
52954      * Returns the container element for this region.
52955      * @return {Roo.Element}
52956      */
52957     getEl : function(){
52958         return this.el;
52959     },
52960
52961     /**
52962      * Hides this region.
52963      */
52964     hide : function(){
52965         if(!this.collapsed){
52966             this.el.dom.style.left = "-2000px";
52967             this.el.hide();
52968         }else{
52969             this.collapsedEl.dom.style.left = "-2000px";
52970             this.collapsedEl.hide();
52971         }
52972         this.visible = false;
52973         this.fireEvent("visibilitychange", this, false);
52974     },
52975
52976     /**
52977      * Shows this region if it was previously hidden.
52978      */
52979     show : function(){
52980         if(!this.collapsed){
52981             this.el.show();
52982         }else{
52983             this.collapsedEl.show();
52984         }
52985         this.visible = true;
52986         this.fireEvent("visibilitychange", this, true);
52987     },
52988
52989     closeClicked : function(){
52990         if(this.activePanel){
52991             this.remove(this.activePanel);
52992         }
52993     },
52994
52995     collapseClick : function(e){
52996         if(this.isSlid){
52997            e.stopPropagation();
52998            this.slideIn();
52999         }else{
53000            e.stopPropagation();
53001            this.slideOut();
53002         }
53003     },
53004
53005     /**
53006      * Collapses this region.
53007      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53008      */
53009     collapse : function(skipAnim, skipCheck){
53010         if(this.collapsed) {
53011             return;
53012         }
53013         
53014         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53015             
53016             this.collapsed = true;
53017             if(this.split){
53018                 this.split.el.hide();
53019             }
53020             if(this.config.animate && skipAnim !== true){
53021                 this.fireEvent("invalidated", this);
53022                 this.animateCollapse();
53023             }else{
53024                 this.el.setLocation(-20000,-20000);
53025                 this.el.hide();
53026                 this.collapsedEl.show();
53027                 this.fireEvent("collapsed", this);
53028                 this.fireEvent("invalidated", this);
53029             }
53030         }
53031         
53032     },
53033
53034     animateCollapse : function(){
53035         // overridden
53036     },
53037
53038     /**
53039      * Expands this region if it was previously collapsed.
53040      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53041      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53042      */
53043     expand : function(e, skipAnim){
53044         if(e) {
53045             e.stopPropagation();
53046         }
53047         if(!this.collapsed || this.el.hasActiveFx()) {
53048             return;
53049         }
53050         if(this.isSlid){
53051             this.afterSlideIn();
53052             skipAnim = true;
53053         }
53054         this.collapsed = false;
53055         if(this.config.animate && skipAnim !== true){
53056             this.animateExpand();
53057         }else{
53058             this.el.show();
53059             if(this.split){
53060                 this.split.el.show();
53061             }
53062             this.collapsedEl.setLocation(-2000,-2000);
53063             this.collapsedEl.hide();
53064             this.fireEvent("invalidated", this);
53065             this.fireEvent("expanded", this);
53066         }
53067     },
53068
53069     animateExpand : function(){
53070         // overridden
53071     },
53072
53073     initTabs : function()
53074     {
53075         this.bodyEl.setStyle("overflow", "hidden");
53076         var ts = new Roo.TabPanel(
53077                 this.bodyEl.dom,
53078                 {
53079                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53080                     disableTooltips: this.config.disableTabTips,
53081                     toolbar : this.config.toolbar
53082                 }
53083         );
53084         if(this.config.hideTabs){
53085             ts.stripWrap.setDisplayed(false);
53086         }
53087         this.tabs = ts;
53088         ts.resizeTabs = this.config.resizeTabs === true;
53089         ts.minTabWidth = this.config.minTabWidth || 40;
53090         ts.maxTabWidth = this.config.maxTabWidth || 250;
53091         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53092         ts.monitorResize = false;
53093         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53094         ts.bodyEl.addClass('x-layout-tabs-body');
53095         this.panels.each(this.initPanelAsTab, this);
53096     },
53097
53098     initPanelAsTab : function(panel){
53099         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53100                     this.config.closeOnTab && panel.isClosable());
53101         if(panel.tabTip !== undefined){
53102             ti.setTooltip(panel.tabTip);
53103         }
53104         ti.on("activate", function(){
53105               this.setActivePanel(panel);
53106         }, this);
53107         if(this.config.closeOnTab){
53108             ti.on("beforeclose", function(t, e){
53109                 e.cancel = true;
53110                 this.remove(panel);
53111             }, this);
53112         }
53113         return ti;
53114     },
53115
53116     updatePanelTitle : function(panel, title){
53117         if(this.activePanel == panel){
53118             this.updateTitle(title);
53119         }
53120         if(this.tabs){
53121             var ti = this.tabs.getTab(panel.getEl().id);
53122             ti.setText(title);
53123             if(panel.tabTip !== undefined){
53124                 ti.setTooltip(panel.tabTip);
53125             }
53126         }
53127     },
53128
53129     updateTitle : function(title){
53130         if(this.titleTextEl && !this.config.title){
53131             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53132         }
53133     },
53134
53135     setActivePanel : function(panel){
53136         panel = this.getPanel(panel);
53137         if(this.activePanel && this.activePanel != panel){
53138             this.activePanel.setActiveState(false);
53139         }
53140         this.activePanel = panel;
53141         panel.setActiveState(true);
53142         if(this.panelSize){
53143             panel.setSize(this.panelSize.width, this.panelSize.height);
53144         }
53145         if(this.closeBtn){
53146             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53147         }
53148         this.updateTitle(panel.getTitle());
53149         if(this.tabs){
53150             this.fireEvent("invalidated", this);
53151         }
53152         this.fireEvent("panelactivated", this, panel);
53153     },
53154
53155     /**
53156      * Shows the specified panel.
53157      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53158      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53159      */
53160     showPanel : function(panel)
53161     {
53162         panel = this.getPanel(panel);
53163         if(panel){
53164             if(this.tabs){
53165                 var tab = this.tabs.getTab(panel.getEl().id);
53166                 if(tab.isHidden()){
53167                     this.tabs.unhideTab(tab.id);
53168                 }
53169                 tab.activate();
53170             }else{
53171                 this.setActivePanel(panel);
53172             }
53173         }
53174         return panel;
53175     },
53176
53177     /**
53178      * Get the active panel for this region.
53179      * @return {Roo.ContentPanel} The active panel or null
53180      */
53181     getActivePanel : function(){
53182         return this.activePanel;
53183     },
53184
53185     validateVisibility : function(){
53186         if(this.panels.getCount() < 1){
53187             this.updateTitle("&#160;");
53188             this.closeBtn.hide();
53189             this.hide();
53190         }else{
53191             if(!this.isVisible()){
53192                 this.show();
53193             }
53194         }
53195     },
53196
53197     /**
53198      * Adds the passed ContentPanel(s) to this region.
53199      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53200      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53201      */
53202     add : function(panel){
53203         if(arguments.length > 1){
53204             for(var i = 0, len = arguments.length; i < len; i++) {
53205                 this.add(arguments[i]);
53206             }
53207             return null;
53208         }
53209         if(this.hasPanel(panel)){
53210             this.showPanel(panel);
53211             return panel;
53212         }
53213         panel.setRegion(this);
53214         this.panels.add(panel);
53215         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53216             this.bodyEl.dom.appendChild(panel.getEl().dom);
53217             if(panel.background !== true){
53218                 this.setActivePanel(panel);
53219             }
53220             this.fireEvent("paneladded", this, panel);
53221             return panel;
53222         }
53223         if(!this.tabs){
53224             this.initTabs();
53225         }else{
53226             this.initPanelAsTab(panel);
53227         }
53228         if(panel.background !== true){
53229             this.tabs.activate(panel.getEl().id);
53230         }
53231         this.fireEvent("paneladded", this, panel);
53232         return panel;
53233     },
53234
53235     /**
53236      * Hides the tab for the specified panel.
53237      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53238      */
53239     hidePanel : function(panel){
53240         if(this.tabs && (panel = this.getPanel(panel))){
53241             this.tabs.hideTab(panel.getEl().id);
53242         }
53243     },
53244
53245     /**
53246      * Unhides the tab for a previously hidden panel.
53247      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53248      */
53249     unhidePanel : function(panel){
53250         if(this.tabs && (panel = this.getPanel(panel))){
53251             this.tabs.unhideTab(panel.getEl().id);
53252         }
53253     },
53254
53255     clearPanels : function(){
53256         while(this.panels.getCount() > 0){
53257              this.remove(this.panels.first());
53258         }
53259     },
53260
53261     /**
53262      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53263      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53264      * @param {Boolean} preservePanel Overrides the config preservePanel option
53265      * @return {Roo.ContentPanel} The panel that was removed
53266      */
53267     remove : function(panel, preservePanel){
53268         panel = this.getPanel(panel);
53269         if(!panel){
53270             return null;
53271         }
53272         var e = {};
53273         this.fireEvent("beforeremove", this, panel, e);
53274         if(e.cancel === true){
53275             return null;
53276         }
53277         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53278         var panelId = panel.getId();
53279         this.panels.removeKey(panelId);
53280         if(preservePanel){
53281             document.body.appendChild(panel.getEl().dom);
53282         }
53283         if(this.tabs){
53284             this.tabs.removeTab(panel.getEl().id);
53285         }else if (!preservePanel){
53286             this.bodyEl.dom.removeChild(panel.getEl().dom);
53287         }
53288         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53289             var p = this.panels.first();
53290             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53291             tempEl.appendChild(p.getEl().dom);
53292             this.bodyEl.update("");
53293             this.bodyEl.dom.appendChild(p.getEl().dom);
53294             tempEl = null;
53295             this.updateTitle(p.getTitle());
53296             this.tabs = null;
53297             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53298             this.setActivePanel(p);
53299         }
53300         panel.setRegion(null);
53301         if(this.activePanel == panel){
53302             this.activePanel = null;
53303         }
53304         if(this.config.autoDestroy !== false && preservePanel !== true){
53305             try{panel.destroy();}catch(e){}
53306         }
53307         this.fireEvent("panelremoved", this, panel);
53308         return panel;
53309     },
53310
53311     /**
53312      * Returns the TabPanel component used by this region
53313      * @return {Roo.TabPanel}
53314      */
53315     getTabs : function(){
53316         return this.tabs;
53317     },
53318
53319     createTool : function(parentEl, className){
53320         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53321             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53322         btn.addClassOnOver("x-layout-tools-button-over");
53323         return btn;
53324     }
53325 });/*
53326  * Based on:
53327  * Ext JS Library 1.1.1
53328  * Copyright(c) 2006-2007, Ext JS, LLC.
53329  *
53330  * Originally Released Under LGPL - original licence link has changed is not relivant.
53331  *
53332  * Fork - LGPL
53333  * <script type="text/javascript">
53334  */
53335  
53336
53337
53338 /**
53339  * @class Roo.SplitLayoutRegion
53340  * @extends Roo.LayoutRegion
53341  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53342  */
53343 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53344     this.cursor = cursor;
53345     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53346 };
53347
53348 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53349     splitTip : "Drag to resize.",
53350     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53351     useSplitTips : false,
53352
53353     applyConfig : function(config){
53354         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53355         if(config.split){
53356             if(!this.split){
53357                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53358                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53359                 /** The SplitBar for this region 
53360                 * @type Roo.SplitBar */
53361                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53362                 this.split.on("moved", this.onSplitMove, this);
53363                 this.split.useShim = config.useShim === true;
53364                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53365                 if(this.useSplitTips){
53366                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53367                 }
53368                 if(config.collapsible){
53369                     this.split.el.on("dblclick", this.collapse,  this);
53370                 }
53371             }
53372             if(typeof config.minSize != "undefined"){
53373                 this.split.minSize = config.minSize;
53374             }
53375             if(typeof config.maxSize != "undefined"){
53376                 this.split.maxSize = config.maxSize;
53377             }
53378             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53379                 this.hideSplitter();
53380             }
53381         }
53382     },
53383
53384     getHMaxSize : function(){
53385          var cmax = this.config.maxSize || 10000;
53386          var center = this.mgr.getRegion("center");
53387          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53388     },
53389
53390     getVMaxSize : function(){
53391          var cmax = this.config.maxSize || 10000;
53392          var center = this.mgr.getRegion("center");
53393          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53394     },
53395
53396     onSplitMove : function(split, newSize){
53397         this.fireEvent("resized", this, newSize);
53398     },
53399     
53400     /** 
53401      * Returns the {@link Roo.SplitBar} for this region.
53402      * @return {Roo.SplitBar}
53403      */
53404     getSplitBar : function(){
53405         return this.split;
53406     },
53407     
53408     hide : function(){
53409         this.hideSplitter();
53410         Roo.SplitLayoutRegion.superclass.hide.call(this);
53411     },
53412
53413     hideSplitter : function(){
53414         if(this.split){
53415             this.split.el.setLocation(-2000,-2000);
53416             this.split.el.hide();
53417         }
53418     },
53419
53420     show : function(){
53421         if(this.split){
53422             this.split.el.show();
53423         }
53424         Roo.SplitLayoutRegion.superclass.show.call(this);
53425     },
53426     
53427     beforeSlide: function(){
53428         if(Roo.isGecko){// firefox overflow auto bug workaround
53429             this.bodyEl.clip();
53430             if(this.tabs) {
53431                 this.tabs.bodyEl.clip();
53432             }
53433             if(this.activePanel){
53434                 this.activePanel.getEl().clip();
53435                 
53436                 if(this.activePanel.beforeSlide){
53437                     this.activePanel.beforeSlide();
53438                 }
53439             }
53440         }
53441     },
53442     
53443     afterSlide : function(){
53444         if(Roo.isGecko){// firefox overflow auto bug workaround
53445             this.bodyEl.unclip();
53446             if(this.tabs) {
53447                 this.tabs.bodyEl.unclip();
53448             }
53449             if(this.activePanel){
53450                 this.activePanel.getEl().unclip();
53451                 if(this.activePanel.afterSlide){
53452                     this.activePanel.afterSlide();
53453                 }
53454             }
53455         }
53456     },
53457
53458     initAutoHide : function(){
53459         if(this.autoHide !== false){
53460             if(!this.autoHideHd){
53461                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53462                 this.autoHideHd = {
53463                     "mouseout": function(e){
53464                         if(!e.within(this.el, true)){
53465                             st.delay(500);
53466                         }
53467                     },
53468                     "mouseover" : function(e){
53469                         st.cancel();
53470                     },
53471                     scope : this
53472                 };
53473             }
53474             this.el.on(this.autoHideHd);
53475         }
53476     },
53477
53478     clearAutoHide : function(){
53479         if(this.autoHide !== false){
53480             this.el.un("mouseout", this.autoHideHd.mouseout);
53481             this.el.un("mouseover", this.autoHideHd.mouseover);
53482         }
53483     },
53484
53485     clearMonitor : function(){
53486         Roo.get(document).un("click", this.slideInIf, this);
53487     },
53488
53489     // these names are backwards but not changed for compat
53490     slideOut : function(){
53491         if(this.isSlid || this.el.hasActiveFx()){
53492             return;
53493         }
53494         this.isSlid = true;
53495         if(this.collapseBtn){
53496             this.collapseBtn.hide();
53497         }
53498         this.closeBtnState = this.closeBtn.getStyle('display');
53499         this.closeBtn.hide();
53500         if(this.stickBtn){
53501             this.stickBtn.show();
53502         }
53503         this.el.show();
53504         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53505         this.beforeSlide();
53506         this.el.setStyle("z-index", 10001);
53507         this.el.slideIn(this.getSlideAnchor(), {
53508             callback: function(){
53509                 this.afterSlide();
53510                 this.initAutoHide();
53511                 Roo.get(document).on("click", this.slideInIf, this);
53512                 this.fireEvent("slideshow", this);
53513             },
53514             scope: this,
53515             block: true
53516         });
53517     },
53518
53519     afterSlideIn : function(){
53520         this.clearAutoHide();
53521         this.isSlid = false;
53522         this.clearMonitor();
53523         this.el.setStyle("z-index", "");
53524         if(this.collapseBtn){
53525             this.collapseBtn.show();
53526         }
53527         this.closeBtn.setStyle('display', this.closeBtnState);
53528         if(this.stickBtn){
53529             this.stickBtn.hide();
53530         }
53531         this.fireEvent("slidehide", this);
53532     },
53533
53534     slideIn : function(cb){
53535         if(!this.isSlid || this.el.hasActiveFx()){
53536             Roo.callback(cb);
53537             return;
53538         }
53539         this.isSlid = false;
53540         this.beforeSlide();
53541         this.el.slideOut(this.getSlideAnchor(), {
53542             callback: function(){
53543                 this.el.setLeftTop(-10000, -10000);
53544                 this.afterSlide();
53545                 this.afterSlideIn();
53546                 Roo.callback(cb);
53547             },
53548             scope: this,
53549             block: true
53550         });
53551     },
53552     
53553     slideInIf : function(e){
53554         if(!e.within(this.el)){
53555             this.slideIn();
53556         }
53557     },
53558
53559     animateCollapse : function(){
53560         this.beforeSlide();
53561         this.el.setStyle("z-index", 20000);
53562         var anchor = this.getSlideAnchor();
53563         this.el.slideOut(anchor, {
53564             callback : function(){
53565                 this.el.setStyle("z-index", "");
53566                 this.collapsedEl.slideIn(anchor, {duration:.3});
53567                 this.afterSlide();
53568                 this.el.setLocation(-10000,-10000);
53569                 this.el.hide();
53570                 this.fireEvent("collapsed", this);
53571             },
53572             scope: this,
53573             block: true
53574         });
53575     },
53576
53577     animateExpand : function(){
53578         this.beforeSlide();
53579         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53580         this.el.setStyle("z-index", 20000);
53581         this.collapsedEl.hide({
53582             duration:.1
53583         });
53584         this.el.slideIn(this.getSlideAnchor(), {
53585             callback : function(){
53586                 this.el.setStyle("z-index", "");
53587                 this.afterSlide();
53588                 if(this.split){
53589                     this.split.el.show();
53590                 }
53591                 this.fireEvent("invalidated", this);
53592                 this.fireEvent("expanded", this);
53593             },
53594             scope: this,
53595             block: true
53596         });
53597     },
53598
53599     anchors : {
53600         "west" : "left",
53601         "east" : "right",
53602         "north" : "top",
53603         "south" : "bottom"
53604     },
53605
53606     sanchors : {
53607         "west" : "l",
53608         "east" : "r",
53609         "north" : "t",
53610         "south" : "b"
53611     },
53612
53613     canchors : {
53614         "west" : "tl-tr",
53615         "east" : "tr-tl",
53616         "north" : "tl-bl",
53617         "south" : "bl-tl"
53618     },
53619
53620     getAnchor : function(){
53621         return this.anchors[this.position];
53622     },
53623
53624     getCollapseAnchor : function(){
53625         return this.canchors[this.position];
53626     },
53627
53628     getSlideAnchor : function(){
53629         return this.sanchors[this.position];
53630     },
53631
53632     getAlignAdj : function(){
53633         var cm = this.cmargins;
53634         switch(this.position){
53635             case "west":
53636                 return [0, 0];
53637             break;
53638             case "east":
53639                 return [0, 0];
53640             break;
53641             case "north":
53642                 return [0, 0];
53643             break;
53644             case "south":
53645                 return [0, 0];
53646             break;
53647         }
53648     },
53649
53650     getExpandAdj : function(){
53651         var c = this.collapsedEl, cm = this.cmargins;
53652         switch(this.position){
53653             case "west":
53654                 return [-(cm.right+c.getWidth()+cm.left), 0];
53655             break;
53656             case "east":
53657                 return [cm.right+c.getWidth()+cm.left, 0];
53658             break;
53659             case "north":
53660                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53661             break;
53662             case "south":
53663                 return [0, cm.top+cm.bottom+c.getHeight()];
53664             break;
53665         }
53666     }
53667 });/*
53668  * Based on:
53669  * Ext JS Library 1.1.1
53670  * Copyright(c) 2006-2007, Ext JS, LLC.
53671  *
53672  * Originally Released Under LGPL - original licence link has changed is not relivant.
53673  *
53674  * Fork - LGPL
53675  * <script type="text/javascript">
53676  */
53677 /*
53678  * These classes are private internal classes
53679  */
53680 Roo.CenterLayoutRegion = function(mgr, config){
53681     Roo.LayoutRegion.call(this, mgr, config, "center");
53682     this.visible = true;
53683     this.minWidth = config.minWidth || 20;
53684     this.minHeight = config.minHeight || 20;
53685 };
53686
53687 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53688     hide : function(){
53689         // center panel can't be hidden
53690     },
53691     
53692     show : function(){
53693         // center panel can't be hidden
53694     },
53695     
53696     getMinWidth: function(){
53697         return this.minWidth;
53698     },
53699     
53700     getMinHeight: function(){
53701         return this.minHeight;
53702     }
53703 });
53704
53705
53706 Roo.NorthLayoutRegion = function(mgr, config){
53707     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53708     if(this.split){
53709         this.split.placement = Roo.SplitBar.TOP;
53710         this.split.orientation = Roo.SplitBar.VERTICAL;
53711         this.split.el.addClass("x-layout-split-v");
53712     }
53713     var size = config.initialSize || config.height;
53714     if(typeof size != "undefined"){
53715         this.el.setHeight(size);
53716     }
53717 };
53718 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53719     orientation: Roo.SplitBar.VERTICAL,
53720     getBox : function(){
53721         if(this.collapsed){
53722             return this.collapsedEl.getBox();
53723         }
53724         var box = this.el.getBox();
53725         if(this.split){
53726             box.height += this.split.el.getHeight();
53727         }
53728         return box;
53729     },
53730     
53731     updateBox : function(box){
53732         if(this.split && !this.collapsed){
53733             box.height -= this.split.el.getHeight();
53734             this.split.el.setLeft(box.x);
53735             this.split.el.setTop(box.y+box.height);
53736             this.split.el.setWidth(box.width);
53737         }
53738         if(this.collapsed){
53739             this.updateBody(box.width, null);
53740         }
53741         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53742     }
53743 });
53744
53745 Roo.SouthLayoutRegion = function(mgr, config){
53746     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53747     if(this.split){
53748         this.split.placement = Roo.SplitBar.BOTTOM;
53749         this.split.orientation = Roo.SplitBar.VERTICAL;
53750         this.split.el.addClass("x-layout-split-v");
53751     }
53752     var size = config.initialSize || config.height;
53753     if(typeof size != "undefined"){
53754         this.el.setHeight(size);
53755     }
53756 };
53757 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53758     orientation: Roo.SplitBar.VERTICAL,
53759     getBox : function(){
53760         if(this.collapsed){
53761             return this.collapsedEl.getBox();
53762         }
53763         var box = this.el.getBox();
53764         if(this.split){
53765             var sh = this.split.el.getHeight();
53766             box.height += sh;
53767             box.y -= sh;
53768         }
53769         return box;
53770     },
53771     
53772     updateBox : function(box){
53773         if(this.split && !this.collapsed){
53774             var sh = this.split.el.getHeight();
53775             box.height -= sh;
53776             box.y += sh;
53777             this.split.el.setLeft(box.x);
53778             this.split.el.setTop(box.y-sh);
53779             this.split.el.setWidth(box.width);
53780         }
53781         if(this.collapsed){
53782             this.updateBody(box.width, null);
53783         }
53784         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53785     }
53786 });
53787
53788 Roo.EastLayoutRegion = function(mgr, config){
53789     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53790     if(this.split){
53791         this.split.placement = Roo.SplitBar.RIGHT;
53792         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53793         this.split.el.addClass("x-layout-split-h");
53794     }
53795     var size = config.initialSize || config.width;
53796     if(typeof size != "undefined"){
53797         this.el.setWidth(size);
53798     }
53799 };
53800 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53801     orientation: Roo.SplitBar.HORIZONTAL,
53802     getBox : function(){
53803         if(this.collapsed){
53804             return this.collapsedEl.getBox();
53805         }
53806         var box = this.el.getBox();
53807         if(this.split){
53808             var sw = this.split.el.getWidth();
53809             box.width += sw;
53810             box.x -= sw;
53811         }
53812         return box;
53813     },
53814
53815     updateBox : function(box){
53816         if(this.split && !this.collapsed){
53817             var sw = this.split.el.getWidth();
53818             box.width -= sw;
53819             this.split.el.setLeft(box.x);
53820             this.split.el.setTop(box.y);
53821             this.split.el.setHeight(box.height);
53822             box.x += sw;
53823         }
53824         if(this.collapsed){
53825             this.updateBody(null, box.height);
53826         }
53827         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53828     }
53829 });
53830
53831 Roo.WestLayoutRegion = function(mgr, config){
53832     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53833     if(this.split){
53834         this.split.placement = Roo.SplitBar.LEFT;
53835         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53836         this.split.el.addClass("x-layout-split-h");
53837     }
53838     var size = config.initialSize || config.width;
53839     if(typeof size != "undefined"){
53840         this.el.setWidth(size);
53841     }
53842 };
53843 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53844     orientation: Roo.SplitBar.HORIZONTAL,
53845     getBox : function(){
53846         if(this.collapsed){
53847             return this.collapsedEl.getBox();
53848         }
53849         var box = this.el.getBox();
53850         if(this.split){
53851             box.width += this.split.el.getWidth();
53852         }
53853         return box;
53854     },
53855     
53856     updateBox : function(box){
53857         if(this.split && !this.collapsed){
53858             var sw = this.split.el.getWidth();
53859             box.width -= sw;
53860             this.split.el.setLeft(box.x+box.width);
53861             this.split.el.setTop(box.y);
53862             this.split.el.setHeight(box.height);
53863         }
53864         if(this.collapsed){
53865             this.updateBody(null, box.height);
53866         }
53867         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53868     }
53869 });
53870 /*
53871  * Based on:
53872  * Ext JS Library 1.1.1
53873  * Copyright(c) 2006-2007, Ext JS, LLC.
53874  *
53875  * Originally Released Under LGPL - original licence link has changed is not relivant.
53876  *
53877  * Fork - LGPL
53878  * <script type="text/javascript">
53879  */
53880  
53881  
53882 /*
53883  * Private internal class for reading and applying state
53884  */
53885 Roo.LayoutStateManager = function(layout){
53886      // default empty state
53887      this.state = {
53888         north: {},
53889         south: {},
53890         east: {},
53891         west: {}       
53892     };
53893 };
53894
53895 Roo.LayoutStateManager.prototype = {
53896     init : function(layout, provider){
53897         this.provider = provider;
53898         var state = provider.get(layout.id+"-layout-state");
53899         if(state){
53900             var wasUpdating = layout.isUpdating();
53901             if(!wasUpdating){
53902                 layout.beginUpdate();
53903             }
53904             for(var key in state){
53905                 if(typeof state[key] != "function"){
53906                     var rstate = state[key];
53907                     var r = layout.getRegion(key);
53908                     if(r && rstate){
53909                         if(rstate.size){
53910                             r.resizeTo(rstate.size);
53911                         }
53912                         if(rstate.collapsed == true){
53913                             r.collapse(true);
53914                         }else{
53915                             r.expand(null, true);
53916                         }
53917                     }
53918                 }
53919             }
53920             if(!wasUpdating){
53921                 layout.endUpdate();
53922             }
53923             this.state = state; 
53924         }
53925         this.layout = layout;
53926         layout.on("regionresized", this.onRegionResized, this);
53927         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53928         layout.on("regionexpanded", this.onRegionExpanded, this);
53929     },
53930     
53931     storeState : function(){
53932         this.provider.set(this.layout.id+"-layout-state", this.state);
53933     },
53934     
53935     onRegionResized : function(region, newSize){
53936         this.state[region.getPosition()].size = newSize;
53937         this.storeState();
53938     },
53939     
53940     onRegionCollapsed : function(region){
53941         this.state[region.getPosition()].collapsed = true;
53942         this.storeState();
53943     },
53944     
53945     onRegionExpanded : function(region){
53946         this.state[region.getPosition()].collapsed = false;
53947         this.storeState();
53948     }
53949 };/*
53950  * Based on:
53951  * Ext JS Library 1.1.1
53952  * Copyright(c) 2006-2007, Ext JS, LLC.
53953  *
53954  * Originally Released Under LGPL - original licence link has changed is not relivant.
53955  *
53956  * Fork - LGPL
53957  * <script type="text/javascript">
53958  */
53959 /**
53960  * @class Roo.ContentPanel
53961  * @extends Roo.util.Observable
53962  * A basic ContentPanel element.
53963  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
53964  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
53965  * @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
53966  * @cfg {Boolean}   closable      True if the panel can be closed/removed
53967  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
53968  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
53969  * @cfg {Toolbar}   toolbar       A toolbar for this panel
53970  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
53971  * @cfg {String} title          The title for this panel
53972  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
53973  * @cfg {String} url            Calls {@link #setUrl} with this value
53974  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
53975  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
53976  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
53977  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
53978
53979  * @constructor
53980  * Create a new ContentPanel.
53981  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
53982  * @param {String/Object} config A string to set only the title or a config object
53983  * @param {String} content (optional) Set the HTML content for this panel
53984  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
53985  */
53986 Roo.ContentPanel = function(el, config, content){
53987     
53988      
53989     /*
53990     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
53991         config = el;
53992         el = Roo.id();
53993     }
53994     if (config && config.parentLayout) { 
53995         el = config.parentLayout.el.createChild(); 
53996     }
53997     */
53998     if(el.autoCreate){ // xtype is available if this is called from factory
53999         config = el;
54000         el = Roo.id();
54001     }
54002     this.el = Roo.get(el);
54003     if(!this.el && config && config.autoCreate){
54004         if(typeof config.autoCreate == "object"){
54005             if(!config.autoCreate.id){
54006                 config.autoCreate.id = config.id||el;
54007             }
54008             this.el = Roo.DomHelper.append(document.body,
54009                         config.autoCreate, true);
54010         }else{
54011             this.el = Roo.DomHelper.append(document.body,
54012                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54013         }
54014     }
54015     this.closable = false;
54016     this.loaded = false;
54017     this.active = false;
54018     if(typeof config == "string"){
54019         this.title = config;
54020     }else{
54021         Roo.apply(this, config);
54022     }
54023     
54024     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54025         this.wrapEl = this.el.wrap();
54026         this.toolbar.container = this.el.insertSibling(false, 'before');
54027         this.toolbar = new Roo.Toolbar(this.toolbar);
54028     }
54029     
54030     // xtype created footer. - not sure if will work as we normally have to render first..
54031     if (this.footer && !this.footer.el && this.footer.xtype) {
54032         if (!this.wrapEl) {
54033             this.wrapEl = this.el.wrap();
54034         }
54035     
54036         this.footer.container = this.wrapEl.createChild();
54037          
54038         this.footer = Roo.factory(this.footer, Roo);
54039         
54040     }
54041     
54042     if(this.resizeEl){
54043         this.resizeEl = Roo.get(this.resizeEl, true);
54044     }else{
54045         this.resizeEl = this.el;
54046     }
54047     // handle view.xtype
54048     
54049  
54050     
54051     
54052     this.addEvents({
54053         /**
54054          * @event activate
54055          * Fires when this panel is activated. 
54056          * @param {Roo.ContentPanel} this
54057          */
54058         "activate" : true,
54059         /**
54060          * @event deactivate
54061          * Fires when this panel is activated. 
54062          * @param {Roo.ContentPanel} this
54063          */
54064         "deactivate" : true,
54065
54066         /**
54067          * @event resize
54068          * Fires when this panel is resized if fitToFrame is true.
54069          * @param {Roo.ContentPanel} this
54070          * @param {Number} width The width after any component adjustments
54071          * @param {Number} height The height after any component adjustments
54072          */
54073         "resize" : true,
54074         
54075          /**
54076          * @event render
54077          * Fires when this tab is created
54078          * @param {Roo.ContentPanel} this
54079          */
54080         "render" : true
54081          
54082         
54083     });
54084     
54085
54086     
54087     
54088     if(this.autoScroll){
54089         this.resizeEl.setStyle("overflow", "auto");
54090     } else {
54091         // fix randome scrolling
54092         this.el.on('scroll', function() {
54093             Roo.log('fix random scolling');
54094             this.scrollTo('top',0); 
54095         });
54096     }
54097     content = content || this.content;
54098     if(content){
54099         this.setContent(content);
54100     }
54101     if(config && config.url){
54102         this.setUrl(this.url, this.params, this.loadOnce);
54103     }
54104     
54105     
54106     
54107     Roo.ContentPanel.superclass.constructor.call(this);
54108     
54109     if (this.view && typeof(this.view.xtype) != 'undefined') {
54110         this.view.el = this.el.appendChild(document.createElement("div"));
54111         this.view = Roo.factory(this.view); 
54112         this.view.render  &&  this.view.render(false, '');  
54113     }
54114     
54115     
54116     this.fireEvent('render', this);
54117 };
54118
54119 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54120     tabTip:'',
54121     setRegion : function(region){
54122         this.region = region;
54123         if(region){
54124            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54125         }else{
54126            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54127         } 
54128     },
54129     
54130     /**
54131      * Returns the toolbar for this Panel if one was configured. 
54132      * @return {Roo.Toolbar} 
54133      */
54134     getToolbar : function(){
54135         return this.toolbar;
54136     },
54137     
54138     setActiveState : function(active){
54139         this.active = active;
54140         if(!active){
54141             this.fireEvent("deactivate", this);
54142         }else{
54143             this.fireEvent("activate", this);
54144         }
54145     },
54146     /**
54147      * Updates this panel's element
54148      * @param {String} content The new content
54149      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54150     */
54151     setContent : function(content, loadScripts){
54152         this.el.update(content, loadScripts);
54153     },
54154
54155     ignoreResize : function(w, h){
54156         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54157             return true;
54158         }else{
54159             this.lastSize = {width: w, height: h};
54160             return false;
54161         }
54162     },
54163     /**
54164      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54165      * @return {Roo.UpdateManager} The UpdateManager
54166      */
54167     getUpdateManager : function(){
54168         return this.el.getUpdateManager();
54169     },
54170      /**
54171      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54172      * @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:
54173 <pre><code>
54174 panel.load({
54175     url: "your-url.php",
54176     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54177     callback: yourFunction,
54178     scope: yourObject, //(optional scope)
54179     discardUrl: false,
54180     nocache: false,
54181     text: "Loading...",
54182     timeout: 30,
54183     scripts: false
54184 });
54185 </code></pre>
54186      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54187      * 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.
54188      * @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}
54189      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54190      * @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.
54191      * @return {Roo.ContentPanel} this
54192      */
54193     load : function(){
54194         var um = this.el.getUpdateManager();
54195         um.update.apply(um, arguments);
54196         return this;
54197     },
54198
54199
54200     /**
54201      * 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.
54202      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54203      * @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)
54204      * @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)
54205      * @return {Roo.UpdateManager} The UpdateManager
54206      */
54207     setUrl : function(url, params, loadOnce){
54208         if(this.refreshDelegate){
54209             this.removeListener("activate", this.refreshDelegate);
54210         }
54211         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54212         this.on("activate", this.refreshDelegate);
54213         return this.el.getUpdateManager();
54214     },
54215     
54216     _handleRefresh : function(url, params, loadOnce){
54217         if(!loadOnce || !this.loaded){
54218             var updater = this.el.getUpdateManager();
54219             updater.update(url, params, this._setLoaded.createDelegate(this));
54220         }
54221     },
54222     
54223     _setLoaded : function(){
54224         this.loaded = true;
54225     }, 
54226     
54227     /**
54228      * Returns this panel's id
54229      * @return {String} 
54230      */
54231     getId : function(){
54232         return this.el.id;
54233     },
54234     
54235     /** 
54236      * Returns this panel's element - used by regiosn to add.
54237      * @return {Roo.Element} 
54238      */
54239     getEl : function(){
54240         return this.wrapEl || this.el;
54241     },
54242     
54243     adjustForComponents : function(width, height)
54244     {
54245         //Roo.log('adjustForComponents ');
54246         if(this.resizeEl != this.el){
54247             width -= this.el.getFrameWidth('lr');
54248             height -= this.el.getFrameWidth('tb');
54249         }
54250         if(this.toolbar){
54251             var te = this.toolbar.getEl();
54252             height -= te.getHeight();
54253             te.setWidth(width);
54254         }
54255         if(this.footer){
54256             var te = this.footer.getEl();
54257             //Roo.log("footer:" + te.getHeight());
54258             
54259             height -= te.getHeight();
54260             te.setWidth(width);
54261         }
54262         
54263         
54264         if(this.adjustments){
54265             width += this.adjustments[0];
54266             height += this.adjustments[1];
54267         }
54268         return {"width": width, "height": height};
54269     },
54270     
54271     setSize : function(width, height){
54272         if(this.fitToFrame && !this.ignoreResize(width, height)){
54273             if(this.fitContainer && this.resizeEl != this.el){
54274                 this.el.setSize(width, height);
54275             }
54276             var size = this.adjustForComponents(width, height);
54277             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54278             this.fireEvent('resize', this, size.width, size.height);
54279         }
54280     },
54281     
54282     /**
54283      * Returns this panel's title
54284      * @return {String} 
54285      */
54286     getTitle : function(){
54287         return this.title;
54288     },
54289     
54290     /**
54291      * Set this panel's title
54292      * @param {String} title
54293      */
54294     setTitle : function(title){
54295         this.title = title;
54296         if(this.region){
54297             this.region.updatePanelTitle(this, title);
54298         }
54299     },
54300     
54301     /**
54302      * Returns true is this panel was configured to be closable
54303      * @return {Boolean} 
54304      */
54305     isClosable : function(){
54306         return this.closable;
54307     },
54308     
54309     beforeSlide : function(){
54310         this.el.clip();
54311         this.resizeEl.clip();
54312     },
54313     
54314     afterSlide : function(){
54315         this.el.unclip();
54316         this.resizeEl.unclip();
54317     },
54318     
54319     /**
54320      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54321      *   Will fail silently if the {@link #setUrl} method has not been called.
54322      *   This does not activate the panel, just updates its content.
54323      */
54324     refresh : function(){
54325         if(this.refreshDelegate){
54326            this.loaded = false;
54327            this.refreshDelegate();
54328         }
54329     },
54330     
54331     /**
54332      * Destroys this panel
54333      */
54334     destroy : function(){
54335         this.el.removeAllListeners();
54336         var tempEl = document.createElement("span");
54337         tempEl.appendChild(this.el.dom);
54338         tempEl.innerHTML = "";
54339         this.el.remove();
54340         this.el = null;
54341     },
54342     
54343     /**
54344      * form - if the content panel contains a form - this is a reference to it.
54345      * @type {Roo.form.Form}
54346      */
54347     form : false,
54348     /**
54349      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54350      *    This contains a reference to it.
54351      * @type {Roo.View}
54352      */
54353     view : false,
54354     
54355       /**
54356      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54357      * <pre><code>
54358
54359 layout.addxtype({
54360        xtype : 'Form',
54361        items: [ .... ]
54362    }
54363 );
54364
54365 </code></pre>
54366      * @param {Object} cfg Xtype definition of item to add.
54367      */
54368     
54369     addxtype : function(cfg) {
54370         // add form..
54371         if (cfg.xtype.match(/^Form$/)) {
54372             
54373             var el;
54374             //if (this.footer) {
54375             //    el = this.footer.container.insertSibling(false, 'before');
54376             //} else {
54377                 el = this.el.createChild();
54378             //}
54379
54380             this.form = new  Roo.form.Form(cfg);
54381             
54382             
54383             if ( this.form.allItems.length) {
54384                 this.form.render(el.dom);
54385             }
54386             return this.form;
54387         }
54388         // should only have one of theses..
54389         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54390             // views.. should not be just added - used named prop 'view''
54391             
54392             cfg.el = this.el.appendChild(document.createElement("div"));
54393             // factory?
54394             
54395             var ret = new Roo.factory(cfg);
54396              
54397              ret.render && ret.render(false, ''); // render blank..
54398             this.view = ret;
54399             return ret;
54400         }
54401         return false;
54402     }
54403 });
54404
54405 /**
54406  * @class Roo.GridPanel
54407  * @extends Roo.ContentPanel
54408  * @constructor
54409  * Create a new GridPanel.
54410  * @param {Roo.grid.Grid} grid The grid for this panel
54411  * @param {String/Object} config A string to set only the panel's title, or a config object
54412  */
54413 Roo.GridPanel = function(grid, config){
54414     
54415   
54416     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54417         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54418         
54419     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54420     
54421     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54422     
54423     if(this.toolbar){
54424         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54425     }
54426     // xtype created footer. - not sure if will work as we normally have to render first..
54427     if (this.footer && !this.footer.el && this.footer.xtype) {
54428         
54429         this.footer.container = this.grid.getView().getFooterPanel(true);
54430         this.footer.dataSource = this.grid.dataSource;
54431         this.footer = Roo.factory(this.footer, Roo);
54432         
54433     }
54434     
54435     grid.monitorWindowResize = false; // turn off autosizing
54436     grid.autoHeight = false;
54437     grid.autoWidth = false;
54438     this.grid = grid;
54439     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54440 };
54441
54442 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54443     getId : function(){
54444         return this.grid.id;
54445     },
54446     
54447     /**
54448      * Returns the grid for this panel
54449      * @return {Roo.grid.Grid} 
54450      */
54451     getGrid : function(){
54452         return this.grid;    
54453     },
54454     
54455     setSize : function(width, height){
54456         if(!this.ignoreResize(width, height)){
54457             var grid = this.grid;
54458             var size = this.adjustForComponents(width, height);
54459             grid.getGridEl().setSize(size.width, size.height);
54460             grid.autoSize();
54461         }
54462     },
54463     
54464     beforeSlide : function(){
54465         this.grid.getView().scroller.clip();
54466     },
54467     
54468     afterSlide : function(){
54469         this.grid.getView().scroller.unclip();
54470     },
54471     
54472     destroy : function(){
54473         this.grid.destroy();
54474         delete this.grid;
54475         Roo.GridPanel.superclass.destroy.call(this); 
54476     }
54477 });
54478
54479
54480 /**
54481  * @class Roo.NestedLayoutPanel
54482  * @extends Roo.ContentPanel
54483  * @constructor
54484  * Create a new NestedLayoutPanel.
54485  * 
54486  * 
54487  * @param {Roo.BorderLayout} layout The layout for this panel
54488  * @param {String/Object} config A string to set only the title or a config object
54489  */
54490 Roo.NestedLayoutPanel = function(layout, config)
54491 {
54492     // construct with only one argument..
54493     /* FIXME - implement nicer consturctors
54494     if (layout.layout) {
54495         config = layout;
54496         layout = config.layout;
54497         delete config.layout;
54498     }
54499     if (layout.xtype && !layout.getEl) {
54500         // then layout needs constructing..
54501         layout = Roo.factory(layout, Roo);
54502     }
54503     */
54504     
54505     
54506     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54507     
54508     layout.monitorWindowResize = false; // turn off autosizing
54509     this.layout = layout;
54510     this.layout.getEl().addClass("x-layout-nested-layout");
54511     
54512     
54513     
54514     
54515 };
54516
54517 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54518
54519     setSize : function(width, height){
54520         if(!this.ignoreResize(width, height)){
54521             var size = this.adjustForComponents(width, height);
54522             var el = this.layout.getEl();
54523             el.setSize(size.width, size.height);
54524             var touch = el.dom.offsetWidth;
54525             this.layout.layout();
54526             // ie requires a double layout on the first pass
54527             if(Roo.isIE && !this.initialized){
54528                 this.initialized = true;
54529                 this.layout.layout();
54530             }
54531         }
54532     },
54533     
54534     // activate all subpanels if not currently active..
54535     
54536     setActiveState : function(active){
54537         this.active = active;
54538         if(!active){
54539             this.fireEvent("deactivate", this);
54540             return;
54541         }
54542         
54543         this.fireEvent("activate", this);
54544         // not sure if this should happen before or after..
54545         if (!this.layout) {
54546             return; // should not happen..
54547         }
54548         var reg = false;
54549         for (var r in this.layout.regions) {
54550             reg = this.layout.getRegion(r);
54551             if (reg.getActivePanel()) {
54552                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54553                 reg.setActivePanel(reg.getActivePanel());
54554                 continue;
54555             }
54556             if (!reg.panels.length) {
54557                 continue;
54558             }
54559             reg.showPanel(reg.getPanel(0));
54560         }
54561         
54562         
54563         
54564         
54565     },
54566     
54567     /**
54568      * Returns the nested BorderLayout for this panel
54569      * @return {Roo.BorderLayout} 
54570      */
54571     getLayout : function(){
54572         return this.layout;
54573     },
54574     
54575      /**
54576      * Adds a xtype elements to the layout of the nested panel
54577      * <pre><code>
54578
54579 panel.addxtype({
54580        xtype : 'ContentPanel',
54581        region: 'west',
54582        items: [ .... ]
54583    }
54584 );
54585
54586 panel.addxtype({
54587         xtype : 'NestedLayoutPanel',
54588         region: 'west',
54589         layout: {
54590            center: { },
54591            west: { }   
54592         },
54593         items : [ ... list of content panels or nested layout panels.. ]
54594    }
54595 );
54596 </code></pre>
54597      * @param {Object} cfg Xtype definition of item to add.
54598      */
54599     addxtype : function(cfg) {
54600         return this.layout.addxtype(cfg);
54601     
54602     }
54603 });
54604
54605 Roo.ScrollPanel = function(el, config, content){
54606     config = config || {};
54607     config.fitToFrame = true;
54608     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54609     
54610     this.el.dom.style.overflow = "hidden";
54611     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54612     this.el.removeClass("x-layout-inactive-content");
54613     this.el.on("mousewheel", this.onWheel, this);
54614
54615     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54616     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54617     up.unselectable(); down.unselectable();
54618     up.on("click", this.scrollUp, this);
54619     down.on("click", this.scrollDown, this);
54620     up.addClassOnOver("x-scroller-btn-over");
54621     down.addClassOnOver("x-scroller-btn-over");
54622     up.addClassOnClick("x-scroller-btn-click");
54623     down.addClassOnClick("x-scroller-btn-click");
54624     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54625
54626     this.resizeEl = this.el;
54627     this.el = wrap; this.up = up; this.down = down;
54628 };
54629
54630 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54631     increment : 100,
54632     wheelIncrement : 5,
54633     scrollUp : function(){
54634         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54635     },
54636
54637     scrollDown : function(){
54638         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54639     },
54640
54641     afterScroll : function(){
54642         var el = this.resizeEl;
54643         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54644         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54645         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54646     },
54647
54648     setSize : function(){
54649         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54650         this.afterScroll();
54651     },
54652
54653     onWheel : function(e){
54654         var d = e.getWheelDelta();
54655         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54656         this.afterScroll();
54657         e.stopEvent();
54658     },
54659
54660     setContent : function(content, loadScripts){
54661         this.resizeEl.update(content, loadScripts);
54662     }
54663
54664 });
54665
54666
54667
54668
54669
54670
54671
54672
54673
54674 /**
54675  * @class Roo.TreePanel
54676  * @extends Roo.ContentPanel
54677  * @constructor
54678  * Create a new TreePanel. - defaults to fit/scoll contents.
54679  * @param {String/Object} config A string to set only the panel's title, or a config object
54680  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54681  */
54682 Roo.TreePanel = function(config){
54683     var el = config.el;
54684     var tree = config.tree;
54685     delete config.tree; 
54686     delete config.el; // hopefull!
54687     
54688     // wrapper for IE7 strict & safari scroll issue
54689     
54690     var treeEl = el.createChild();
54691     config.resizeEl = treeEl;
54692     
54693     
54694     
54695     Roo.TreePanel.superclass.constructor.call(this, el, config);
54696  
54697  
54698     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54699     //console.log(tree);
54700     this.on('activate', function()
54701     {
54702         if (this.tree.rendered) {
54703             return;
54704         }
54705         //console.log('render tree');
54706         this.tree.render();
54707     });
54708     // this should not be needed.. - it's actually the 'el' that resizes?
54709     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54710     
54711     //this.on('resize',  function (cp, w, h) {
54712     //        this.tree.innerCt.setWidth(w);
54713     //        this.tree.innerCt.setHeight(h);
54714     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54715     //});
54716
54717         
54718     
54719 };
54720
54721 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54722     fitToFrame : true,
54723     autoScroll : true
54724 });
54725
54726
54727
54728
54729
54730
54731
54732
54733
54734
54735
54736 /*
54737  * Based on:
54738  * Ext JS Library 1.1.1
54739  * Copyright(c) 2006-2007, Ext JS, LLC.
54740  *
54741  * Originally Released Under LGPL - original licence link has changed is not relivant.
54742  *
54743  * Fork - LGPL
54744  * <script type="text/javascript">
54745  */
54746  
54747
54748 /**
54749  * @class Roo.ReaderLayout
54750  * @extends Roo.BorderLayout
54751  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54752  * center region containing two nested regions (a top one for a list view and one for item preview below),
54753  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54754  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54755  * expedites the setup of the overall layout and regions for this common application style.
54756  * Example:
54757  <pre><code>
54758 var reader = new Roo.ReaderLayout();
54759 var CP = Roo.ContentPanel;  // shortcut for adding
54760
54761 reader.beginUpdate();
54762 reader.add("north", new CP("north", "North"));
54763 reader.add("west", new CP("west", {title: "West"}));
54764 reader.add("east", new CP("east", {title: "East"}));
54765
54766 reader.regions.listView.add(new CP("listView", "List"));
54767 reader.regions.preview.add(new CP("preview", "Preview"));
54768 reader.endUpdate();
54769 </code></pre>
54770 * @constructor
54771 * Create a new ReaderLayout
54772 * @param {Object} config Configuration options
54773 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54774 * document.body if omitted)
54775 */
54776 Roo.ReaderLayout = function(config, renderTo){
54777     var c = config || {size:{}};
54778     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54779         north: c.north !== false ? Roo.apply({
54780             split:false,
54781             initialSize: 32,
54782             titlebar: false
54783         }, c.north) : false,
54784         west: c.west !== false ? Roo.apply({
54785             split:true,
54786             initialSize: 200,
54787             minSize: 175,
54788             maxSize: 400,
54789             titlebar: true,
54790             collapsible: true,
54791             animate: true,
54792             margins:{left:5,right:0,bottom:5,top:5},
54793             cmargins:{left:5,right:5,bottom:5,top:5}
54794         }, c.west) : false,
54795         east: c.east !== false ? Roo.apply({
54796             split:true,
54797             initialSize: 200,
54798             minSize: 175,
54799             maxSize: 400,
54800             titlebar: true,
54801             collapsible: true,
54802             animate: true,
54803             margins:{left:0,right:5,bottom:5,top:5},
54804             cmargins:{left:5,right:5,bottom:5,top:5}
54805         }, c.east) : false,
54806         center: Roo.apply({
54807             tabPosition: 'top',
54808             autoScroll:false,
54809             closeOnTab: true,
54810             titlebar:false,
54811             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54812         }, c.center)
54813     });
54814
54815     this.el.addClass('x-reader');
54816
54817     this.beginUpdate();
54818
54819     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54820         south: c.preview !== false ? Roo.apply({
54821             split:true,
54822             initialSize: 200,
54823             minSize: 100,
54824             autoScroll:true,
54825             collapsible:true,
54826             titlebar: true,
54827             cmargins:{top:5,left:0, right:0, bottom:0}
54828         }, c.preview) : false,
54829         center: Roo.apply({
54830             autoScroll:false,
54831             titlebar:false,
54832             minHeight:200
54833         }, c.listView)
54834     });
54835     this.add('center', new Roo.NestedLayoutPanel(inner,
54836             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54837
54838     this.endUpdate();
54839
54840     this.regions.preview = inner.getRegion('south');
54841     this.regions.listView = inner.getRegion('center');
54842 };
54843
54844 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54845  * Based on:
54846  * Ext JS Library 1.1.1
54847  * Copyright(c) 2006-2007, Ext JS, LLC.
54848  *
54849  * Originally Released Under LGPL - original licence link has changed is not relivant.
54850  *
54851  * Fork - LGPL
54852  * <script type="text/javascript">
54853  */
54854  
54855 /**
54856  * @class Roo.grid.Grid
54857  * @extends Roo.util.Observable
54858  * This class represents the primary interface of a component based grid control.
54859  * <br><br>Usage:<pre><code>
54860  var grid = new Roo.grid.Grid("my-container-id", {
54861      ds: myDataStore,
54862      cm: myColModel,
54863      selModel: mySelectionModel,
54864      autoSizeColumns: true,
54865      monitorWindowResize: false,
54866      trackMouseOver: true
54867  });
54868  // set any options
54869  grid.render();
54870  * </code></pre>
54871  * <b>Common Problems:</b><br/>
54872  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54873  * element will correct this<br/>
54874  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54875  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54876  * are unpredictable.<br/>
54877  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54878  * grid to calculate dimensions/offsets.<br/>
54879   * @constructor
54880  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54881  * The container MUST have some type of size defined for the grid to fill. The container will be
54882  * automatically set to position relative if it isn't already.
54883  * @param {Object} config A config object that sets properties on this grid.
54884  */
54885 Roo.grid.Grid = function(container, config){
54886         // initialize the container
54887         this.container = Roo.get(container);
54888         this.container.update("");
54889         this.container.setStyle("overflow", "hidden");
54890     this.container.addClass('x-grid-container');
54891
54892     this.id = this.container.id;
54893
54894     Roo.apply(this, config);
54895     // check and correct shorthanded configs
54896     if(this.ds){
54897         this.dataSource = this.ds;
54898         delete this.ds;
54899     }
54900     if(this.cm){
54901         this.colModel = this.cm;
54902         delete this.cm;
54903     }
54904     if(this.sm){
54905         this.selModel = this.sm;
54906         delete this.sm;
54907     }
54908
54909     if (this.selModel) {
54910         this.selModel = Roo.factory(this.selModel, Roo.grid);
54911         this.sm = this.selModel;
54912         this.sm.xmodule = this.xmodule || false;
54913     }
54914     if (typeof(this.colModel.config) == 'undefined') {
54915         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54916         this.cm = this.colModel;
54917         this.cm.xmodule = this.xmodule || false;
54918     }
54919     if (this.dataSource) {
54920         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54921         this.ds = this.dataSource;
54922         this.ds.xmodule = this.xmodule || false;
54923          
54924     }
54925     
54926     
54927     
54928     if(this.width){
54929         this.container.setWidth(this.width);
54930     }
54931
54932     if(this.height){
54933         this.container.setHeight(this.height);
54934     }
54935     /** @private */
54936         this.addEvents({
54937         // raw events
54938         /**
54939          * @event click
54940          * The raw click event for the entire grid.
54941          * @param {Roo.EventObject} e
54942          */
54943         "click" : true,
54944         /**
54945          * @event dblclick
54946          * The raw dblclick event for the entire grid.
54947          * @param {Roo.EventObject} e
54948          */
54949         "dblclick" : true,
54950         /**
54951          * @event contextmenu
54952          * The raw contextmenu event for the entire grid.
54953          * @param {Roo.EventObject} e
54954          */
54955         "contextmenu" : true,
54956         /**
54957          * @event mousedown
54958          * The raw mousedown event for the entire grid.
54959          * @param {Roo.EventObject} e
54960          */
54961         "mousedown" : true,
54962         /**
54963          * @event mouseup
54964          * The raw mouseup event for the entire grid.
54965          * @param {Roo.EventObject} e
54966          */
54967         "mouseup" : true,
54968         /**
54969          * @event mouseover
54970          * The raw mouseover event for the entire grid.
54971          * @param {Roo.EventObject} e
54972          */
54973         "mouseover" : true,
54974         /**
54975          * @event mouseout
54976          * The raw mouseout event for the entire grid.
54977          * @param {Roo.EventObject} e
54978          */
54979         "mouseout" : true,
54980         /**
54981          * @event keypress
54982          * The raw keypress event for the entire grid.
54983          * @param {Roo.EventObject} e
54984          */
54985         "keypress" : true,
54986         /**
54987          * @event keydown
54988          * The raw keydown event for the entire grid.
54989          * @param {Roo.EventObject} e
54990          */
54991         "keydown" : true,
54992
54993         // custom events
54994
54995         /**
54996          * @event cellclick
54997          * Fires when a cell is clicked
54998          * @param {Grid} this
54999          * @param {Number} rowIndex
55000          * @param {Number} columnIndex
55001          * @param {Roo.EventObject} e
55002          */
55003         "cellclick" : true,
55004         /**
55005          * @event celldblclick
55006          * Fires when a cell is double clicked
55007          * @param {Grid} this
55008          * @param {Number} rowIndex
55009          * @param {Number} columnIndex
55010          * @param {Roo.EventObject} e
55011          */
55012         "celldblclick" : true,
55013         /**
55014          * @event rowclick
55015          * Fires when a row is clicked
55016          * @param {Grid} this
55017          * @param {Number} rowIndex
55018          * @param {Roo.EventObject} e
55019          */
55020         "rowclick" : true,
55021         /**
55022          * @event rowdblclick
55023          * Fires when a row is double clicked
55024          * @param {Grid} this
55025          * @param {Number} rowIndex
55026          * @param {Roo.EventObject} e
55027          */
55028         "rowdblclick" : true,
55029         /**
55030          * @event headerclick
55031          * Fires when a header is clicked
55032          * @param {Grid} this
55033          * @param {Number} columnIndex
55034          * @param {Roo.EventObject} e
55035          */
55036         "headerclick" : true,
55037         /**
55038          * @event headerdblclick
55039          * Fires when a header cell is double clicked
55040          * @param {Grid} this
55041          * @param {Number} columnIndex
55042          * @param {Roo.EventObject} e
55043          */
55044         "headerdblclick" : true,
55045         /**
55046          * @event rowcontextmenu
55047          * Fires when a row is right clicked
55048          * @param {Grid} this
55049          * @param {Number} rowIndex
55050          * @param {Roo.EventObject} e
55051          */
55052         "rowcontextmenu" : true,
55053         /**
55054          * @event cellcontextmenu
55055          * Fires when a cell is right clicked
55056          * @param {Grid} this
55057          * @param {Number} rowIndex
55058          * @param {Number} cellIndex
55059          * @param {Roo.EventObject} e
55060          */
55061          "cellcontextmenu" : true,
55062         /**
55063          * @event headercontextmenu
55064          * Fires when a header is right clicked
55065          * @param {Grid} this
55066          * @param {Number} columnIndex
55067          * @param {Roo.EventObject} e
55068          */
55069         "headercontextmenu" : true,
55070         /**
55071          * @event bodyscroll
55072          * Fires when the body element is scrolled
55073          * @param {Number} scrollLeft
55074          * @param {Number} scrollTop
55075          */
55076         "bodyscroll" : true,
55077         /**
55078          * @event columnresize
55079          * Fires when the user resizes a column
55080          * @param {Number} columnIndex
55081          * @param {Number} newSize
55082          */
55083         "columnresize" : true,
55084         /**
55085          * @event columnmove
55086          * Fires when the user moves a column
55087          * @param {Number} oldIndex
55088          * @param {Number} newIndex
55089          */
55090         "columnmove" : true,
55091         /**
55092          * @event startdrag
55093          * Fires when row(s) start being dragged
55094          * @param {Grid} this
55095          * @param {Roo.GridDD} dd The drag drop object
55096          * @param {event} e The raw browser event
55097          */
55098         "startdrag" : true,
55099         /**
55100          * @event enddrag
55101          * Fires when a drag operation is complete
55102          * @param {Grid} this
55103          * @param {Roo.GridDD} dd The drag drop object
55104          * @param {event} e The raw browser event
55105          */
55106         "enddrag" : true,
55107         /**
55108          * @event dragdrop
55109          * Fires when dragged row(s) are dropped on a valid DD target
55110          * @param {Grid} this
55111          * @param {Roo.GridDD} dd The drag drop object
55112          * @param {String} targetId The target drag drop object
55113          * @param {event} e The raw browser event
55114          */
55115         "dragdrop" : true,
55116         /**
55117          * @event dragover
55118          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55119          * @param {Grid} this
55120          * @param {Roo.GridDD} dd The drag drop object
55121          * @param {String} targetId The target drag drop object
55122          * @param {event} e The raw browser event
55123          */
55124         "dragover" : true,
55125         /**
55126          * @event dragenter
55127          *  Fires when the dragged row(s) first cross another DD target while being dragged
55128          * @param {Grid} this
55129          * @param {Roo.GridDD} dd The drag drop object
55130          * @param {String} targetId The target drag drop object
55131          * @param {event} e The raw browser event
55132          */
55133         "dragenter" : true,
55134         /**
55135          * @event dragout
55136          * Fires when the dragged row(s) leave another DD target while being dragged
55137          * @param {Grid} this
55138          * @param {Roo.GridDD} dd The drag drop object
55139          * @param {String} targetId The target drag drop object
55140          * @param {event} e The raw browser event
55141          */
55142         "dragout" : true,
55143         /**
55144          * @event rowclass
55145          * Fires when a row is rendered, so you can change add a style to it.
55146          * @param {GridView} gridview   The grid view
55147          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55148          */
55149         'rowclass' : true,
55150
55151         /**
55152          * @event render
55153          * Fires when the grid is rendered
55154          * @param {Grid} grid
55155          */
55156         'render' : true
55157     });
55158
55159     Roo.grid.Grid.superclass.constructor.call(this);
55160 };
55161 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55162     
55163     /**
55164      * @cfg {String} ddGroup - drag drop group.
55165      */
55166
55167     /**
55168      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55169      */
55170     minColumnWidth : 25,
55171
55172     /**
55173      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55174      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55175      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55176      */
55177     autoSizeColumns : false,
55178
55179     /**
55180      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55181      */
55182     autoSizeHeaders : true,
55183
55184     /**
55185      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55186      */
55187     monitorWindowResize : true,
55188
55189     /**
55190      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55191      * rows measured to get a columns size. Default is 0 (all rows).
55192      */
55193     maxRowsToMeasure : 0,
55194
55195     /**
55196      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55197      */
55198     trackMouseOver : true,
55199
55200     /**
55201     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55202     */
55203     
55204     /**
55205     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55206     */
55207     enableDragDrop : false,
55208     
55209     /**
55210     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55211     */
55212     enableColumnMove : true,
55213     
55214     /**
55215     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55216     */
55217     enableColumnHide : true,
55218     
55219     /**
55220     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55221     */
55222     enableRowHeightSync : false,
55223     
55224     /**
55225     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55226     */
55227     stripeRows : true,
55228     
55229     /**
55230     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55231     */
55232     autoHeight : false,
55233
55234     /**
55235      * @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.
55236      */
55237     autoExpandColumn : false,
55238
55239     /**
55240     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55241     * Default is 50.
55242     */
55243     autoExpandMin : 50,
55244
55245     /**
55246     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55247     */
55248     autoExpandMax : 1000,
55249
55250     /**
55251     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55252     */
55253     view : null,
55254
55255     /**
55256     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55257     */
55258     loadMask : false,
55259     /**
55260     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55261     */
55262     dropTarget: false,
55263     
55264    
55265     
55266     // private
55267     rendered : false,
55268
55269     /**
55270     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55271     * of a fixed width. Default is false.
55272     */
55273     /**
55274     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55275     */
55276     /**
55277      * Called once after all setup has been completed and the grid is ready to be rendered.
55278      * @return {Roo.grid.Grid} this
55279      */
55280     render : function()
55281     {
55282         var c = this.container;
55283         // try to detect autoHeight/width mode
55284         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55285             this.autoHeight = true;
55286         }
55287         var view = this.getView();
55288         view.init(this);
55289
55290         c.on("click", this.onClick, this);
55291         c.on("dblclick", this.onDblClick, this);
55292         c.on("contextmenu", this.onContextMenu, this);
55293         c.on("keydown", this.onKeyDown, this);
55294         if (Roo.isTouch) {
55295             c.on("touchstart", this.onTouchStart, this);
55296         }
55297
55298         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55299
55300         this.getSelectionModel().init(this);
55301
55302         view.render();
55303
55304         if(this.loadMask){
55305             this.loadMask = new Roo.LoadMask(this.container,
55306                     Roo.apply({store:this.dataSource}, this.loadMask));
55307         }
55308         
55309         
55310         if (this.toolbar && this.toolbar.xtype) {
55311             this.toolbar.container = this.getView().getHeaderPanel(true);
55312             this.toolbar = new Roo.Toolbar(this.toolbar);
55313         }
55314         if (this.footer && this.footer.xtype) {
55315             this.footer.dataSource = this.getDataSource();
55316             this.footer.container = this.getView().getFooterPanel(true);
55317             this.footer = Roo.factory(this.footer, Roo);
55318         }
55319         if (this.dropTarget && this.dropTarget.xtype) {
55320             delete this.dropTarget.xtype;
55321             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55322         }
55323         
55324         
55325         this.rendered = true;
55326         this.fireEvent('render', this);
55327         return this;
55328     },
55329
55330         /**
55331          * Reconfigures the grid to use a different Store and Column Model.
55332          * The View will be bound to the new objects and refreshed.
55333          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55334          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55335          */
55336     reconfigure : function(dataSource, colModel){
55337         if(this.loadMask){
55338             this.loadMask.destroy();
55339             this.loadMask = new Roo.LoadMask(this.container,
55340                     Roo.apply({store:dataSource}, this.loadMask));
55341         }
55342         this.view.bind(dataSource, colModel);
55343         this.dataSource = dataSource;
55344         this.colModel = colModel;
55345         this.view.refresh(true);
55346     },
55347
55348     // private
55349     onKeyDown : function(e){
55350         this.fireEvent("keydown", e);
55351     },
55352
55353     /**
55354      * Destroy this grid.
55355      * @param {Boolean} removeEl True to remove the element
55356      */
55357     destroy : function(removeEl, keepListeners){
55358         if(this.loadMask){
55359             this.loadMask.destroy();
55360         }
55361         var c = this.container;
55362         c.removeAllListeners();
55363         this.view.destroy();
55364         this.colModel.purgeListeners();
55365         if(!keepListeners){
55366             this.purgeListeners();
55367         }
55368         c.update("");
55369         if(removeEl === true){
55370             c.remove();
55371         }
55372     },
55373
55374     // private
55375     processEvent : function(name, e){
55376         // does this fire select???
55377         //Roo.log('grid:processEvent '  + name);
55378         
55379         if (name != 'touchstart' ) {
55380             this.fireEvent(name, e);    
55381         }
55382         
55383         var t = e.getTarget();
55384         var v = this.view;
55385         var header = v.findHeaderIndex(t);
55386         if(header !== false){
55387             var ename = name == 'touchstart' ? 'click' : name;
55388              
55389             this.fireEvent("header" + ename, this, header, e);
55390         }else{
55391             var row = v.findRowIndex(t);
55392             var cell = v.findCellIndex(t);
55393             if (name == 'touchstart') {
55394                 // first touch is always a click.
55395                 // hopefull this happens after selection is updated.?
55396                 name = false;
55397                 
55398                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55399                     var cs = this.selModel.getSelectedCell();
55400                     if (row == cs[0] && cell == cs[1]){
55401                         name = 'dblclick';
55402                     }
55403                 }
55404                 if (typeof(this.selModel.getSelections) != 'undefined') {
55405                     var cs = this.selModel.getSelections();
55406                     var ds = this.dataSource;
55407                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55408                         name = 'dblclick';
55409                     }
55410                 }
55411                 if (!name) {
55412                     return;
55413                 }
55414             }
55415             
55416             
55417             if(row !== false){
55418                 this.fireEvent("row" + name, this, row, e);
55419                 if(cell !== false){
55420                     this.fireEvent("cell" + name, this, row, cell, e);
55421                 }
55422             }
55423         }
55424     },
55425
55426     // private
55427     onClick : function(e){
55428         this.processEvent("click", e);
55429     },
55430    // private
55431     onTouchStart : function(e){
55432         this.processEvent("touchstart", e);
55433     },
55434
55435     // private
55436     onContextMenu : function(e, t){
55437         this.processEvent("contextmenu", e);
55438     },
55439
55440     // private
55441     onDblClick : function(e){
55442         this.processEvent("dblclick", e);
55443     },
55444
55445     // private
55446     walkCells : function(row, col, step, fn, scope){
55447         var cm = this.colModel, clen = cm.getColumnCount();
55448         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55449         if(step < 0){
55450             if(col < 0){
55451                 row--;
55452                 first = false;
55453             }
55454             while(row >= 0){
55455                 if(!first){
55456                     col = clen-1;
55457                 }
55458                 first = false;
55459                 while(col >= 0){
55460                     if(fn.call(scope || this, row, col, cm) === true){
55461                         return [row, col];
55462                     }
55463                     col--;
55464                 }
55465                 row--;
55466             }
55467         } else {
55468             if(col >= clen){
55469                 row++;
55470                 first = false;
55471             }
55472             while(row < rlen){
55473                 if(!first){
55474                     col = 0;
55475                 }
55476                 first = false;
55477                 while(col < clen){
55478                     if(fn.call(scope || this, row, col, cm) === true){
55479                         return [row, col];
55480                     }
55481                     col++;
55482                 }
55483                 row++;
55484             }
55485         }
55486         return null;
55487     },
55488
55489     // private
55490     getSelections : function(){
55491         return this.selModel.getSelections();
55492     },
55493
55494     /**
55495      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55496      * but if manual update is required this method will initiate it.
55497      */
55498     autoSize : function(){
55499         if(this.rendered){
55500             this.view.layout();
55501             if(this.view.adjustForScroll){
55502                 this.view.adjustForScroll();
55503             }
55504         }
55505     },
55506
55507     /**
55508      * Returns the grid's underlying element.
55509      * @return {Element} The element
55510      */
55511     getGridEl : function(){
55512         return this.container;
55513     },
55514
55515     // private for compatibility, overridden by editor grid
55516     stopEditing : function(){},
55517
55518     /**
55519      * Returns the grid's SelectionModel.
55520      * @return {SelectionModel}
55521      */
55522     getSelectionModel : function(){
55523         if(!this.selModel){
55524             this.selModel = new Roo.grid.RowSelectionModel();
55525         }
55526         return this.selModel;
55527     },
55528
55529     /**
55530      * Returns the grid's DataSource.
55531      * @return {DataSource}
55532      */
55533     getDataSource : function(){
55534         return this.dataSource;
55535     },
55536
55537     /**
55538      * Returns the grid's ColumnModel.
55539      * @return {ColumnModel}
55540      */
55541     getColumnModel : function(){
55542         return this.colModel;
55543     },
55544
55545     /**
55546      * Returns the grid's GridView object.
55547      * @return {GridView}
55548      */
55549     getView : function(){
55550         if(!this.view){
55551             this.view = new Roo.grid.GridView(this.viewConfig);
55552         }
55553         return this.view;
55554     },
55555     /**
55556      * Called to get grid's drag proxy text, by default returns this.ddText.
55557      * @return {String}
55558      */
55559     getDragDropText : function(){
55560         var count = this.selModel.getCount();
55561         return String.format(this.ddText, count, count == 1 ? '' : 's');
55562     }
55563 });
55564 /**
55565  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55566  * %0 is replaced with the number of selected rows.
55567  * @type String
55568  */
55569 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55570  * Based on:
55571  * Ext JS Library 1.1.1
55572  * Copyright(c) 2006-2007, Ext JS, LLC.
55573  *
55574  * Originally Released Under LGPL - original licence link has changed is not relivant.
55575  *
55576  * Fork - LGPL
55577  * <script type="text/javascript">
55578  */
55579  
55580 Roo.grid.AbstractGridView = function(){
55581         this.grid = null;
55582         
55583         this.events = {
55584             "beforerowremoved" : true,
55585             "beforerowsinserted" : true,
55586             "beforerefresh" : true,
55587             "rowremoved" : true,
55588             "rowsinserted" : true,
55589             "rowupdated" : true,
55590             "refresh" : true
55591         };
55592     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55593 };
55594
55595 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55596     rowClass : "x-grid-row",
55597     cellClass : "x-grid-cell",
55598     tdClass : "x-grid-td",
55599     hdClass : "x-grid-hd",
55600     splitClass : "x-grid-hd-split",
55601     
55602     init: function(grid){
55603         this.grid = grid;
55604                 var cid = this.grid.getGridEl().id;
55605         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55606         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55607         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55608         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55609         },
55610         
55611     getColumnRenderers : function(){
55612         var renderers = [];
55613         var cm = this.grid.colModel;
55614         var colCount = cm.getColumnCount();
55615         for(var i = 0; i < colCount; i++){
55616             renderers[i] = cm.getRenderer(i);
55617         }
55618         return renderers;
55619     },
55620     
55621     getColumnIds : function(){
55622         var ids = [];
55623         var cm = this.grid.colModel;
55624         var colCount = cm.getColumnCount();
55625         for(var i = 0; i < colCount; i++){
55626             ids[i] = cm.getColumnId(i);
55627         }
55628         return ids;
55629     },
55630     
55631     getDataIndexes : function(){
55632         if(!this.indexMap){
55633             this.indexMap = this.buildIndexMap();
55634         }
55635         return this.indexMap.colToData;
55636     },
55637     
55638     getColumnIndexByDataIndex : function(dataIndex){
55639         if(!this.indexMap){
55640             this.indexMap = this.buildIndexMap();
55641         }
55642         return this.indexMap.dataToCol[dataIndex];
55643     },
55644     
55645     /**
55646      * Set a css style for a column dynamically. 
55647      * @param {Number} colIndex The index of the column
55648      * @param {String} name The css property name
55649      * @param {String} value The css value
55650      */
55651     setCSSStyle : function(colIndex, name, value){
55652         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55653         Roo.util.CSS.updateRule(selector, name, value);
55654     },
55655     
55656     generateRules : function(cm){
55657         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55658         Roo.util.CSS.removeStyleSheet(rulesId);
55659         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55660             var cid = cm.getColumnId(i);
55661             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55662                          this.tdSelector, cid, " {\n}\n",
55663                          this.hdSelector, cid, " {\n}\n",
55664                          this.splitSelector, cid, " {\n}\n");
55665         }
55666         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55667     }
55668 });/*
55669  * Based on:
55670  * Ext JS Library 1.1.1
55671  * Copyright(c) 2006-2007, Ext JS, LLC.
55672  *
55673  * Originally Released Under LGPL - original licence link has changed is not relivant.
55674  *
55675  * Fork - LGPL
55676  * <script type="text/javascript">
55677  */
55678
55679 // private
55680 // This is a support class used internally by the Grid components
55681 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55682     this.grid = grid;
55683     this.view = grid.getView();
55684     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55685     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55686     if(hd2){
55687         this.setHandleElId(Roo.id(hd));
55688         this.setOuterHandleElId(Roo.id(hd2));
55689     }
55690     this.scroll = false;
55691 };
55692 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55693     maxDragWidth: 120,
55694     getDragData : function(e){
55695         var t = Roo.lib.Event.getTarget(e);
55696         var h = this.view.findHeaderCell(t);
55697         if(h){
55698             return {ddel: h.firstChild, header:h};
55699         }
55700         return false;
55701     },
55702
55703     onInitDrag : function(e){
55704         this.view.headersDisabled = true;
55705         var clone = this.dragData.ddel.cloneNode(true);
55706         clone.id = Roo.id();
55707         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55708         this.proxy.update(clone);
55709         return true;
55710     },
55711
55712     afterValidDrop : function(){
55713         var v = this.view;
55714         setTimeout(function(){
55715             v.headersDisabled = false;
55716         }, 50);
55717     },
55718
55719     afterInvalidDrop : function(){
55720         var v = this.view;
55721         setTimeout(function(){
55722             v.headersDisabled = false;
55723         }, 50);
55724     }
55725 });
55726 /*
55727  * Based on:
55728  * Ext JS Library 1.1.1
55729  * Copyright(c) 2006-2007, Ext JS, LLC.
55730  *
55731  * Originally Released Under LGPL - original licence link has changed is not relivant.
55732  *
55733  * Fork - LGPL
55734  * <script type="text/javascript">
55735  */
55736 // private
55737 // This is a support class used internally by the Grid components
55738 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55739     this.grid = grid;
55740     this.view = grid.getView();
55741     // split the proxies so they don't interfere with mouse events
55742     this.proxyTop = Roo.DomHelper.append(document.body, {
55743         cls:"col-move-top", html:"&#160;"
55744     }, true);
55745     this.proxyBottom = Roo.DomHelper.append(document.body, {
55746         cls:"col-move-bottom", html:"&#160;"
55747     }, true);
55748     this.proxyTop.hide = this.proxyBottom.hide = function(){
55749         this.setLeftTop(-100,-100);
55750         this.setStyle("visibility", "hidden");
55751     };
55752     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55753     // temporarily disabled
55754     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55755     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55756 };
55757 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55758     proxyOffsets : [-4, -9],
55759     fly: Roo.Element.fly,
55760
55761     getTargetFromEvent : function(e){
55762         var t = Roo.lib.Event.getTarget(e);
55763         var cindex = this.view.findCellIndex(t);
55764         if(cindex !== false){
55765             return this.view.getHeaderCell(cindex);
55766         }
55767         return null;
55768     },
55769
55770     nextVisible : function(h){
55771         var v = this.view, cm = this.grid.colModel;
55772         h = h.nextSibling;
55773         while(h){
55774             if(!cm.isHidden(v.getCellIndex(h))){
55775                 return h;
55776             }
55777             h = h.nextSibling;
55778         }
55779         return null;
55780     },
55781
55782     prevVisible : function(h){
55783         var v = this.view, cm = this.grid.colModel;
55784         h = h.prevSibling;
55785         while(h){
55786             if(!cm.isHidden(v.getCellIndex(h))){
55787                 return h;
55788             }
55789             h = h.prevSibling;
55790         }
55791         return null;
55792     },
55793
55794     positionIndicator : function(h, n, e){
55795         var x = Roo.lib.Event.getPageX(e);
55796         var r = Roo.lib.Dom.getRegion(n.firstChild);
55797         var px, pt, py = r.top + this.proxyOffsets[1];
55798         if((r.right - x) <= (r.right-r.left)/2){
55799             px = r.right+this.view.borderWidth;
55800             pt = "after";
55801         }else{
55802             px = r.left;
55803             pt = "before";
55804         }
55805         var oldIndex = this.view.getCellIndex(h);
55806         var newIndex = this.view.getCellIndex(n);
55807
55808         if(this.grid.colModel.isFixed(newIndex)){
55809             return false;
55810         }
55811
55812         var locked = this.grid.colModel.isLocked(newIndex);
55813
55814         if(pt == "after"){
55815             newIndex++;
55816         }
55817         if(oldIndex < newIndex){
55818             newIndex--;
55819         }
55820         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55821             return false;
55822         }
55823         px +=  this.proxyOffsets[0];
55824         this.proxyTop.setLeftTop(px, py);
55825         this.proxyTop.show();
55826         if(!this.bottomOffset){
55827             this.bottomOffset = this.view.mainHd.getHeight();
55828         }
55829         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55830         this.proxyBottom.show();
55831         return pt;
55832     },
55833
55834     onNodeEnter : function(n, dd, e, data){
55835         if(data.header != n){
55836             this.positionIndicator(data.header, n, e);
55837         }
55838     },
55839
55840     onNodeOver : function(n, dd, e, data){
55841         var result = false;
55842         if(data.header != n){
55843             result = this.positionIndicator(data.header, n, e);
55844         }
55845         if(!result){
55846             this.proxyTop.hide();
55847             this.proxyBottom.hide();
55848         }
55849         return result ? this.dropAllowed : this.dropNotAllowed;
55850     },
55851
55852     onNodeOut : function(n, dd, e, data){
55853         this.proxyTop.hide();
55854         this.proxyBottom.hide();
55855     },
55856
55857     onNodeDrop : function(n, dd, e, data){
55858         var h = data.header;
55859         if(h != n){
55860             var cm = this.grid.colModel;
55861             var x = Roo.lib.Event.getPageX(e);
55862             var r = Roo.lib.Dom.getRegion(n.firstChild);
55863             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55864             var oldIndex = this.view.getCellIndex(h);
55865             var newIndex = this.view.getCellIndex(n);
55866             var locked = cm.isLocked(newIndex);
55867             if(pt == "after"){
55868                 newIndex++;
55869             }
55870             if(oldIndex < newIndex){
55871                 newIndex--;
55872             }
55873             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55874                 return false;
55875             }
55876             cm.setLocked(oldIndex, locked, true);
55877             cm.moveColumn(oldIndex, newIndex);
55878             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55879             return true;
55880         }
55881         return false;
55882     }
55883 });
55884 /*
55885  * Based on:
55886  * Ext JS Library 1.1.1
55887  * Copyright(c) 2006-2007, Ext JS, LLC.
55888  *
55889  * Originally Released Under LGPL - original licence link has changed is not relivant.
55890  *
55891  * Fork - LGPL
55892  * <script type="text/javascript">
55893  */
55894   
55895 /**
55896  * @class Roo.grid.GridView
55897  * @extends Roo.util.Observable
55898  *
55899  * @constructor
55900  * @param {Object} config
55901  */
55902 Roo.grid.GridView = function(config){
55903     Roo.grid.GridView.superclass.constructor.call(this);
55904     this.el = null;
55905
55906     Roo.apply(this, config);
55907 };
55908
55909 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55910
55911     unselectable :  'unselectable="on"',
55912     unselectableCls :  'x-unselectable',
55913     
55914     
55915     rowClass : "x-grid-row",
55916
55917     cellClass : "x-grid-col",
55918
55919     tdClass : "x-grid-td",
55920
55921     hdClass : "x-grid-hd",
55922
55923     splitClass : "x-grid-split",
55924
55925     sortClasses : ["sort-asc", "sort-desc"],
55926
55927     enableMoveAnim : false,
55928
55929     hlColor: "C3DAF9",
55930
55931     dh : Roo.DomHelper,
55932
55933     fly : Roo.Element.fly,
55934
55935     css : Roo.util.CSS,
55936
55937     borderWidth: 1,
55938
55939     splitOffset: 3,
55940
55941     scrollIncrement : 22,
55942
55943     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
55944
55945     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
55946
55947     bind : function(ds, cm){
55948         if(this.ds){
55949             this.ds.un("load", this.onLoad, this);
55950             this.ds.un("datachanged", this.onDataChange, this);
55951             this.ds.un("add", this.onAdd, this);
55952             this.ds.un("remove", this.onRemove, this);
55953             this.ds.un("update", this.onUpdate, this);
55954             this.ds.un("clear", this.onClear, this);
55955         }
55956         if(ds){
55957             ds.on("load", this.onLoad, this);
55958             ds.on("datachanged", this.onDataChange, this);
55959             ds.on("add", this.onAdd, this);
55960             ds.on("remove", this.onRemove, this);
55961             ds.on("update", this.onUpdate, this);
55962             ds.on("clear", this.onClear, this);
55963         }
55964         this.ds = ds;
55965
55966         if(this.cm){
55967             this.cm.un("widthchange", this.onColWidthChange, this);
55968             this.cm.un("headerchange", this.onHeaderChange, this);
55969             this.cm.un("hiddenchange", this.onHiddenChange, this);
55970             this.cm.un("columnmoved", this.onColumnMove, this);
55971             this.cm.un("columnlockchange", this.onColumnLock, this);
55972         }
55973         if(cm){
55974             this.generateRules(cm);
55975             cm.on("widthchange", this.onColWidthChange, this);
55976             cm.on("headerchange", this.onHeaderChange, this);
55977             cm.on("hiddenchange", this.onHiddenChange, this);
55978             cm.on("columnmoved", this.onColumnMove, this);
55979             cm.on("columnlockchange", this.onColumnLock, this);
55980         }
55981         this.cm = cm;
55982     },
55983
55984     init: function(grid){
55985         Roo.grid.GridView.superclass.init.call(this, grid);
55986
55987         this.bind(grid.dataSource, grid.colModel);
55988
55989         grid.on("headerclick", this.handleHeaderClick, this);
55990
55991         if(grid.trackMouseOver){
55992             grid.on("mouseover", this.onRowOver, this);
55993             grid.on("mouseout", this.onRowOut, this);
55994         }
55995         grid.cancelTextSelection = function(){};
55996         this.gridId = grid.id;
55997
55998         var tpls = this.templates || {};
55999
56000         if(!tpls.master){
56001             tpls.master = new Roo.Template(
56002                '<div class="x-grid" hidefocus="true">',
56003                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56004                   '<div class="x-grid-topbar"></div>',
56005                   '<div class="x-grid-scroller"><div></div></div>',
56006                   '<div class="x-grid-locked">',
56007                       '<div class="x-grid-header">{lockedHeader}</div>',
56008                       '<div class="x-grid-body">{lockedBody}</div>',
56009                   "</div>",
56010                   '<div class="x-grid-viewport">',
56011                       '<div class="x-grid-header">{header}</div>',
56012                       '<div class="x-grid-body">{body}</div>',
56013                   "</div>",
56014                   '<div class="x-grid-bottombar"></div>',
56015                  
56016                   '<div class="x-grid-resize-proxy">&#160;</div>',
56017                "</div>"
56018             );
56019             tpls.master.disableformats = true;
56020         }
56021
56022         if(!tpls.header){
56023             tpls.header = new Roo.Template(
56024                '<table border="0" cellspacing="0" cellpadding="0">',
56025                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56026                "</table>{splits}"
56027             );
56028             tpls.header.disableformats = true;
56029         }
56030         tpls.header.compile();
56031
56032         if(!tpls.hcell){
56033             tpls.hcell = new Roo.Template(
56034                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56035                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56036                 "</div></td>"
56037              );
56038              tpls.hcell.disableFormats = true;
56039         }
56040         tpls.hcell.compile();
56041
56042         if(!tpls.hsplit){
56043             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56044                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56045             tpls.hsplit.disableFormats = true;
56046         }
56047         tpls.hsplit.compile();
56048
56049         if(!tpls.body){
56050             tpls.body = new Roo.Template(
56051                '<table border="0" cellspacing="0" cellpadding="0">',
56052                "<tbody>{rows}</tbody>",
56053                "</table>"
56054             );
56055             tpls.body.disableFormats = true;
56056         }
56057         tpls.body.compile();
56058
56059         if(!tpls.row){
56060             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56061             tpls.row.disableFormats = true;
56062         }
56063         tpls.row.compile();
56064
56065         if(!tpls.cell){
56066             tpls.cell = new Roo.Template(
56067                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56068                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56069                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56070                 "</td>"
56071             );
56072             tpls.cell.disableFormats = true;
56073         }
56074         tpls.cell.compile();
56075
56076         this.templates = tpls;
56077     },
56078
56079     // remap these for backwards compat
56080     onColWidthChange : function(){
56081         this.updateColumns.apply(this, arguments);
56082     },
56083     onHeaderChange : function(){
56084         this.updateHeaders.apply(this, arguments);
56085     }, 
56086     onHiddenChange : function(){
56087         this.handleHiddenChange.apply(this, arguments);
56088     },
56089     onColumnMove : function(){
56090         this.handleColumnMove.apply(this, arguments);
56091     },
56092     onColumnLock : function(){
56093         this.handleLockChange.apply(this, arguments);
56094     },
56095
56096     onDataChange : function(){
56097         this.refresh();
56098         this.updateHeaderSortState();
56099     },
56100
56101     onClear : function(){
56102         this.refresh();
56103     },
56104
56105     onUpdate : function(ds, record){
56106         this.refreshRow(record);
56107     },
56108
56109     refreshRow : function(record){
56110         var ds = this.ds, index;
56111         if(typeof record == 'number'){
56112             index = record;
56113             record = ds.getAt(index);
56114         }else{
56115             index = ds.indexOf(record);
56116         }
56117         this.insertRows(ds, index, index, true);
56118         this.onRemove(ds, record, index+1, true);
56119         this.syncRowHeights(index, index);
56120         this.layout();
56121         this.fireEvent("rowupdated", this, index, record);
56122     },
56123
56124     onAdd : function(ds, records, index){
56125         this.insertRows(ds, index, index + (records.length-1));
56126     },
56127
56128     onRemove : function(ds, record, index, isUpdate){
56129         if(isUpdate !== true){
56130             this.fireEvent("beforerowremoved", this, index, record);
56131         }
56132         var bt = this.getBodyTable(), lt = this.getLockedTable();
56133         if(bt.rows[index]){
56134             bt.firstChild.removeChild(bt.rows[index]);
56135         }
56136         if(lt.rows[index]){
56137             lt.firstChild.removeChild(lt.rows[index]);
56138         }
56139         if(isUpdate !== true){
56140             this.stripeRows(index);
56141             this.syncRowHeights(index, index);
56142             this.layout();
56143             this.fireEvent("rowremoved", this, index, record);
56144         }
56145     },
56146
56147     onLoad : function(){
56148         this.scrollToTop();
56149     },
56150
56151     /**
56152      * Scrolls the grid to the top
56153      */
56154     scrollToTop : function(){
56155         if(this.scroller){
56156             this.scroller.dom.scrollTop = 0;
56157             this.syncScroll();
56158         }
56159     },
56160
56161     /**
56162      * Gets a panel in the header of the grid that can be used for toolbars etc.
56163      * After modifying the contents of this panel a call to grid.autoSize() may be
56164      * required to register any changes in size.
56165      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56166      * @return Roo.Element
56167      */
56168     getHeaderPanel : function(doShow){
56169         if(doShow){
56170             this.headerPanel.show();
56171         }
56172         return this.headerPanel;
56173     },
56174
56175     /**
56176      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56177      * After modifying the contents of this panel a call to grid.autoSize() may be
56178      * required to register any changes in size.
56179      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56180      * @return Roo.Element
56181      */
56182     getFooterPanel : function(doShow){
56183         if(doShow){
56184             this.footerPanel.show();
56185         }
56186         return this.footerPanel;
56187     },
56188
56189     initElements : function(){
56190         var E = Roo.Element;
56191         var el = this.grid.getGridEl().dom.firstChild;
56192         var cs = el.childNodes;
56193
56194         this.el = new E(el);
56195         
56196          this.focusEl = new E(el.firstChild);
56197         this.focusEl.swallowEvent("click", true);
56198         
56199         this.headerPanel = new E(cs[1]);
56200         this.headerPanel.enableDisplayMode("block");
56201
56202         this.scroller = new E(cs[2]);
56203         this.scrollSizer = new E(this.scroller.dom.firstChild);
56204
56205         this.lockedWrap = new E(cs[3]);
56206         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56207         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56208
56209         this.mainWrap = new E(cs[4]);
56210         this.mainHd = new E(this.mainWrap.dom.firstChild);
56211         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56212
56213         this.footerPanel = new E(cs[5]);
56214         this.footerPanel.enableDisplayMode("block");
56215
56216         this.resizeProxy = new E(cs[6]);
56217
56218         this.headerSelector = String.format(
56219            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56220            this.lockedHd.id, this.mainHd.id
56221         );
56222
56223         this.splitterSelector = String.format(
56224            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56225            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56226         );
56227     },
56228     idToCssName : function(s)
56229     {
56230         return s.replace(/[^a-z0-9]+/ig, '-');
56231     },
56232
56233     getHeaderCell : function(index){
56234         return Roo.DomQuery.select(this.headerSelector)[index];
56235     },
56236
56237     getHeaderCellMeasure : function(index){
56238         return this.getHeaderCell(index).firstChild;
56239     },
56240
56241     getHeaderCellText : function(index){
56242         return this.getHeaderCell(index).firstChild.firstChild;
56243     },
56244
56245     getLockedTable : function(){
56246         return this.lockedBody.dom.firstChild;
56247     },
56248
56249     getBodyTable : function(){
56250         return this.mainBody.dom.firstChild;
56251     },
56252
56253     getLockedRow : function(index){
56254         return this.getLockedTable().rows[index];
56255     },
56256
56257     getRow : function(index){
56258         return this.getBodyTable().rows[index];
56259     },
56260
56261     getRowComposite : function(index){
56262         if(!this.rowEl){
56263             this.rowEl = new Roo.CompositeElementLite();
56264         }
56265         var els = [], lrow, mrow;
56266         if(lrow = this.getLockedRow(index)){
56267             els.push(lrow);
56268         }
56269         if(mrow = this.getRow(index)){
56270             els.push(mrow);
56271         }
56272         this.rowEl.elements = els;
56273         return this.rowEl;
56274     },
56275     /**
56276      * Gets the 'td' of the cell
56277      * 
56278      * @param {Integer} rowIndex row to select
56279      * @param {Integer} colIndex column to select
56280      * 
56281      * @return {Object} 
56282      */
56283     getCell : function(rowIndex, colIndex){
56284         var locked = this.cm.getLockedCount();
56285         var source;
56286         if(colIndex < locked){
56287             source = this.lockedBody.dom.firstChild;
56288         }else{
56289             source = this.mainBody.dom.firstChild;
56290             colIndex -= locked;
56291         }
56292         return source.rows[rowIndex].childNodes[colIndex];
56293     },
56294
56295     getCellText : function(rowIndex, colIndex){
56296         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56297     },
56298
56299     getCellBox : function(cell){
56300         var b = this.fly(cell).getBox();
56301         if(Roo.isOpera){ // opera fails to report the Y
56302             b.y = cell.offsetTop + this.mainBody.getY();
56303         }
56304         return b;
56305     },
56306
56307     getCellIndex : function(cell){
56308         var id = String(cell.className).match(this.cellRE);
56309         if(id){
56310             return parseInt(id[1], 10);
56311         }
56312         return 0;
56313     },
56314
56315     findHeaderIndex : function(n){
56316         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56317         return r ? this.getCellIndex(r) : false;
56318     },
56319
56320     findHeaderCell : function(n){
56321         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56322         return r ? r : false;
56323     },
56324
56325     findRowIndex : function(n){
56326         if(!n){
56327             return false;
56328         }
56329         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56330         return r ? r.rowIndex : false;
56331     },
56332
56333     findCellIndex : function(node){
56334         var stop = this.el.dom;
56335         while(node && node != stop){
56336             if(this.findRE.test(node.className)){
56337                 return this.getCellIndex(node);
56338             }
56339             node = node.parentNode;
56340         }
56341         return false;
56342     },
56343
56344     getColumnId : function(index){
56345         return this.cm.getColumnId(index);
56346     },
56347
56348     getSplitters : function()
56349     {
56350         if(this.splitterSelector){
56351            return Roo.DomQuery.select(this.splitterSelector);
56352         }else{
56353             return null;
56354       }
56355     },
56356
56357     getSplitter : function(index){
56358         return this.getSplitters()[index];
56359     },
56360
56361     onRowOver : function(e, t){
56362         var row;
56363         if((row = this.findRowIndex(t)) !== false){
56364             this.getRowComposite(row).addClass("x-grid-row-over");
56365         }
56366     },
56367
56368     onRowOut : function(e, t){
56369         var row;
56370         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56371             this.getRowComposite(row).removeClass("x-grid-row-over");
56372         }
56373     },
56374
56375     renderHeaders : function(){
56376         var cm = this.cm;
56377         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56378         var cb = [], lb = [], sb = [], lsb = [], p = {};
56379         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56380             p.cellId = "x-grid-hd-0-" + i;
56381             p.splitId = "x-grid-csplit-0-" + i;
56382             p.id = cm.getColumnId(i);
56383             p.value = cm.getColumnHeader(i) || "";
56384             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56385             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56386             if(!cm.isLocked(i)){
56387                 cb[cb.length] = ct.apply(p);
56388                 sb[sb.length] = st.apply(p);
56389             }else{
56390                 lb[lb.length] = ct.apply(p);
56391                 lsb[lsb.length] = st.apply(p);
56392             }
56393         }
56394         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56395                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56396     },
56397
56398     updateHeaders : function(){
56399         var html = this.renderHeaders();
56400         this.lockedHd.update(html[0]);
56401         this.mainHd.update(html[1]);
56402     },
56403
56404     /**
56405      * Focuses the specified row.
56406      * @param {Number} row The row index
56407      */
56408     focusRow : function(row)
56409     {
56410         //Roo.log('GridView.focusRow');
56411         var x = this.scroller.dom.scrollLeft;
56412         this.focusCell(row, 0, false);
56413         this.scroller.dom.scrollLeft = x;
56414     },
56415
56416     /**
56417      * Focuses the specified cell.
56418      * @param {Number} row The row index
56419      * @param {Number} col The column index
56420      * @param {Boolean} hscroll false to disable horizontal scrolling
56421      */
56422     focusCell : function(row, col, hscroll)
56423     {
56424         //Roo.log('GridView.focusCell');
56425         var el = this.ensureVisible(row, col, hscroll);
56426         this.focusEl.alignTo(el, "tl-tl");
56427         if(Roo.isGecko){
56428             this.focusEl.focus();
56429         }else{
56430             this.focusEl.focus.defer(1, this.focusEl);
56431         }
56432     },
56433
56434     /**
56435      * Scrolls the specified cell into view
56436      * @param {Number} row The row index
56437      * @param {Number} col The column index
56438      * @param {Boolean} hscroll false to disable horizontal scrolling
56439      */
56440     ensureVisible : function(row, col, hscroll)
56441     {
56442         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56443         //return null; //disable for testing.
56444         if(typeof row != "number"){
56445             row = row.rowIndex;
56446         }
56447         if(row < 0 && row >= this.ds.getCount()){
56448             return  null;
56449         }
56450         col = (col !== undefined ? col : 0);
56451         var cm = this.grid.colModel;
56452         while(cm.isHidden(col)){
56453             col++;
56454         }
56455
56456         var el = this.getCell(row, col);
56457         if(!el){
56458             return null;
56459         }
56460         var c = this.scroller.dom;
56461
56462         var ctop = parseInt(el.offsetTop, 10);
56463         var cleft = parseInt(el.offsetLeft, 10);
56464         var cbot = ctop + el.offsetHeight;
56465         var cright = cleft + el.offsetWidth;
56466         
56467         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56468         var stop = parseInt(c.scrollTop, 10);
56469         var sleft = parseInt(c.scrollLeft, 10);
56470         var sbot = stop + ch;
56471         var sright = sleft + c.clientWidth;
56472         /*
56473         Roo.log('GridView.ensureVisible:' +
56474                 ' ctop:' + ctop +
56475                 ' c.clientHeight:' + c.clientHeight +
56476                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56477                 ' stop:' + stop +
56478                 ' cbot:' + cbot +
56479                 ' sbot:' + sbot +
56480                 ' ch:' + ch  
56481                 );
56482         */
56483         if(ctop < stop){
56484              c.scrollTop = ctop;
56485             //Roo.log("set scrolltop to ctop DISABLE?");
56486         }else if(cbot > sbot){
56487             //Roo.log("set scrolltop to cbot-ch");
56488             c.scrollTop = cbot-ch;
56489         }
56490         
56491         if(hscroll !== false){
56492             if(cleft < sleft){
56493                 c.scrollLeft = cleft;
56494             }else if(cright > sright){
56495                 c.scrollLeft = cright-c.clientWidth;
56496             }
56497         }
56498          
56499         return el;
56500     },
56501
56502     updateColumns : function(){
56503         this.grid.stopEditing();
56504         var cm = this.grid.colModel, colIds = this.getColumnIds();
56505         //var totalWidth = cm.getTotalWidth();
56506         var pos = 0;
56507         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56508             //if(cm.isHidden(i)) continue;
56509             var w = cm.getColumnWidth(i);
56510             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56511             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56512         }
56513         this.updateSplitters();
56514     },
56515
56516     generateRules : function(cm){
56517         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56518         Roo.util.CSS.removeStyleSheet(rulesId);
56519         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56520             var cid = cm.getColumnId(i);
56521             var align = '';
56522             if(cm.config[i].align){
56523                 align = 'text-align:'+cm.config[i].align+';';
56524             }
56525             var hidden = '';
56526             if(cm.isHidden(i)){
56527                 hidden = 'display:none;';
56528             }
56529             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56530             ruleBuf.push(
56531                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56532                     this.hdSelector, cid, " {\n", align, width, "}\n",
56533                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56534                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56535         }
56536         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56537     },
56538
56539     updateSplitters : function(){
56540         var cm = this.cm, s = this.getSplitters();
56541         if(s){ // splitters not created yet
56542             var pos = 0, locked = true;
56543             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56544                 if(cm.isHidden(i)) {
56545                     continue;
56546                 }
56547                 var w = cm.getColumnWidth(i); // make sure it's a number
56548                 if(!cm.isLocked(i) && locked){
56549                     pos = 0;
56550                     locked = false;
56551                 }
56552                 pos += w;
56553                 s[i].style.left = (pos-this.splitOffset) + "px";
56554             }
56555         }
56556     },
56557
56558     handleHiddenChange : function(colModel, colIndex, hidden){
56559         if(hidden){
56560             this.hideColumn(colIndex);
56561         }else{
56562             this.unhideColumn(colIndex);
56563         }
56564     },
56565
56566     hideColumn : function(colIndex){
56567         var cid = this.getColumnId(colIndex);
56568         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56569         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56570         if(Roo.isSafari){
56571             this.updateHeaders();
56572         }
56573         this.updateSplitters();
56574         this.layout();
56575     },
56576
56577     unhideColumn : function(colIndex){
56578         var cid = this.getColumnId(colIndex);
56579         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56580         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56581
56582         if(Roo.isSafari){
56583             this.updateHeaders();
56584         }
56585         this.updateSplitters();
56586         this.layout();
56587     },
56588
56589     insertRows : function(dm, firstRow, lastRow, isUpdate){
56590         if(firstRow == 0 && lastRow == dm.getCount()-1){
56591             this.refresh();
56592         }else{
56593             if(!isUpdate){
56594                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56595             }
56596             var s = this.getScrollState();
56597             var markup = this.renderRows(firstRow, lastRow);
56598             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56599             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56600             this.restoreScroll(s);
56601             if(!isUpdate){
56602                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56603                 this.syncRowHeights(firstRow, lastRow);
56604                 this.stripeRows(firstRow);
56605                 this.layout();
56606             }
56607         }
56608     },
56609
56610     bufferRows : function(markup, target, index){
56611         var before = null, trows = target.rows, tbody = target.tBodies[0];
56612         if(index < trows.length){
56613             before = trows[index];
56614         }
56615         var b = document.createElement("div");
56616         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56617         var rows = b.firstChild.rows;
56618         for(var i = 0, len = rows.length; i < len; i++){
56619             if(before){
56620                 tbody.insertBefore(rows[0], before);
56621             }else{
56622                 tbody.appendChild(rows[0]);
56623             }
56624         }
56625         b.innerHTML = "";
56626         b = null;
56627     },
56628
56629     deleteRows : function(dm, firstRow, lastRow){
56630         if(dm.getRowCount()<1){
56631             this.fireEvent("beforerefresh", this);
56632             this.mainBody.update("");
56633             this.lockedBody.update("");
56634             this.fireEvent("refresh", this);
56635         }else{
56636             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56637             var bt = this.getBodyTable();
56638             var tbody = bt.firstChild;
56639             var rows = bt.rows;
56640             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56641                 tbody.removeChild(rows[firstRow]);
56642             }
56643             this.stripeRows(firstRow);
56644             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56645         }
56646     },
56647
56648     updateRows : function(dataSource, firstRow, lastRow){
56649         var s = this.getScrollState();
56650         this.refresh();
56651         this.restoreScroll(s);
56652     },
56653
56654     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56655         if(!noRefresh){
56656            this.refresh();
56657         }
56658         this.updateHeaderSortState();
56659     },
56660
56661     getScrollState : function(){
56662         
56663         var sb = this.scroller.dom;
56664         return {left: sb.scrollLeft, top: sb.scrollTop};
56665     },
56666
56667     stripeRows : function(startRow){
56668         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56669             return;
56670         }
56671         startRow = startRow || 0;
56672         var rows = this.getBodyTable().rows;
56673         var lrows = this.getLockedTable().rows;
56674         var cls = ' x-grid-row-alt ';
56675         for(var i = startRow, len = rows.length; i < len; i++){
56676             var row = rows[i], lrow = lrows[i];
56677             var isAlt = ((i+1) % 2 == 0);
56678             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56679             if(isAlt == hasAlt){
56680                 continue;
56681             }
56682             if(isAlt){
56683                 row.className += " x-grid-row-alt";
56684             }else{
56685                 row.className = row.className.replace("x-grid-row-alt", "");
56686             }
56687             if(lrow){
56688                 lrow.className = row.className;
56689             }
56690         }
56691     },
56692
56693     restoreScroll : function(state){
56694         //Roo.log('GridView.restoreScroll');
56695         var sb = this.scroller.dom;
56696         sb.scrollLeft = state.left;
56697         sb.scrollTop = state.top;
56698         this.syncScroll();
56699     },
56700
56701     syncScroll : function(){
56702         //Roo.log('GridView.syncScroll');
56703         var sb = this.scroller.dom;
56704         var sh = this.mainHd.dom;
56705         var bs = this.mainBody.dom;
56706         var lv = this.lockedBody.dom;
56707         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56708         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56709     },
56710
56711     handleScroll : function(e){
56712         this.syncScroll();
56713         var sb = this.scroller.dom;
56714         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56715         e.stopEvent();
56716     },
56717
56718     handleWheel : function(e){
56719         var d = e.getWheelDelta();
56720         this.scroller.dom.scrollTop -= d*22;
56721         // set this here to prevent jumpy scrolling on large tables
56722         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56723         e.stopEvent();
56724     },
56725
56726     renderRows : function(startRow, endRow){
56727         // pull in all the crap needed to render rows
56728         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56729         var colCount = cm.getColumnCount();
56730
56731         if(ds.getCount() < 1){
56732             return ["", ""];
56733         }
56734
56735         // build a map for all the columns
56736         var cs = [];
56737         for(var i = 0; i < colCount; i++){
56738             var name = cm.getDataIndex(i);
56739             cs[i] = {
56740                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56741                 renderer : cm.getRenderer(i),
56742                 id : cm.getColumnId(i),
56743                 locked : cm.isLocked(i),
56744                 has_editor : cm.isCellEditable(i)
56745             };
56746         }
56747
56748         startRow = startRow || 0;
56749         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56750
56751         // records to render
56752         var rs = ds.getRange(startRow, endRow);
56753
56754         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56755     },
56756
56757     // As much as I hate to duplicate code, this was branched because FireFox really hates
56758     // [].join("") on strings. The performance difference was substantial enough to
56759     // branch this function
56760     doRender : Roo.isGecko ?
56761             function(cs, rs, ds, startRow, colCount, stripe){
56762                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56763                 // buffers
56764                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56765                 
56766                 var hasListener = this.grid.hasListener('rowclass');
56767                 var rowcfg = {};
56768                 for(var j = 0, len = rs.length; j < len; j++){
56769                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56770                     for(var i = 0; i < colCount; i++){
56771                         c = cs[i];
56772                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56773                         p.id = c.id;
56774                         p.css = p.attr = "";
56775                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56776                         if(p.value == undefined || p.value === "") {
56777                             p.value = "&#160;";
56778                         }
56779                         if(c.has_editor){
56780                             p.css += ' x-grid-editable-cell';
56781                         }
56782                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56783                             p.css +=  ' x-grid-dirty-cell';
56784                         }
56785                         var markup = ct.apply(p);
56786                         if(!c.locked){
56787                             cb+= markup;
56788                         }else{
56789                             lcb+= markup;
56790                         }
56791                     }
56792                     var alt = [];
56793                     if(stripe && ((rowIndex+1) % 2 == 0)){
56794                         alt.push("x-grid-row-alt")
56795                     }
56796                     if(r.dirty){
56797                         alt.push(  " x-grid-dirty-row");
56798                     }
56799                     rp.cells = lcb;
56800                     if(this.getRowClass){
56801                         alt.push(this.getRowClass(r, rowIndex));
56802                     }
56803                     if (hasListener) {
56804                         rowcfg = {
56805                              
56806                             record: r,
56807                             rowIndex : rowIndex,
56808                             rowClass : ''
56809                         };
56810                         this.grid.fireEvent('rowclass', this, rowcfg);
56811                         alt.push(rowcfg.rowClass);
56812                     }
56813                     rp.alt = alt.join(" ");
56814                     lbuf+= rt.apply(rp);
56815                     rp.cells = cb;
56816                     buf+=  rt.apply(rp);
56817                 }
56818                 return [lbuf, buf];
56819             } :
56820             function(cs, rs, ds, startRow, colCount, stripe){
56821                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56822                 // buffers
56823                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56824                 var hasListener = this.grid.hasListener('rowclass');
56825  
56826                 var rowcfg = {};
56827                 for(var j = 0, len = rs.length; j < len; j++){
56828                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56829                     for(var i = 0; i < colCount; i++){
56830                         c = cs[i];
56831                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56832                         p.id = c.id;
56833                         p.css = p.attr = "";
56834                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56835                         if(p.value == undefined || p.value === "") {
56836                             p.value = "&#160;";
56837                         }
56838                         //Roo.log(c);
56839                          if(c.has_editor){
56840                             p.css += ' x-grid-editable-cell';
56841                         }
56842                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56843                             p.css += ' x-grid-dirty-cell' 
56844                         }
56845                         
56846                         var markup = ct.apply(p);
56847                         if(!c.locked){
56848                             cb[cb.length] = markup;
56849                         }else{
56850                             lcb[lcb.length] = markup;
56851                         }
56852                     }
56853                     var alt = [];
56854                     if(stripe && ((rowIndex+1) % 2 == 0)){
56855                         alt.push( "x-grid-row-alt");
56856                     }
56857                     if(r.dirty){
56858                         alt.push(" x-grid-dirty-row");
56859                     }
56860                     rp.cells = lcb;
56861                     if(this.getRowClass){
56862                         alt.push( this.getRowClass(r, rowIndex));
56863                     }
56864                     if (hasListener) {
56865                         rowcfg = {
56866                              
56867                             record: r,
56868                             rowIndex : rowIndex,
56869                             rowClass : ''
56870                         };
56871                         this.grid.fireEvent('rowclass', this, rowcfg);
56872                         alt.push(rowcfg.rowClass);
56873                     }
56874                     
56875                     rp.alt = alt.join(" ");
56876                     rp.cells = lcb.join("");
56877                     lbuf[lbuf.length] = rt.apply(rp);
56878                     rp.cells = cb.join("");
56879                     buf[buf.length] =  rt.apply(rp);
56880                 }
56881                 return [lbuf.join(""), buf.join("")];
56882             },
56883
56884     renderBody : function(){
56885         var markup = this.renderRows();
56886         var bt = this.templates.body;
56887         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56888     },
56889
56890     /**
56891      * Refreshes the grid
56892      * @param {Boolean} headersToo
56893      */
56894     refresh : function(headersToo){
56895         this.fireEvent("beforerefresh", this);
56896         this.grid.stopEditing();
56897         var result = this.renderBody();
56898         this.lockedBody.update(result[0]);
56899         this.mainBody.update(result[1]);
56900         if(headersToo === true){
56901             this.updateHeaders();
56902             this.updateColumns();
56903             this.updateSplitters();
56904             this.updateHeaderSortState();
56905         }
56906         this.syncRowHeights();
56907         this.layout();
56908         this.fireEvent("refresh", this);
56909     },
56910
56911     handleColumnMove : function(cm, oldIndex, newIndex){
56912         this.indexMap = null;
56913         var s = this.getScrollState();
56914         this.refresh(true);
56915         this.restoreScroll(s);
56916         this.afterMove(newIndex);
56917     },
56918
56919     afterMove : function(colIndex){
56920         if(this.enableMoveAnim && Roo.enableFx){
56921             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56922         }
56923         // if multisort - fix sortOrder, and reload..
56924         if (this.grid.dataSource.multiSort) {
56925             // the we can call sort again..
56926             var dm = this.grid.dataSource;
56927             var cm = this.grid.colModel;
56928             var so = [];
56929             for(var i = 0; i < cm.config.length; i++ ) {
56930                 
56931                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56932                     continue; // dont' bother, it's not in sort list or being set.
56933                 }
56934                 
56935                 so.push(cm.config[i].dataIndex);
56936             };
56937             dm.sortOrder = so;
56938             dm.load(dm.lastOptions);
56939             
56940             
56941         }
56942         
56943     },
56944
56945     updateCell : function(dm, rowIndex, dataIndex){
56946         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
56947         if(typeof colIndex == "undefined"){ // not present in grid
56948             return;
56949         }
56950         var cm = this.grid.colModel;
56951         var cell = this.getCell(rowIndex, colIndex);
56952         var cellText = this.getCellText(rowIndex, colIndex);
56953
56954         var p = {
56955             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
56956             id : cm.getColumnId(colIndex),
56957             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
56958         };
56959         var renderer = cm.getRenderer(colIndex);
56960         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
56961         if(typeof val == "undefined" || val === "") {
56962             val = "&#160;";
56963         }
56964         cellText.innerHTML = val;
56965         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
56966         this.syncRowHeights(rowIndex, rowIndex);
56967     },
56968
56969     calcColumnWidth : function(colIndex, maxRowsToMeasure){
56970         var maxWidth = 0;
56971         if(this.grid.autoSizeHeaders){
56972             var h = this.getHeaderCellMeasure(colIndex);
56973             maxWidth = Math.max(maxWidth, h.scrollWidth);
56974         }
56975         var tb, index;
56976         if(this.cm.isLocked(colIndex)){
56977             tb = this.getLockedTable();
56978             index = colIndex;
56979         }else{
56980             tb = this.getBodyTable();
56981             index = colIndex - this.cm.getLockedCount();
56982         }
56983         if(tb && tb.rows){
56984             var rows = tb.rows;
56985             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
56986             for(var i = 0; i < stopIndex; i++){
56987                 var cell = rows[i].childNodes[index].firstChild;
56988                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
56989             }
56990         }
56991         return maxWidth + /*margin for error in IE*/ 5;
56992     },
56993     /**
56994      * Autofit a column to its content.
56995      * @param {Number} colIndex
56996      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
56997      */
56998      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
56999          if(this.cm.isHidden(colIndex)){
57000              return; // can't calc a hidden column
57001          }
57002         if(forceMinSize){
57003             var cid = this.cm.getColumnId(colIndex);
57004             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57005            if(this.grid.autoSizeHeaders){
57006                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57007            }
57008         }
57009         var newWidth = this.calcColumnWidth(colIndex);
57010         this.cm.setColumnWidth(colIndex,
57011             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57012         if(!suppressEvent){
57013             this.grid.fireEvent("columnresize", colIndex, newWidth);
57014         }
57015     },
57016
57017     /**
57018      * Autofits all columns to their content and then expands to fit any extra space in the grid
57019      */
57020      autoSizeColumns : function(){
57021         var cm = this.grid.colModel;
57022         var colCount = cm.getColumnCount();
57023         for(var i = 0; i < colCount; i++){
57024             this.autoSizeColumn(i, true, true);
57025         }
57026         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57027             this.fitColumns();
57028         }else{
57029             this.updateColumns();
57030             this.layout();
57031         }
57032     },
57033
57034     /**
57035      * Autofits all columns to the grid's width proportionate with their current size
57036      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57037      */
57038     fitColumns : function(reserveScrollSpace){
57039         var cm = this.grid.colModel;
57040         var colCount = cm.getColumnCount();
57041         var cols = [];
57042         var width = 0;
57043         var i, w;
57044         for (i = 0; i < colCount; i++){
57045             if(!cm.isHidden(i) && !cm.isFixed(i)){
57046                 w = cm.getColumnWidth(i);
57047                 cols.push(i);
57048                 cols.push(w);
57049                 width += w;
57050             }
57051         }
57052         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57053         if(reserveScrollSpace){
57054             avail -= 17;
57055         }
57056         var frac = (avail - cm.getTotalWidth())/width;
57057         while (cols.length){
57058             w = cols.pop();
57059             i = cols.pop();
57060             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57061         }
57062         this.updateColumns();
57063         this.layout();
57064     },
57065
57066     onRowSelect : function(rowIndex){
57067         var row = this.getRowComposite(rowIndex);
57068         row.addClass("x-grid-row-selected");
57069     },
57070
57071     onRowDeselect : function(rowIndex){
57072         var row = this.getRowComposite(rowIndex);
57073         row.removeClass("x-grid-row-selected");
57074     },
57075
57076     onCellSelect : function(row, col){
57077         var cell = this.getCell(row, col);
57078         if(cell){
57079             Roo.fly(cell).addClass("x-grid-cell-selected");
57080         }
57081     },
57082
57083     onCellDeselect : function(row, col){
57084         var cell = this.getCell(row, col);
57085         if(cell){
57086             Roo.fly(cell).removeClass("x-grid-cell-selected");
57087         }
57088     },
57089
57090     updateHeaderSortState : function(){
57091         
57092         // sort state can be single { field: xxx, direction : yyy}
57093         // or   { xxx=>ASC , yyy : DESC ..... }
57094         
57095         var mstate = {};
57096         if (!this.ds.multiSort) { 
57097             var state = this.ds.getSortState();
57098             if(!state){
57099                 return;
57100             }
57101             mstate[state.field] = state.direction;
57102             // FIXME... - this is not used here.. but might be elsewhere..
57103             this.sortState = state;
57104             
57105         } else {
57106             mstate = this.ds.sortToggle;
57107         }
57108         //remove existing sort classes..
57109         
57110         var sc = this.sortClasses;
57111         var hds = this.el.select(this.headerSelector).removeClass(sc);
57112         
57113         for(var f in mstate) {
57114         
57115             var sortColumn = this.cm.findColumnIndex(f);
57116             
57117             if(sortColumn != -1){
57118                 var sortDir = mstate[f];        
57119                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57120             }
57121         }
57122         
57123          
57124         
57125     },
57126
57127
57128     handleHeaderClick : function(g, index,e){
57129         
57130         Roo.log("header click");
57131         
57132         if (Roo.isTouch) {
57133             // touch events on header are handled by context
57134             this.handleHdCtx(g,index,e);
57135             return;
57136         }
57137         
57138         
57139         if(this.headersDisabled){
57140             return;
57141         }
57142         var dm = g.dataSource, cm = g.colModel;
57143         if(!cm.isSortable(index)){
57144             return;
57145         }
57146         g.stopEditing();
57147         
57148         if (dm.multiSort) {
57149             // update the sortOrder
57150             var so = [];
57151             for(var i = 0; i < cm.config.length; i++ ) {
57152                 
57153                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57154                     continue; // dont' bother, it's not in sort list or being set.
57155                 }
57156                 
57157                 so.push(cm.config[i].dataIndex);
57158             };
57159             dm.sortOrder = so;
57160         }
57161         
57162         
57163         dm.sort(cm.getDataIndex(index));
57164     },
57165
57166
57167     destroy : function(){
57168         if(this.colMenu){
57169             this.colMenu.removeAll();
57170             Roo.menu.MenuMgr.unregister(this.colMenu);
57171             this.colMenu.getEl().remove();
57172             delete this.colMenu;
57173         }
57174         if(this.hmenu){
57175             this.hmenu.removeAll();
57176             Roo.menu.MenuMgr.unregister(this.hmenu);
57177             this.hmenu.getEl().remove();
57178             delete this.hmenu;
57179         }
57180         if(this.grid.enableColumnMove){
57181             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57182             if(dds){
57183                 for(var dd in dds){
57184                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57185                         var elid = dds[dd].dragElId;
57186                         dds[dd].unreg();
57187                         Roo.get(elid).remove();
57188                     } else if(dds[dd].config.isTarget){
57189                         dds[dd].proxyTop.remove();
57190                         dds[dd].proxyBottom.remove();
57191                         dds[dd].unreg();
57192                     }
57193                     if(Roo.dd.DDM.locationCache[dd]){
57194                         delete Roo.dd.DDM.locationCache[dd];
57195                     }
57196                 }
57197                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57198             }
57199         }
57200         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57201         this.bind(null, null);
57202         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57203     },
57204
57205     handleLockChange : function(){
57206         this.refresh(true);
57207     },
57208
57209     onDenyColumnLock : function(){
57210
57211     },
57212
57213     onDenyColumnHide : function(){
57214
57215     },
57216
57217     handleHdMenuClick : function(item){
57218         var index = this.hdCtxIndex;
57219         var cm = this.cm, ds = this.ds;
57220         switch(item.id){
57221             case "asc":
57222                 ds.sort(cm.getDataIndex(index), "ASC");
57223                 break;
57224             case "desc":
57225                 ds.sort(cm.getDataIndex(index), "DESC");
57226                 break;
57227             case "lock":
57228                 var lc = cm.getLockedCount();
57229                 if(cm.getColumnCount(true) <= lc+1){
57230                     this.onDenyColumnLock();
57231                     return;
57232                 }
57233                 if(lc != index){
57234                     cm.setLocked(index, true, true);
57235                     cm.moveColumn(index, lc);
57236                     this.grid.fireEvent("columnmove", index, lc);
57237                 }else{
57238                     cm.setLocked(index, true);
57239                 }
57240             break;
57241             case "unlock":
57242                 var lc = cm.getLockedCount();
57243                 if((lc-1) != index){
57244                     cm.setLocked(index, false, true);
57245                     cm.moveColumn(index, lc-1);
57246                     this.grid.fireEvent("columnmove", index, lc-1);
57247                 }else{
57248                     cm.setLocked(index, false);
57249                 }
57250             break;
57251             case 'wider': // used to expand cols on touch..
57252             case 'narrow':
57253                 var cw = cm.getColumnWidth(index);
57254                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57255                 cw = Math.max(0, cw);
57256                 cw = Math.min(cw,4000);
57257                 cm.setColumnWidth(index, cw);
57258                 break;
57259                 
57260             default:
57261                 index = cm.getIndexById(item.id.substr(4));
57262                 if(index != -1){
57263                     if(item.checked && cm.getColumnCount(true) <= 1){
57264                         this.onDenyColumnHide();
57265                         return false;
57266                     }
57267                     cm.setHidden(index, item.checked);
57268                 }
57269         }
57270         return true;
57271     },
57272
57273     beforeColMenuShow : function(){
57274         var cm = this.cm,  colCount = cm.getColumnCount();
57275         this.colMenu.removeAll();
57276         for(var i = 0; i < colCount; i++){
57277             this.colMenu.add(new Roo.menu.CheckItem({
57278                 id: "col-"+cm.getColumnId(i),
57279                 text: cm.getColumnHeader(i),
57280                 checked: !cm.isHidden(i),
57281                 hideOnClick:false
57282             }));
57283         }
57284     },
57285
57286     handleHdCtx : function(g, index, e){
57287         e.stopEvent();
57288         var hd = this.getHeaderCell(index);
57289         this.hdCtxIndex = index;
57290         var ms = this.hmenu.items, cm = this.cm;
57291         ms.get("asc").setDisabled(!cm.isSortable(index));
57292         ms.get("desc").setDisabled(!cm.isSortable(index));
57293         if(this.grid.enableColLock !== false){
57294             ms.get("lock").setDisabled(cm.isLocked(index));
57295             ms.get("unlock").setDisabled(!cm.isLocked(index));
57296         }
57297         this.hmenu.show(hd, "tl-bl");
57298     },
57299
57300     handleHdOver : function(e){
57301         var hd = this.findHeaderCell(e.getTarget());
57302         if(hd && !this.headersDisabled){
57303             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57304                this.fly(hd).addClass("x-grid-hd-over");
57305             }
57306         }
57307     },
57308
57309     handleHdOut : function(e){
57310         var hd = this.findHeaderCell(e.getTarget());
57311         if(hd){
57312             this.fly(hd).removeClass("x-grid-hd-over");
57313         }
57314     },
57315
57316     handleSplitDblClick : function(e, t){
57317         var i = this.getCellIndex(t);
57318         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57319             this.autoSizeColumn(i, true);
57320             this.layout();
57321         }
57322     },
57323
57324     render : function(){
57325
57326         var cm = this.cm;
57327         var colCount = cm.getColumnCount();
57328
57329         if(this.grid.monitorWindowResize === true){
57330             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57331         }
57332         var header = this.renderHeaders();
57333         var body = this.templates.body.apply({rows:""});
57334         var html = this.templates.master.apply({
57335             lockedBody: body,
57336             body: body,
57337             lockedHeader: header[0],
57338             header: header[1]
57339         });
57340
57341         //this.updateColumns();
57342
57343         this.grid.getGridEl().dom.innerHTML = html;
57344
57345         this.initElements();
57346         
57347         // a kludge to fix the random scolling effect in webkit
57348         this.el.on("scroll", function() {
57349             this.el.dom.scrollTop=0; // hopefully not recursive..
57350         },this);
57351
57352         this.scroller.on("scroll", this.handleScroll, this);
57353         this.lockedBody.on("mousewheel", this.handleWheel, this);
57354         this.mainBody.on("mousewheel", this.handleWheel, this);
57355
57356         this.mainHd.on("mouseover", this.handleHdOver, this);
57357         this.mainHd.on("mouseout", this.handleHdOut, this);
57358         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57359                 {delegate: "."+this.splitClass});
57360
57361         this.lockedHd.on("mouseover", this.handleHdOver, this);
57362         this.lockedHd.on("mouseout", this.handleHdOut, this);
57363         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57364                 {delegate: "."+this.splitClass});
57365
57366         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57367             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57368         }
57369
57370         this.updateSplitters();
57371
57372         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57373             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57374             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57375         }
57376
57377         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57378             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57379             this.hmenu.add(
57380                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57381                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57382             );
57383             if(this.grid.enableColLock !== false){
57384                 this.hmenu.add('-',
57385                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57386                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57387                 );
57388             }
57389             if (Roo.isTouch) {
57390                  this.hmenu.add('-',
57391                     {id:"wider", text: this.columnsWiderText},
57392                     {id:"narrow", text: this.columnsNarrowText }
57393                 );
57394                 
57395                  
57396             }
57397             
57398             if(this.grid.enableColumnHide !== false){
57399
57400                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57401                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57402                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57403
57404                 this.hmenu.add('-',
57405                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57406                 );
57407             }
57408             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57409
57410             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57411         }
57412
57413         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57414             this.dd = new Roo.grid.GridDragZone(this.grid, {
57415                 ddGroup : this.grid.ddGroup || 'GridDD'
57416             });
57417             
57418         }
57419
57420         /*
57421         for(var i = 0; i < colCount; i++){
57422             if(cm.isHidden(i)){
57423                 this.hideColumn(i);
57424             }
57425             if(cm.config[i].align){
57426                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57427                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57428             }
57429         }*/
57430         
57431         this.updateHeaderSortState();
57432
57433         this.beforeInitialResize();
57434         this.layout(true);
57435
57436         // two part rendering gives faster view to the user
57437         this.renderPhase2.defer(1, this);
57438     },
57439
57440     renderPhase2 : function(){
57441         // render the rows now
57442         this.refresh();
57443         if(this.grid.autoSizeColumns){
57444             this.autoSizeColumns();
57445         }
57446     },
57447
57448     beforeInitialResize : function(){
57449
57450     },
57451
57452     onColumnSplitterMoved : function(i, w){
57453         this.userResized = true;
57454         var cm = this.grid.colModel;
57455         cm.setColumnWidth(i, w, true);
57456         var cid = cm.getColumnId(i);
57457         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57458         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57459         this.updateSplitters();
57460         this.layout();
57461         this.grid.fireEvent("columnresize", i, w);
57462     },
57463
57464     syncRowHeights : function(startIndex, endIndex){
57465         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57466             startIndex = startIndex || 0;
57467             var mrows = this.getBodyTable().rows;
57468             var lrows = this.getLockedTable().rows;
57469             var len = mrows.length-1;
57470             endIndex = Math.min(endIndex || len, len);
57471             for(var i = startIndex; i <= endIndex; i++){
57472                 var m = mrows[i], l = lrows[i];
57473                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57474                 m.style.height = l.style.height = h + "px";
57475             }
57476         }
57477     },
57478
57479     layout : function(initialRender, is2ndPass){
57480         var g = this.grid;
57481         var auto = g.autoHeight;
57482         var scrollOffset = 16;
57483         var c = g.getGridEl(), cm = this.cm,
57484                 expandCol = g.autoExpandColumn,
57485                 gv = this;
57486         //c.beginMeasure();
57487
57488         if(!c.dom.offsetWidth){ // display:none?
57489             if(initialRender){
57490                 this.lockedWrap.show();
57491                 this.mainWrap.show();
57492             }
57493             return;
57494         }
57495
57496         var hasLock = this.cm.isLocked(0);
57497
57498         var tbh = this.headerPanel.getHeight();
57499         var bbh = this.footerPanel.getHeight();
57500
57501         if(auto){
57502             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57503             var newHeight = ch + c.getBorderWidth("tb");
57504             if(g.maxHeight){
57505                 newHeight = Math.min(g.maxHeight, newHeight);
57506             }
57507             c.setHeight(newHeight);
57508         }
57509
57510         if(g.autoWidth){
57511             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57512         }
57513
57514         var s = this.scroller;
57515
57516         var csize = c.getSize(true);
57517
57518         this.el.setSize(csize.width, csize.height);
57519
57520         this.headerPanel.setWidth(csize.width);
57521         this.footerPanel.setWidth(csize.width);
57522
57523         var hdHeight = this.mainHd.getHeight();
57524         var vw = csize.width;
57525         var vh = csize.height - (tbh + bbh);
57526
57527         s.setSize(vw, vh);
57528
57529         var bt = this.getBodyTable();
57530         
57531         if(cm.getLockedCount() == cm.config.length){
57532             bt = this.getLockedTable();
57533         }
57534         
57535         var ltWidth = hasLock ?
57536                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57537
57538         var scrollHeight = bt.offsetHeight;
57539         var scrollWidth = ltWidth + bt.offsetWidth;
57540         var vscroll = false, hscroll = false;
57541
57542         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57543
57544         var lw = this.lockedWrap, mw = this.mainWrap;
57545         var lb = this.lockedBody, mb = this.mainBody;
57546
57547         setTimeout(function(){
57548             var t = s.dom.offsetTop;
57549             var w = s.dom.clientWidth,
57550                 h = s.dom.clientHeight;
57551
57552             lw.setTop(t);
57553             lw.setSize(ltWidth, h);
57554
57555             mw.setLeftTop(ltWidth, t);
57556             mw.setSize(w-ltWidth, h);
57557
57558             lb.setHeight(h-hdHeight);
57559             mb.setHeight(h-hdHeight);
57560
57561             if(is2ndPass !== true && !gv.userResized && expandCol){
57562                 // high speed resize without full column calculation
57563                 
57564                 var ci = cm.getIndexById(expandCol);
57565                 if (ci < 0) {
57566                     ci = cm.findColumnIndex(expandCol);
57567                 }
57568                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57569                 var expandId = cm.getColumnId(ci);
57570                 var  tw = cm.getTotalWidth(false);
57571                 var currentWidth = cm.getColumnWidth(ci);
57572                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57573                 if(currentWidth != cw){
57574                     cm.setColumnWidth(ci, cw, true);
57575                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57576                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57577                     gv.updateSplitters();
57578                     gv.layout(false, true);
57579                 }
57580             }
57581
57582             if(initialRender){
57583                 lw.show();
57584                 mw.show();
57585             }
57586             //c.endMeasure();
57587         }, 10);
57588     },
57589
57590     onWindowResize : function(){
57591         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57592             return;
57593         }
57594         this.layout();
57595     },
57596
57597     appendFooter : function(parentEl){
57598         return null;
57599     },
57600
57601     sortAscText : "Sort Ascending",
57602     sortDescText : "Sort Descending",
57603     lockText : "Lock Column",
57604     unlockText : "Unlock Column",
57605     columnsText : "Columns",
57606  
57607     columnsWiderText : "Wider",
57608     columnsNarrowText : "Thinner"
57609 });
57610
57611
57612 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57613     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57614     this.proxy.el.addClass('x-grid3-col-dd');
57615 };
57616
57617 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57618     handleMouseDown : function(e){
57619
57620     },
57621
57622     callHandleMouseDown : function(e){
57623         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57624     }
57625 });
57626 /*
57627  * Based on:
57628  * Ext JS Library 1.1.1
57629  * Copyright(c) 2006-2007, Ext JS, LLC.
57630  *
57631  * Originally Released Under LGPL - original licence link has changed is not relivant.
57632  *
57633  * Fork - LGPL
57634  * <script type="text/javascript">
57635  */
57636  
57637 // private
57638 // This is a support class used internally by the Grid components
57639 Roo.grid.SplitDragZone = function(grid, hd, hd2){
57640     this.grid = grid;
57641     this.view = grid.getView();
57642     this.proxy = this.view.resizeProxy;
57643     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57644         "gridSplitters" + this.grid.getGridEl().id, {
57645         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57646     });
57647     this.setHandleElId(Roo.id(hd));
57648     this.setOuterHandleElId(Roo.id(hd2));
57649     this.scroll = false;
57650 };
57651 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57652     fly: Roo.Element.fly,
57653
57654     b4StartDrag : function(x, y){
57655         this.view.headersDisabled = true;
57656         this.proxy.setHeight(this.view.mainWrap.getHeight());
57657         var w = this.cm.getColumnWidth(this.cellIndex);
57658         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57659         this.resetConstraints();
57660         this.setXConstraint(minw, 1000);
57661         this.setYConstraint(0, 0);
57662         this.minX = x - minw;
57663         this.maxX = x + 1000;
57664         this.startPos = x;
57665         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57666     },
57667
57668
57669     handleMouseDown : function(e){
57670         ev = Roo.EventObject.setEvent(e);
57671         var t = this.fly(ev.getTarget());
57672         if(t.hasClass("x-grid-split")){
57673             this.cellIndex = this.view.getCellIndex(t.dom);
57674             this.split = t.dom;
57675             this.cm = this.grid.colModel;
57676             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57677                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57678             }
57679         }
57680     },
57681
57682     endDrag : function(e){
57683         this.view.headersDisabled = false;
57684         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57685         var diff = endX - this.startPos;
57686         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57687     },
57688
57689     autoOffset : function(){
57690         this.setDelta(0,0);
57691     }
57692 });/*
57693  * Based on:
57694  * Ext JS Library 1.1.1
57695  * Copyright(c) 2006-2007, Ext JS, LLC.
57696  *
57697  * Originally Released Under LGPL - original licence link has changed is not relivant.
57698  *
57699  * Fork - LGPL
57700  * <script type="text/javascript">
57701  */
57702  
57703 // private
57704 // This is a support class used internally by the Grid components
57705 Roo.grid.GridDragZone = function(grid, config){
57706     this.view = grid.getView();
57707     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57708     if(this.view.lockedBody){
57709         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57710         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57711     }
57712     this.scroll = false;
57713     this.grid = grid;
57714     this.ddel = document.createElement('div');
57715     this.ddel.className = 'x-grid-dd-wrap';
57716 };
57717
57718 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57719     ddGroup : "GridDD",
57720
57721     getDragData : function(e){
57722         var t = Roo.lib.Event.getTarget(e);
57723         var rowIndex = this.view.findRowIndex(t);
57724         var sm = this.grid.selModel;
57725             
57726         //Roo.log(rowIndex);
57727         
57728         if (sm.getSelectedCell) {
57729             // cell selection..
57730             if (!sm.getSelectedCell()) {
57731                 return false;
57732             }
57733             if (rowIndex != sm.getSelectedCell()[0]) {
57734                 return false;
57735             }
57736         
57737         }
57738         
57739         if(rowIndex !== false){
57740             
57741             // if editorgrid.. 
57742             
57743             
57744             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57745                
57746             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57747               //  
57748             //}
57749             if (e.hasModifier()){
57750                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57751             }
57752             
57753             Roo.log("getDragData");
57754             
57755             return {
57756                 grid: this.grid,
57757                 ddel: this.ddel,
57758                 rowIndex: rowIndex,
57759                 selections:sm.getSelections ? sm.getSelections() : (
57760                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57761                 )
57762             };
57763         }
57764         return false;
57765     },
57766
57767     onInitDrag : function(e){
57768         var data = this.dragData;
57769         this.ddel.innerHTML = this.grid.getDragDropText();
57770         this.proxy.update(this.ddel);
57771         // fire start drag?
57772     },
57773
57774     afterRepair : function(){
57775         this.dragging = false;
57776     },
57777
57778     getRepairXY : function(e, data){
57779         return false;
57780     },
57781
57782     onEndDrag : function(data, e){
57783         // fire end drag?
57784     },
57785
57786     onValidDrop : function(dd, e, id){
57787         // fire drag drop?
57788         this.hideProxy();
57789     },
57790
57791     beforeInvalidDrop : function(e, id){
57792
57793     }
57794 });/*
57795  * Based on:
57796  * Ext JS Library 1.1.1
57797  * Copyright(c) 2006-2007, Ext JS, LLC.
57798  *
57799  * Originally Released Under LGPL - original licence link has changed is not relivant.
57800  *
57801  * Fork - LGPL
57802  * <script type="text/javascript">
57803  */
57804  
57805
57806 /**
57807  * @class Roo.grid.ColumnModel
57808  * @extends Roo.util.Observable
57809  * This is the default implementation of a ColumnModel used by the Grid. It defines
57810  * the columns in the grid.
57811  * <br>Usage:<br>
57812  <pre><code>
57813  var colModel = new Roo.grid.ColumnModel([
57814         {header: "Ticker", width: 60, sortable: true, locked: true},
57815         {header: "Company Name", width: 150, sortable: true},
57816         {header: "Market Cap.", width: 100, sortable: true},
57817         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57818         {header: "Employees", width: 100, sortable: true, resizable: false}
57819  ]);
57820  </code></pre>
57821  * <p>
57822  
57823  * The config options listed for this class are options which may appear in each
57824  * individual column definition.
57825  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57826  * @constructor
57827  * @param {Object} config An Array of column config objects. See this class's
57828  * config objects for details.
57829 */
57830 Roo.grid.ColumnModel = function(config){
57831         /**
57832      * The config passed into the constructor
57833      */
57834     this.config = config;
57835     this.lookup = {};
57836
57837     // if no id, create one
57838     // if the column does not have a dataIndex mapping,
57839     // map it to the order it is in the config
57840     for(var i = 0, len = config.length; i < len; i++){
57841         var c = config[i];
57842         if(typeof c.dataIndex == "undefined"){
57843             c.dataIndex = i;
57844         }
57845         if(typeof c.renderer == "string"){
57846             c.renderer = Roo.util.Format[c.renderer];
57847         }
57848         if(typeof c.id == "undefined"){
57849             c.id = Roo.id();
57850         }
57851         if(c.editor && c.editor.xtype){
57852             c.editor  = Roo.factory(c.editor, Roo.grid);
57853         }
57854         if(c.editor && c.editor.isFormField){
57855             c.editor = new Roo.grid.GridEditor(c.editor);
57856         }
57857         this.lookup[c.id] = c;
57858     }
57859
57860     /**
57861      * The width of columns which have no width specified (defaults to 100)
57862      * @type Number
57863      */
57864     this.defaultWidth = 100;
57865
57866     /**
57867      * Default sortable of columns which have no sortable specified (defaults to false)
57868      * @type Boolean
57869      */
57870     this.defaultSortable = false;
57871
57872     this.addEvents({
57873         /**
57874              * @event widthchange
57875              * Fires when the width of a column changes.
57876              * @param {ColumnModel} this
57877              * @param {Number} columnIndex The column index
57878              * @param {Number} newWidth The new width
57879              */
57880             "widthchange": true,
57881         /**
57882              * @event headerchange
57883              * Fires when the text of a header changes.
57884              * @param {ColumnModel} this
57885              * @param {Number} columnIndex The column index
57886              * @param {Number} newText The new header text
57887              */
57888             "headerchange": true,
57889         /**
57890              * @event hiddenchange
57891              * Fires when a column is hidden or "unhidden".
57892              * @param {ColumnModel} this
57893              * @param {Number} columnIndex The column index
57894              * @param {Boolean} hidden true if hidden, false otherwise
57895              */
57896             "hiddenchange": true,
57897             /**
57898          * @event columnmoved
57899          * Fires when a column is moved.
57900          * @param {ColumnModel} this
57901          * @param {Number} oldIndex
57902          * @param {Number} newIndex
57903          */
57904         "columnmoved" : true,
57905         /**
57906          * @event columlockchange
57907          * Fires when a column's locked state is changed
57908          * @param {ColumnModel} this
57909          * @param {Number} colIndex
57910          * @param {Boolean} locked true if locked
57911          */
57912         "columnlockchange" : true
57913     });
57914     Roo.grid.ColumnModel.superclass.constructor.call(this);
57915 };
57916 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57917     /**
57918      * @cfg {String} header The header text to display in the Grid view.
57919      */
57920     /**
57921      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57922      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57923      * specified, the column's index is used as an index into the Record's data Array.
57924      */
57925     /**
57926      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57927      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57928      */
57929     /**
57930      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57931      * Defaults to the value of the {@link #defaultSortable} property.
57932      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57933      */
57934     /**
57935      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
57936      */
57937     /**
57938      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
57939      */
57940     /**
57941      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
57942      */
57943     /**
57944      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
57945      */
57946     /**
57947      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
57948      * given the cell's data value. See {@link #setRenderer}. If not specified, the
57949      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
57950      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
57951      */
57952        /**
57953      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
57954      */
57955     /**
57956      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
57957      */
57958     /**
57959      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
57960      */
57961     /**
57962      * @cfg {String} cursor (Optional)
57963      */
57964     /**
57965      * @cfg {String} tooltip (Optional)
57966      */
57967     /**
57968      * @cfg {Number} xs (Optional)
57969      */
57970     /**
57971      * @cfg {Number} sm (Optional)
57972      */
57973     /**
57974      * @cfg {Number} md (Optional)
57975      */
57976     /**
57977      * @cfg {Number} lg (Optional)
57978      */
57979     /**
57980      * Returns the id of the column at the specified index.
57981      * @param {Number} index The column index
57982      * @return {String} the id
57983      */
57984     getColumnId : function(index){
57985         return this.config[index].id;
57986     },
57987
57988     /**
57989      * Returns the column for a specified id.
57990      * @param {String} id The column id
57991      * @return {Object} the column
57992      */
57993     getColumnById : function(id){
57994         return this.lookup[id];
57995     },
57996
57997     
57998     /**
57999      * Returns the column for a specified dataIndex.
58000      * @param {String} dataIndex The column dataIndex
58001      * @return {Object|Boolean} the column or false if not found
58002      */
58003     getColumnByDataIndex: function(dataIndex){
58004         var index = this.findColumnIndex(dataIndex);
58005         return index > -1 ? this.config[index] : false;
58006     },
58007     
58008     /**
58009      * Returns the index for a specified column id.
58010      * @param {String} id The column id
58011      * @return {Number} the index, or -1 if not found
58012      */
58013     getIndexById : function(id){
58014         for(var i = 0, len = this.config.length; i < len; i++){
58015             if(this.config[i].id == id){
58016                 return i;
58017             }
58018         }
58019         return -1;
58020     },
58021     
58022     /**
58023      * Returns the index for a specified column dataIndex.
58024      * @param {String} dataIndex The column dataIndex
58025      * @return {Number} the index, or -1 if not found
58026      */
58027     
58028     findColumnIndex : function(dataIndex){
58029         for(var i = 0, len = this.config.length; i < len; i++){
58030             if(this.config[i].dataIndex == dataIndex){
58031                 return i;
58032             }
58033         }
58034         return -1;
58035     },
58036     
58037     
58038     moveColumn : function(oldIndex, newIndex){
58039         var c = this.config[oldIndex];
58040         this.config.splice(oldIndex, 1);
58041         this.config.splice(newIndex, 0, c);
58042         this.dataMap = null;
58043         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58044     },
58045
58046     isLocked : function(colIndex){
58047         return this.config[colIndex].locked === true;
58048     },
58049
58050     setLocked : function(colIndex, value, suppressEvent){
58051         if(this.isLocked(colIndex) == value){
58052             return;
58053         }
58054         this.config[colIndex].locked = value;
58055         if(!suppressEvent){
58056             this.fireEvent("columnlockchange", this, colIndex, value);
58057         }
58058     },
58059
58060     getTotalLockedWidth : function(){
58061         var totalWidth = 0;
58062         for(var i = 0; i < this.config.length; i++){
58063             if(this.isLocked(i) && !this.isHidden(i)){
58064                 this.totalWidth += this.getColumnWidth(i);
58065             }
58066         }
58067         return totalWidth;
58068     },
58069
58070     getLockedCount : function(){
58071         for(var i = 0, len = this.config.length; i < len; i++){
58072             if(!this.isLocked(i)){
58073                 return i;
58074             }
58075         }
58076         
58077         return this.config.length;
58078     },
58079
58080     /**
58081      * Returns the number of columns.
58082      * @return {Number}
58083      */
58084     getColumnCount : function(visibleOnly){
58085         if(visibleOnly === true){
58086             var c = 0;
58087             for(var i = 0, len = this.config.length; i < len; i++){
58088                 if(!this.isHidden(i)){
58089                     c++;
58090                 }
58091             }
58092             return c;
58093         }
58094         return this.config.length;
58095     },
58096
58097     /**
58098      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58099      * @param {Function} fn
58100      * @param {Object} scope (optional)
58101      * @return {Array} result
58102      */
58103     getColumnsBy : function(fn, scope){
58104         var r = [];
58105         for(var i = 0, len = this.config.length; i < len; i++){
58106             var c = this.config[i];
58107             if(fn.call(scope||this, c, i) === true){
58108                 r[r.length] = c;
58109             }
58110         }
58111         return r;
58112     },
58113
58114     /**
58115      * Returns true if the specified column is sortable.
58116      * @param {Number} col The column index
58117      * @return {Boolean}
58118      */
58119     isSortable : function(col){
58120         if(typeof this.config[col].sortable == "undefined"){
58121             return this.defaultSortable;
58122         }
58123         return this.config[col].sortable;
58124     },
58125
58126     /**
58127      * Returns the rendering (formatting) function defined for the column.
58128      * @param {Number} col The column index.
58129      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58130      */
58131     getRenderer : function(col){
58132         if(!this.config[col].renderer){
58133             return Roo.grid.ColumnModel.defaultRenderer;
58134         }
58135         return this.config[col].renderer;
58136     },
58137
58138     /**
58139      * Sets the rendering (formatting) function for a column.
58140      * @param {Number} col The column index
58141      * @param {Function} fn The function to use to process the cell's raw data
58142      * to return HTML markup for the grid view. The render function is called with
58143      * the following parameters:<ul>
58144      * <li>Data value.</li>
58145      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58146      * <li>css A CSS style string to apply to the table cell.</li>
58147      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58148      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58149      * <li>Row index</li>
58150      * <li>Column index</li>
58151      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58152      */
58153     setRenderer : function(col, fn){
58154         this.config[col].renderer = fn;
58155     },
58156
58157     /**
58158      * Returns the width for the specified column.
58159      * @param {Number} col The column index
58160      * @return {Number}
58161      */
58162     getColumnWidth : function(col){
58163         return this.config[col].width * 1 || this.defaultWidth;
58164     },
58165
58166     /**
58167      * Sets the width for a column.
58168      * @param {Number} col The column index
58169      * @param {Number} width The new width
58170      */
58171     setColumnWidth : function(col, width, suppressEvent){
58172         this.config[col].width = width;
58173         this.totalWidth = null;
58174         if(!suppressEvent){
58175              this.fireEvent("widthchange", this, col, width);
58176         }
58177     },
58178
58179     /**
58180      * Returns the total width of all columns.
58181      * @param {Boolean} includeHidden True to include hidden column widths
58182      * @return {Number}
58183      */
58184     getTotalWidth : function(includeHidden){
58185         if(!this.totalWidth){
58186             this.totalWidth = 0;
58187             for(var i = 0, len = this.config.length; i < len; i++){
58188                 if(includeHidden || !this.isHidden(i)){
58189                     this.totalWidth += this.getColumnWidth(i);
58190                 }
58191             }
58192         }
58193         return this.totalWidth;
58194     },
58195
58196     /**
58197      * Returns the header for the specified column.
58198      * @param {Number} col The column index
58199      * @return {String}
58200      */
58201     getColumnHeader : function(col){
58202         return this.config[col].header;
58203     },
58204
58205     /**
58206      * Sets the header for a column.
58207      * @param {Number} col The column index
58208      * @param {String} header The new header
58209      */
58210     setColumnHeader : function(col, header){
58211         this.config[col].header = header;
58212         this.fireEvent("headerchange", this, col, header);
58213     },
58214
58215     /**
58216      * Returns the tooltip for the specified column.
58217      * @param {Number} col The column index
58218      * @return {String}
58219      */
58220     getColumnTooltip : function(col){
58221             return this.config[col].tooltip;
58222     },
58223     /**
58224      * Sets the tooltip for a column.
58225      * @param {Number} col The column index
58226      * @param {String} tooltip The new tooltip
58227      */
58228     setColumnTooltip : function(col, tooltip){
58229             this.config[col].tooltip = tooltip;
58230     },
58231
58232     /**
58233      * Returns the dataIndex for the specified column.
58234      * @param {Number} col The column index
58235      * @return {Number}
58236      */
58237     getDataIndex : function(col){
58238         return this.config[col].dataIndex;
58239     },
58240
58241     /**
58242      * Sets the dataIndex for a column.
58243      * @param {Number} col The column index
58244      * @param {Number} dataIndex The new dataIndex
58245      */
58246     setDataIndex : function(col, dataIndex){
58247         this.config[col].dataIndex = dataIndex;
58248     },
58249
58250     
58251     
58252     /**
58253      * Returns true if the cell is editable.
58254      * @param {Number} colIndex The column index
58255      * @param {Number} rowIndex The row index - this is nto actually used..?
58256      * @return {Boolean}
58257      */
58258     isCellEditable : function(colIndex, rowIndex){
58259         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58260     },
58261
58262     /**
58263      * Returns the editor defined for the cell/column.
58264      * return false or null to disable editing.
58265      * @param {Number} colIndex The column index
58266      * @param {Number} rowIndex The row index
58267      * @return {Object}
58268      */
58269     getCellEditor : function(colIndex, rowIndex){
58270         return this.config[colIndex].editor;
58271     },
58272
58273     /**
58274      * Sets if a column is editable.
58275      * @param {Number} col The column index
58276      * @param {Boolean} editable True if the column is editable
58277      */
58278     setEditable : function(col, editable){
58279         this.config[col].editable = editable;
58280     },
58281
58282
58283     /**
58284      * Returns true if the column is hidden.
58285      * @param {Number} colIndex The column index
58286      * @return {Boolean}
58287      */
58288     isHidden : function(colIndex){
58289         return this.config[colIndex].hidden;
58290     },
58291
58292
58293     /**
58294      * Returns true if the column width cannot be changed
58295      */
58296     isFixed : function(colIndex){
58297         return this.config[colIndex].fixed;
58298     },
58299
58300     /**
58301      * Returns true if the column can be resized
58302      * @return {Boolean}
58303      */
58304     isResizable : function(colIndex){
58305         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58306     },
58307     /**
58308      * Sets if a column is hidden.
58309      * @param {Number} colIndex The column index
58310      * @param {Boolean} hidden True if the column is hidden
58311      */
58312     setHidden : function(colIndex, hidden){
58313         this.config[colIndex].hidden = hidden;
58314         this.totalWidth = null;
58315         this.fireEvent("hiddenchange", this, colIndex, hidden);
58316     },
58317
58318     /**
58319      * Sets the editor for a column.
58320      * @param {Number} col The column index
58321      * @param {Object} editor The editor object
58322      */
58323     setEditor : function(col, editor){
58324         this.config[col].editor = editor;
58325     }
58326 });
58327
58328 Roo.grid.ColumnModel.defaultRenderer = function(value)
58329 {
58330     if(typeof value == "object") {
58331         return value;
58332     }
58333         if(typeof value == "string" && value.length < 1){
58334             return "&#160;";
58335         }
58336     
58337         return String.format("{0}", value);
58338 };
58339
58340 // Alias for backwards compatibility
58341 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58342 /*
58343  * Based on:
58344  * Ext JS Library 1.1.1
58345  * Copyright(c) 2006-2007, Ext JS, LLC.
58346  *
58347  * Originally Released Under LGPL - original licence link has changed is not relivant.
58348  *
58349  * Fork - LGPL
58350  * <script type="text/javascript">
58351  */
58352
58353 /**
58354  * @class Roo.grid.AbstractSelectionModel
58355  * @extends Roo.util.Observable
58356  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58357  * implemented by descendant classes.  This class should not be directly instantiated.
58358  * @constructor
58359  */
58360 Roo.grid.AbstractSelectionModel = function(){
58361     this.locked = false;
58362     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58363 };
58364
58365 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58366     /** @ignore Called by the grid automatically. Do not call directly. */
58367     init : function(grid){
58368         this.grid = grid;
58369         this.initEvents();
58370     },
58371
58372     /**
58373      * Locks the selections.
58374      */
58375     lock : function(){
58376         this.locked = true;
58377     },
58378
58379     /**
58380      * Unlocks the selections.
58381      */
58382     unlock : function(){
58383         this.locked = false;
58384     },
58385
58386     /**
58387      * Returns true if the selections are locked.
58388      * @return {Boolean}
58389      */
58390     isLocked : function(){
58391         return this.locked;
58392     }
58393 });/*
58394  * Based on:
58395  * Ext JS Library 1.1.1
58396  * Copyright(c) 2006-2007, Ext JS, LLC.
58397  *
58398  * Originally Released Under LGPL - original licence link has changed is not relivant.
58399  *
58400  * Fork - LGPL
58401  * <script type="text/javascript">
58402  */
58403 /**
58404  * @extends Roo.grid.AbstractSelectionModel
58405  * @class Roo.grid.RowSelectionModel
58406  * The default SelectionModel used by {@link Roo.grid.Grid}.
58407  * It supports multiple selections and keyboard selection/navigation. 
58408  * @constructor
58409  * @param {Object} config
58410  */
58411 Roo.grid.RowSelectionModel = function(config){
58412     Roo.apply(this, config);
58413     this.selections = new Roo.util.MixedCollection(false, function(o){
58414         return o.id;
58415     });
58416
58417     this.last = false;
58418     this.lastActive = false;
58419
58420     this.addEvents({
58421         /**
58422              * @event selectionchange
58423              * Fires when the selection changes
58424              * @param {SelectionModel} this
58425              */
58426             "selectionchange" : true,
58427         /**
58428              * @event afterselectionchange
58429              * Fires after the selection changes (eg. by key press or clicking)
58430              * @param {SelectionModel} this
58431              */
58432             "afterselectionchange" : true,
58433         /**
58434              * @event beforerowselect
58435              * Fires when a row is selected being selected, return false to cancel.
58436              * @param {SelectionModel} this
58437              * @param {Number} rowIndex The selected index
58438              * @param {Boolean} keepExisting False if other selections will be cleared
58439              */
58440             "beforerowselect" : true,
58441         /**
58442              * @event rowselect
58443              * Fires when a row is selected.
58444              * @param {SelectionModel} this
58445              * @param {Number} rowIndex The selected index
58446              * @param {Roo.data.Record} r The record
58447              */
58448             "rowselect" : true,
58449         /**
58450              * @event rowdeselect
58451              * Fires when a row is deselected.
58452              * @param {SelectionModel} this
58453              * @param {Number} rowIndex The selected index
58454              */
58455         "rowdeselect" : true
58456     });
58457     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58458     this.locked = false;
58459 };
58460
58461 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58462     /**
58463      * @cfg {Boolean} singleSelect
58464      * True to allow selection of only one row at a time (defaults to false)
58465      */
58466     singleSelect : false,
58467
58468     // private
58469     initEvents : function(){
58470
58471         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58472             this.grid.on("mousedown", this.handleMouseDown, this);
58473         }else{ // allow click to work like normal
58474             this.grid.on("rowclick", this.handleDragableRowClick, this);
58475         }
58476
58477         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58478             "up" : function(e){
58479                 if(!e.shiftKey){
58480                     this.selectPrevious(e.shiftKey);
58481                 }else if(this.last !== false && this.lastActive !== false){
58482                     var last = this.last;
58483                     this.selectRange(this.last,  this.lastActive-1);
58484                     this.grid.getView().focusRow(this.lastActive);
58485                     if(last !== false){
58486                         this.last = last;
58487                     }
58488                 }else{
58489                     this.selectFirstRow();
58490                 }
58491                 this.fireEvent("afterselectionchange", this);
58492             },
58493             "down" : function(e){
58494                 if(!e.shiftKey){
58495                     this.selectNext(e.shiftKey);
58496                 }else if(this.last !== false && this.lastActive !== false){
58497                     var last = this.last;
58498                     this.selectRange(this.last,  this.lastActive+1);
58499                     this.grid.getView().focusRow(this.lastActive);
58500                     if(last !== false){
58501                         this.last = last;
58502                     }
58503                 }else{
58504                     this.selectFirstRow();
58505                 }
58506                 this.fireEvent("afterselectionchange", this);
58507             },
58508             scope: this
58509         });
58510
58511         var view = this.grid.view;
58512         view.on("refresh", this.onRefresh, this);
58513         view.on("rowupdated", this.onRowUpdated, this);
58514         view.on("rowremoved", this.onRemove, this);
58515     },
58516
58517     // private
58518     onRefresh : function(){
58519         var ds = this.grid.dataSource, i, v = this.grid.view;
58520         var s = this.selections;
58521         s.each(function(r){
58522             if((i = ds.indexOfId(r.id)) != -1){
58523                 v.onRowSelect(i);
58524                 s.add(ds.getAt(i)); // updating the selection relate data
58525             }else{
58526                 s.remove(r);
58527             }
58528         });
58529     },
58530
58531     // private
58532     onRemove : function(v, index, r){
58533         this.selections.remove(r);
58534     },
58535
58536     // private
58537     onRowUpdated : function(v, index, r){
58538         if(this.isSelected(r)){
58539             v.onRowSelect(index);
58540         }
58541     },
58542
58543     /**
58544      * Select records.
58545      * @param {Array} records The records to select
58546      * @param {Boolean} keepExisting (optional) True to keep existing selections
58547      */
58548     selectRecords : function(records, keepExisting){
58549         if(!keepExisting){
58550             this.clearSelections();
58551         }
58552         var ds = this.grid.dataSource;
58553         for(var i = 0, len = records.length; i < len; i++){
58554             this.selectRow(ds.indexOf(records[i]), true);
58555         }
58556     },
58557
58558     /**
58559      * Gets the number of selected rows.
58560      * @return {Number}
58561      */
58562     getCount : function(){
58563         return this.selections.length;
58564     },
58565
58566     /**
58567      * Selects the first row in the grid.
58568      */
58569     selectFirstRow : function(){
58570         this.selectRow(0);
58571     },
58572
58573     /**
58574      * Select the last row.
58575      * @param {Boolean} keepExisting (optional) True to keep existing selections
58576      */
58577     selectLastRow : function(keepExisting){
58578         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58579     },
58580
58581     /**
58582      * Selects the row immediately following the last selected row.
58583      * @param {Boolean} keepExisting (optional) True to keep existing selections
58584      */
58585     selectNext : function(keepExisting){
58586         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58587             this.selectRow(this.last+1, keepExisting);
58588             this.grid.getView().focusRow(this.last);
58589         }
58590     },
58591
58592     /**
58593      * Selects the row that precedes the last selected row.
58594      * @param {Boolean} keepExisting (optional) True to keep existing selections
58595      */
58596     selectPrevious : function(keepExisting){
58597         if(this.last){
58598             this.selectRow(this.last-1, keepExisting);
58599             this.grid.getView().focusRow(this.last);
58600         }
58601     },
58602
58603     /**
58604      * Returns the selected records
58605      * @return {Array} Array of selected records
58606      */
58607     getSelections : function(){
58608         return [].concat(this.selections.items);
58609     },
58610
58611     /**
58612      * Returns the first selected record.
58613      * @return {Record}
58614      */
58615     getSelected : function(){
58616         return this.selections.itemAt(0);
58617     },
58618
58619
58620     /**
58621      * Clears all selections.
58622      */
58623     clearSelections : function(fast){
58624         if(this.locked) {
58625             return;
58626         }
58627         if(fast !== true){
58628             var ds = this.grid.dataSource;
58629             var s = this.selections;
58630             s.each(function(r){
58631                 this.deselectRow(ds.indexOfId(r.id));
58632             }, this);
58633             s.clear();
58634         }else{
58635             this.selections.clear();
58636         }
58637         this.last = false;
58638     },
58639
58640
58641     /**
58642      * Selects all rows.
58643      */
58644     selectAll : function(){
58645         if(this.locked) {
58646             return;
58647         }
58648         this.selections.clear();
58649         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58650             this.selectRow(i, true);
58651         }
58652     },
58653
58654     /**
58655      * Returns True if there is a selection.
58656      * @return {Boolean}
58657      */
58658     hasSelection : function(){
58659         return this.selections.length > 0;
58660     },
58661
58662     /**
58663      * Returns True if the specified row is selected.
58664      * @param {Number/Record} record The record or index of the record to check
58665      * @return {Boolean}
58666      */
58667     isSelected : function(index){
58668         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58669         return (r && this.selections.key(r.id) ? true : false);
58670     },
58671
58672     /**
58673      * Returns True if the specified record id is selected.
58674      * @param {String} id The id of record to check
58675      * @return {Boolean}
58676      */
58677     isIdSelected : function(id){
58678         return (this.selections.key(id) ? true : false);
58679     },
58680
58681     // private
58682     handleMouseDown : function(e, t){
58683         var view = this.grid.getView(), rowIndex;
58684         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58685             return;
58686         };
58687         if(e.shiftKey && this.last !== false){
58688             var last = this.last;
58689             this.selectRange(last, rowIndex, e.ctrlKey);
58690             this.last = last; // reset the last
58691             view.focusRow(rowIndex);
58692         }else{
58693             var isSelected = this.isSelected(rowIndex);
58694             if(e.button !== 0 && isSelected){
58695                 view.focusRow(rowIndex);
58696             }else if(e.ctrlKey && isSelected){
58697                 this.deselectRow(rowIndex);
58698             }else if(!isSelected){
58699                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58700                 view.focusRow(rowIndex);
58701             }
58702         }
58703         this.fireEvent("afterselectionchange", this);
58704     },
58705     // private
58706     handleDragableRowClick :  function(grid, rowIndex, e) 
58707     {
58708         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58709             this.selectRow(rowIndex, false);
58710             grid.view.focusRow(rowIndex);
58711              this.fireEvent("afterselectionchange", this);
58712         }
58713     },
58714     
58715     /**
58716      * Selects multiple rows.
58717      * @param {Array} rows Array of the indexes of the row to select
58718      * @param {Boolean} keepExisting (optional) True to keep existing selections
58719      */
58720     selectRows : function(rows, keepExisting){
58721         if(!keepExisting){
58722             this.clearSelections();
58723         }
58724         for(var i = 0, len = rows.length; i < len; i++){
58725             this.selectRow(rows[i], true);
58726         }
58727     },
58728
58729     /**
58730      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58731      * @param {Number} startRow The index of the first row in the range
58732      * @param {Number} endRow The index of the last row in the range
58733      * @param {Boolean} keepExisting (optional) True to retain existing selections
58734      */
58735     selectRange : function(startRow, endRow, keepExisting){
58736         if(this.locked) {
58737             return;
58738         }
58739         if(!keepExisting){
58740             this.clearSelections();
58741         }
58742         if(startRow <= endRow){
58743             for(var i = startRow; i <= endRow; i++){
58744                 this.selectRow(i, true);
58745             }
58746         }else{
58747             for(var i = startRow; i >= endRow; i--){
58748                 this.selectRow(i, true);
58749             }
58750         }
58751     },
58752
58753     /**
58754      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58755      * @param {Number} startRow The index of the first row in the range
58756      * @param {Number} endRow The index of the last row in the range
58757      */
58758     deselectRange : function(startRow, endRow, preventViewNotify){
58759         if(this.locked) {
58760             return;
58761         }
58762         for(var i = startRow; i <= endRow; i++){
58763             this.deselectRow(i, preventViewNotify);
58764         }
58765     },
58766
58767     /**
58768      * Selects a row.
58769      * @param {Number} row The index of the row to select
58770      * @param {Boolean} keepExisting (optional) True to keep existing selections
58771      */
58772     selectRow : function(index, keepExisting, preventViewNotify){
58773         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58774             return;
58775         }
58776         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58777             if(!keepExisting || this.singleSelect){
58778                 this.clearSelections();
58779             }
58780             var r = this.grid.dataSource.getAt(index);
58781             this.selections.add(r);
58782             this.last = this.lastActive = index;
58783             if(!preventViewNotify){
58784                 this.grid.getView().onRowSelect(index);
58785             }
58786             this.fireEvent("rowselect", this, index, r);
58787             this.fireEvent("selectionchange", this);
58788         }
58789     },
58790
58791     /**
58792      * Deselects a row.
58793      * @param {Number} row The index of the row to deselect
58794      */
58795     deselectRow : function(index, preventViewNotify){
58796         if(this.locked) {
58797             return;
58798         }
58799         if(this.last == index){
58800             this.last = false;
58801         }
58802         if(this.lastActive == index){
58803             this.lastActive = false;
58804         }
58805         var r = this.grid.dataSource.getAt(index);
58806         this.selections.remove(r);
58807         if(!preventViewNotify){
58808             this.grid.getView().onRowDeselect(index);
58809         }
58810         this.fireEvent("rowdeselect", this, index);
58811         this.fireEvent("selectionchange", this);
58812     },
58813
58814     // private
58815     restoreLast : function(){
58816         if(this._last){
58817             this.last = this._last;
58818         }
58819     },
58820
58821     // private
58822     acceptsNav : function(row, col, cm){
58823         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58824     },
58825
58826     // private
58827     onEditorKey : function(field, e){
58828         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58829         if(k == e.TAB){
58830             e.stopEvent();
58831             ed.completeEdit();
58832             if(e.shiftKey){
58833                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58834             }else{
58835                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58836             }
58837         }else if(k == e.ENTER && !e.ctrlKey){
58838             e.stopEvent();
58839             ed.completeEdit();
58840             if(e.shiftKey){
58841                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58842             }else{
58843                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58844             }
58845         }else if(k == e.ESC){
58846             ed.cancelEdit();
58847         }
58848         if(newCell){
58849             g.startEditing(newCell[0], newCell[1]);
58850         }
58851     }
58852 });/*
58853  * Based on:
58854  * Ext JS Library 1.1.1
58855  * Copyright(c) 2006-2007, Ext JS, LLC.
58856  *
58857  * Originally Released Under LGPL - original licence link has changed is not relivant.
58858  *
58859  * Fork - LGPL
58860  * <script type="text/javascript">
58861  */
58862 /**
58863  * @class Roo.grid.CellSelectionModel
58864  * @extends Roo.grid.AbstractSelectionModel
58865  * This class provides the basic implementation for cell selection in a grid.
58866  * @constructor
58867  * @param {Object} config The object containing the configuration of this model.
58868  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58869  */
58870 Roo.grid.CellSelectionModel = function(config){
58871     Roo.apply(this, config);
58872
58873     this.selection = null;
58874
58875     this.addEvents({
58876         /**
58877              * @event beforerowselect
58878              * Fires before a cell is selected.
58879              * @param {SelectionModel} this
58880              * @param {Number} rowIndex The selected row index
58881              * @param {Number} colIndex The selected cell index
58882              */
58883             "beforecellselect" : true,
58884         /**
58885              * @event cellselect
58886              * Fires when a cell is selected.
58887              * @param {SelectionModel} this
58888              * @param {Number} rowIndex The selected row index
58889              * @param {Number} colIndex The selected cell index
58890              */
58891             "cellselect" : true,
58892         /**
58893              * @event selectionchange
58894              * Fires when the active selection changes.
58895              * @param {SelectionModel} this
58896              * @param {Object} selection null for no selection or an object (o) with two properties
58897                 <ul>
58898                 <li>o.record: the record object for the row the selection is in</li>
58899                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58900                 </ul>
58901              */
58902             "selectionchange" : true,
58903         /**
58904              * @event tabend
58905              * Fires when the tab (or enter) was pressed on the last editable cell
58906              * You can use this to trigger add new row.
58907              * @param {SelectionModel} this
58908              */
58909             "tabend" : true,
58910          /**
58911              * @event beforeeditnext
58912              * Fires before the next editable sell is made active
58913              * You can use this to skip to another cell or fire the tabend
58914              *    if you set cell to false
58915              * @param {Object} eventdata object : { cell : [ row, col ] } 
58916              */
58917             "beforeeditnext" : true
58918     });
58919     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58920 };
58921
58922 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58923     
58924     enter_is_tab: false,
58925
58926     /** @ignore */
58927     initEvents : function(){
58928         this.grid.on("mousedown", this.handleMouseDown, this);
58929         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58930         var view = this.grid.view;
58931         view.on("refresh", this.onViewChange, this);
58932         view.on("rowupdated", this.onRowUpdated, this);
58933         view.on("beforerowremoved", this.clearSelections, this);
58934         view.on("beforerowsinserted", this.clearSelections, this);
58935         if(this.grid.isEditor){
58936             this.grid.on("beforeedit", this.beforeEdit,  this);
58937         }
58938     },
58939
58940         //private
58941     beforeEdit : function(e){
58942         this.select(e.row, e.column, false, true, e.record);
58943     },
58944
58945         //private
58946     onRowUpdated : function(v, index, r){
58947         if(this.selection && this.selection.record == r){
58948             v.onCellSelect(index, this.selection.cell[1]);
58949         }
58950     },
58951
58952         //private
58953     onViewChange : function(){
58954         this.clearSelections(true);
58955     },
58956
58957         /**
58958          * Returns the currently selected cell,.
58959          * @return {Array} The selected cell (row, column) or null if none selected.
58960          */
58961     getSelectedCell : function(){
58962         return this.selection ? this.selection.cell : null;
58963     },
58964
58965     /**
58966      * Clears all selections.
58967      * @param {Boolean} true to prevent the gridview from being notified about the change.
58968      */
58969     clearSelections : function(preventNotify){
58970         var s = this.selection;
58971         if(s){
58972             if(preventNotify !== true){
58973                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
58974             }
58975             this.selection = null;
58976             this.fireEvent("selectionchange", this, null);
58977         }
58978     },
58979
58980     /**
58981      * Returns true if there is a selection.
58982      * @return {Boolean}
58983      */
58984     hasSelection : function(){
58985         return this.selection ? true : false;
58986     },
58987
58988     /** @ignore */
58989     handleMouseDown : function(e, t){
58990         var v = this.grid.getView();
58991         if(this.isLocked()){
58992             return;
58993         };
58994         var row = v.findRowIndex(t);
58995         var cell = v.findCellIndex(t);
58996         if(row !== false && cell !== false){
58997             this.select(row, cell);
58998         }
58999     },
59000
59001     /**
59002      * Selects a cell.
59003      * @param {Number} rowIndex
59004      * @param {Number} collIndex
59005      */
59006     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59007         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59008             this.clearSelections();
59009             r = r || this.grid.dataSource.getAt(rowIndex);
59010             this.selection = {
59011                 record : r,
59012                 cell : [rowIndex, colIndex]
59013             };
59014             if(!preventViewNotify){
59015                 var v = this.grid.getView();
59016                 v.onCellSelect(rowIndex, colIndex);
59017                 if(preventFocus !== true){
59018                     v.focusCell(rowIndex, colIndex);
59019                 }
59020             }
59021             this.fireEvent("cellselect", this, rowIndex, colIndex);
59022             this.fireEvent("selectionchange", this, this.selection);
59023         }
59024     },
59025
59026         //private
59027     isSelectable : function(rowIndex, colIndex, cm){
59028         return !cm.isHidden(colIndex);
59029     },
59030
59031     /** @ignore */
59032     handleKeyDown : function(e){
59033         //Roo.log('Cell Sel Model handleKeyDown');
59034         if(!e.isNavKeyPress()){
59035             return;
59036         }
59037         var g = this.grid, s = this.selection;
59038         if(!s){
59039             e.stopEvent();
59040             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59041             if(cell){
59042                 this.select(cell[0], cell[1]);
59043             }
59044             return;
59045         }
59046         var sm = this;
59047         var walk = function(row, col, step){
59048             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59049         };
59050         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59051         var newCell;
59052
59053       
59054
59055         switch(k){
59056             case e.TAB:
59057                 // handled by onEditorKey
59058                 if (g.isEditor && g.editing) {
59059                     return;
59060                 }
59061                 if(e.shiftKey) {
59062                     newCell = walk(r, c-1, -1);
59063                 } else {
59064                     newCell = walk(r, c+1, 1);
59065                 }
59066                 break;
59067             
59068             case e.DOWN:
59069                newCell = walk(r+1, c, 1);
59070                 break;
59071             
59072             case e.UP:
59073                 newCell = walk(r-1, c, -1);
59074                 break;
59075             
59076             case e.RIGHT:
59077                 newCell = walk(r, c+1, 1);
59078                 break;
59079             
59080             case e.LEFT:
59081                 newCell = walk(r, c-1, -1);
59082                 break;
59083             
59084             case e.ENTER:
59085                 
59086                 if(g.isEditor && !g.editing){
59087                    g.startEditing(r, c);
59088                    e.stopEvent();
59089                    return;
59090                 }
59091                 
59092                 
59093              break;
59094         };
59095         if(newCell){
59096             this.select(newCell[0], newCell[1]);
59097             e.stopEvent();
59098             
59099         }
59100     },
59101
59102     acceptsNav : function(row, col, cm){
59103         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59104     },
59105     /**
59106      * Selects a cell.
59107      * @param {Number} field (not used) - as it's normally used as a listener
59108      * @param {Number} e - event - fake it by using
59109      *
59110      * var e = Roo.EventObjectImpl.prototype;
59111      * e.keyCode = e.TAB
59112      *
59113      * 
59114      */
59115     onEditorKey : function(field, e){
59116         
59117         var k = e.getKey(),
59118             newCell,
59119             g = this.grid,
59120             ed = g.activeEditor,
59121             forward = false;
59122         ///Roo.log('onEditorKey' + k);
59123         
59124         
59125         if (this.enter_is_tab && k == e.ENTER) {
59126             k = e.TAB;
59127         }
59128         
59129         if(k == e.TAB){
59130             if(e.shiftKey){
59131                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59132             }else{
59133                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59134                 forward = true;
59135             }
59136             
59137             e.stopEvent();
59138             
59139         } else if(k == e.ENTER &&  !e.ctrlKey){
59140             ed.completeEdit();
59141             e.stopEvent();
59142             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59143         
59144                 } else if(k == e.ESC){
59145             ed.cancelEdit();
59146         }
59147                 
59148         if (newCell) {
59149             var ecall = { cell : newCell, forward : forward };
59150             this.fireEvent('beforeeditnext', ecall );
59151             newCell = ecall.cell;
59152                         forward = ecall.forward;
59153         }
59154                 
59155         if(newCell){
59156             //Roo.log('next cell after edit');
59157             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59158         } else if (forward) {
59159             // tabbed past last
59160             this.fireEvent.defer(100, this, ['tabend',this]);
59161         }
59162     }
59163 });/*
59164  * Based on:
59165  * Ext JS Library 1.1.1
59166  * Copyright(c) 2006-2007, Ext JS, LLC.
59167  *
59168  * Originally Released Under LGPL - original licence link has changed is not relivant.
59169  *
59170  * Fork - LGPL
59171  * <script type="text/javascript">
59172  */
59173  
59174 /**
59175  * @class Roo.grid.EditorGrid
59176  * @extends Roo.grid.Grid
59177  * Class for creating and editable grid.
59178  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59179  * The container MUST have some type of size defined for the grid to fill. The container will be 
59180  * automatically set to position relative if it isn't already.
59181  * @param {Object} dataSource The data model to bind to
59182  * @param {Object} colModel The column model with info about this grid's columns
59183  */
59184 Roo.grid.EditorGrid = function(container, config){
59185     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59186     this.getGridEl().addClass("xedit-grid");
59187
59188     if(!this.selModel){
59189         this.selModel = new Roo.grid.CellSelectionModel();
59190     }
59191
59192     this.activeEditor = null;
59193
59194         this.addEvents({
59195             /**
59196              * @event beforeedit
59197              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59198              * <ul style="padding:5px;padding-left:16px;">
59199              * <li>grid - This grid</li>
59200              * <li>record - The record being edited</li>
59201              * <li>field - The field name being edited</li>
59202              * <li>value - The value for the field being edited.</li>
59203              * <li>row - The grid row index</li>
59204              * <li>column - The grid column index</li>
59205              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59206              * </ul>
59207              * @param {Object} e An edit event (see above for description)
59208              */
59209             "beforeedit" : true,
59210             /**
59211              * @event afteredit
59212              * Fires after a cell is edited. <br />
59213              * <ul style="padding:5px;padding-left:16px;">
59214              * <li>grid - This grid</li>
59215              * <li>record - The record being edited</li>
59216              * <li>field - The field name being edited</li>
59217              * <li>value - The value being set</li>
59218              * <li>originalValue - The original value for the field, before the edit.</li>
59219              * <li>row - The grid row index</li>
59220              * <li>column - The grid column index</li>
59221              * </ul>
59222              * @param {Object} e An edit event (see above for description)
59223              */
59224             "afteredit" : true,
59225             /**
59226              * @event validateedit
59227              * Fires after a cell is edited, but before the value is set in the record. 
59228          * You can use this to modify the value being set in the field, Return false
59229              * to cancel the change. The edit event object has the following properties <br />
59230              * <ul style="padding:5px;padding-left:16px;">
59231          * <li>editor - This editor</li>
59232              * <li>grid - This grid</li>
59233              * <li>record - The record being edited</li>
59234              * <li>field - The field name being edited</li>
59235              * <li>value - The value being set</li>
59236              * <li>originalValue - The original value for the field, before the edit.</li>
59237              * <li>row - The grid row index</li>
59238              * <li>column - The grid column index</li>
59239              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59240              * </ul>
59241              * @param {Object} e An edit event (see above for description)
59242              */
59243             "validateedit" : true
59244         });
59245     this.on("bodyscroll", this.stopEditing,  this);
59246     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59247 };
59248
59249 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59250     /**
59251      * @cfg {Number} clicksToEdit
59252      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59253      */
59254     clicksToEdit: 2,
59255
59256     // private
59257     isEditor : true,
59258     // private
59259     trackMouseOver: false, // causes very odd FF errors
59260
59261     onCellDblClick : function(g, row, col){
59262         this.startEditing(row, col);
59263     },
59264
59265     onEditComplete : function(ed, value, startValue){
59266         this.editing = false;
59267         this.activeEditor = null;
59268         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59269         var r = ed.record;
59270         var field = this.colModel.getDataIndex(ed.col);
59271         var e = {
59272             grid: this,
59273             record: r,
59274             field: field,
59275             originalValue: startValue,
59276             value: value,
59277             row: ed.row,
59278             column: ed.col,
59279             cancel:false,
59280             editor: ed
59281         };
59282         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59283         cell.show();
59284           
59285         if(String(value) !== String(startValue)){
59286             
59287             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59288                 r.set(field, e.value);
59289                 // if we are dealing with a combo box..
59290                 // then we also set the 'name' colum to be the displayField
59291                 if (ed.field.displayField && ed.field.name) {
59292                     r.set(ed.field.name, ed.field.el.dom.value);
59293                 }
59294                 
59295                 delete e.cancel; //?? why!!!
59296                 this.fireEvent("afteredit", e);
59297             }
59298         } else {
59299             this.fireEvent("afteredit", e); // always fire it!
59300         }
59301         this.view.focusCell(ed.row, ed.col);
59302     },
59303
59304     /**
59305      * Starts editing the specified for the specified row/column
59306      * @param {Number} rowIndex
59307      * @param {Number} colIndex
59308      */
59309     startEditing : function(row, col){
59310         this.stopEditing();
59311         if(this.colModel.isCellEditable(col, row)){
59312             this.view.ensureVisible(row, col, true);
59313           
59314             var r = this.dataSource.getAt(row);
59315             var field = this.colModel.getDataIndex(col);
59316             var cell = Roo.get(this.view.getCell(row,col));
59317             var e = {
59318                 grid: this,
59319                 record: r,
59320                 field: field,
59321                 value: r.data[field],
59322                 row: row,
59323                 column: col,
59324                 cancel:false 
59325             };
59326             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59327                 this.editing = true;
59328                 var ed = this.colModel.getCellEditor(col, row);
59329                 
59330                 if (!ed) {
59331                     return;
59332                 }
59333                 if(!ed.rendered){
59334                     ed.render(ed.parentEl || document.body);
59335                 }
59336                 ed.field.reset();
59337                
59338                 cell.hide();
59339                 
59340                 (function(){ // complex but required for focus issues in safari, ie and opera
59341                     ed.row = row;
59342                     ed.col = col;
59343                     ed.record = r;
59344                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59345                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59346                     this.activeEditor = ed;
59347                     var v = r.data[field];
59348                     ed.startEdit(this.view.getCell(row, col), v);
59349                     // combo's with 'displayField and name set
59350                     if (ed.field.displayField && ed.field.name) {
59351                         ed.field.el.dom.value = r.data[ed.field.name];
59352                     }
59353                     
59354                     
59355                 }).defer(50, this);
59356             }
59357         }
59358     },
59359         
59360     /**
59361      * Stops any active editing
59362      */
59363     stopEditing : function(){
59364         if(this.activeEditor){
59365             this.activeEditor.completeEdit();
59366         }
59367         this.activeEditor = null;
59368     },
59369         
59370          /**
59371      * Called to get grid's drag proxy text, by default returns this.ddText.
59372      * @return {String}
59373      */
59374     getDragDropText : function(){
59375         var count = this.selModel.getSelectedCell() ? 1 : 0;
59376         return String.format(this.ddText, count, count == 1 ? '' : 's');
59377     }
59378         
59379 });/*
59380  * Based on:
59381  * Ext JS Library 1.1.1
59382  * Copyright(c) 2006-2007, Ext JS, LLC.
59383  *
59384  * Originally Released Under LGPL - original licence link has changed is not relivant.
59385  *
59386  * Fork - LGPL
59387  * <script type="text/javascript">
59388  */
59389
59390 // private - not really -- you end up using it !
59391 // This is a support class used internally by the Grid components
59392
59393 /**
59394  * @class Roo.grid.GridEditor
59395  * @extends Roo.Editor
59396  * Class for creating and editable grid elements.
59397  * @param {Object} config any settings (must include field)
59398  */
59399 Roo.grid.GridEditor = function(field, config){
59400     if (!config && field.field) {
59401         config = field;
59402         field = Roo.factory(config.field, Roo.form);
59403     }
59404     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59405     field.monitorTab = false;
59406 };
59407
59408 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59409     
59410     /**
59411      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59412      */
59413     
59414     alignment: "tl-tl",
59415     autoSize: "width",
59416     hideEl : false,
59417     cls: "x-small-editor x-grid-editor",
59418     shim:false,
59419     shadow:"frame"
59420 });/*
59421  * Based on:
59422  * Ext JS Library 1.1.1
59423  * Copyright(c) 2006-2007, Ext JS, LLC.
59424  *
59425  * Originally Released Under LGPL - original licence link has changed is not relivant.
59426  *
59427  * Fork - LGPL
59428  * <script type="text/javascript">
59429  */
59430   
59431
59432   
59433 Roo.grid.PropertyRecord = Roo.data.Record.create([
59434     {name:'name',type:'string'},  'value'
59435 ]);
59436
59437
59438 Roo.grid.PropertyStore = function(grid, source){
59439     this.grid = grid;
59440     this.store = new Roo.data.Store({
59441         recordType : Roo.grid.PropertyRecord
59442     });
59443     this.store.on('update', this.onUpdate,  this);
59444     if(source){
59445         this.setSource(source);
59446     }
59447     Roo.grid.PropertyStore.superclass.constructor.call(this);
59448 };
59449
59450
59451
59452 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59453     setSource : function(o){
59454         this.source = o;
59455         this.store.removeAll();
59456         var data = [];
59457         for(var k in o){
59458             if(this.isEditableValue(o[k])){
59459                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59460             }
59461         }
59462         this.store.loadRecords({records: data}, {}, true);
59463     },
59464
59465     onUpdate : function(ds, record, type){
59466         if(type == Roo.data.Record.EDIT){
59467             var v = record.data['value'];
59468             var oldValue = record.modified['value'];
59469             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59470                 this.source[record.id] = v;
59471                 record.commit();
59472                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59473             }else{
59474                 record.reject();
59475             }
59476         }
59477     },
59478
59479     getProperty : function(row){
59480        return this.store.getAt(row);
59481     },
59482
59483     isEditableValue: function(val){
59484         if(val && val instanceof Date){
59485             return true;
59486         }else if(typeof val == 'object' || typeof val == 'function'){
59487             return false;
59488         }
59489         return true;
59490     },
59491
59492     setValue : function(prop, value){
59493         this.source[prop] = value;
59494         this.store.getById(prop).set('value', value);
59495     },
59496
59497     getSource : function(){
59498         return this.source;
59499     }
59500 });
59501
59502 Roo.grid.PropertyColumnModel = function(grid, store){
59503     this.grid = grid;
59504     var g = Roo.grid;
59505     g.PropertyColumnModel.superclass.constructor.call(this, [
59506         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59507         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59508     ]);
59509     this.store = store;
59510     this.bselect = Roo.DomHelper.append(document.body, {
59511         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59512             {tag: 'option', value: 'true', html: 'true'},
59513             {tag: 'option', value: 'false', html: 'false'}
59514         ]
59515     });
59516     Roo.id(this.bselect);
59517     var f = Roo.form;
59518     this.editors = {
59519         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59520         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59521         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59522         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59523         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59524     };
59525     this.renderCellDelegate = this.renderCell.createDelegate(this);
59526     this.renderPropDelegate = this.renderProp.createDelegate(this);
59527 };
59528
59529 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59530     
59531     
59532     nameText : 'Name',
59533     valueText : 'Value',
59534     
59535     dateFormat : 'm/j/Y',
59536     
59537     
59538     renderDate : function(dateVal){
59539         return dateVal.dateFormat(this.dateFormat);
59540     },
59541
59542     renderBool : function(bVal){
59543         return bVal ? 'true' : 'false';
59544     },
59545
59546     isCellEditable : function(colIndex, rowIndex){
59547         return colIndex == 1;
59548     },
59549
59550     getRenderer : function(col){
59551         return col == 1 ?
59552             this.renderCellDelegate : this.renderPropDelegate;
59553     },
59554
59555     renderProp : function(v){
59556         return this.getPropertyName(v);
59557     },
59558
59559     renderCell : function(val){
59560         var rv = val;
59561         if(val instanceof Date){
59562             rv = this.renderDate(val);
59563         }else if(typeof val == 'boolean'){
59564             rv = this.renderBool(val);
59565         }
59566         return Roo.util.Format.htmlEncode(rv);
59567     },
59568
59569     getPropertyName : function(name){
59570         var pn = this.grid.propertyNames;
59571         return pn && pn[name] ? pn[name] : name;
59572     },
59573
59574     getCellEditor : function(colIndex, rowIndex){
59575         var p = this.store.getProperty(rowIndex);
59576         var n = p.data['name'], val = p.data['value'];
59577         
59578         if(typeof(this.grid.customEditors[n]) == 'string'){
59579             return this.editors[this.grid.customEditors[n]];
59580         }
59581         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59582             return this.grid.customEditors[n];
59583         }
59584         if(val instanceof Date){
59585             return this.editors['date'];
59586         }else if(typeof val == 'number'){
59587             return this.editors['number'];
59588         }else if(typeof val == 'boolean'){
59589             return this.editors['boolean'];
59590         }else{
59591             return this.editors['string'];
59592         }
59593     }
59594 });
59595
59596 /**
59597  * @class Roo.grid.PropertyGrid
59598  * @extends Roo.grid.EditorGrid
59599  * This class represents the  interface of a component based property grid control.
59600  * <br><br>Usage:<pre><code>
59601  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59602       
59603  });
59604  // set any options
59605  grid.render();
59606  * </code></pre>
59607   
59608  * @constructor
59609  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59610  * The container MUST have some type of size defined for the grid to fill. The container will be
59611  * automatically set to position relative if it isn't already.
59612  * @param {Object} config A config object that sets properties on this grid.
59613  */
59614 Roo.grid.PropertyGrid = function(container, config){
59615     config = config || {};
59616     var store = new Roo.grid.PropertyStore(this);
59617     this.store = store;
59618     var cm = new Roo.grid.PropertyColumnModel(this, store);
59619     store.store.sort('name', 'ASC');
59620     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59621         ds: store.store,
59622         cm: cm,
59623         enableColLock:false,
59624         enableColumnMove:false,
59625         stripeRows:false,
59626         trackMouseOver: false,
59627         clicksToEdit:1
59628     }, config));
59629     this.getGridEl().addClass('x-props-grid');
59630     this.lastEditRow = null;
59631     this.on('columnresize', this.onColumnResize, this);
59632     this.addEvents({
59633          /**
59634              * @event beforepropertychange
59635              * Fires before a property changes (return false to stop?)
59636              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59637              * @param {String} id Record Id
59638              * @param {String} newval New Value
59639          * @param {String} oldval Old Value
59640              */
59641         "beforepropertychange": true,
59642         /**
59643              * @event propertychange
59644              * Fires after a property changes
59645              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59646              * @param {String} id Record Id
59647              * @param {String} newval New Value
59648          * @param {String} oldval Old Value
59649              */
59650         "propertychange": true
59651     });
59652     this.customEditors = this.customEditors || {};
59653 };
59654 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59655     
59656      /**
59657      * @cfg {Object} customEditors map of colnames=> custom editors.
59658      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59659      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59660      * false disables editing of the field.
59661          */
59662     
59663       /**
59664      * @cfg {Object} propertyNames map of property Names to their displayed value
59665          */
59666     
59667     render : function(){
59668         Roo.grid.PropertyGrid.superclass.render.call(this);
59669         this.autoSize.defer(100, this);
59670     },
59671
59672     autoSize : function(){
59673         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59674         if(this.view){
59675             this.view.fitColumns();
59676         }
59677     },
59678
59679     onColumnResize : function(){
59680         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59681         this.autoSize();
59682     },
59683     /**
59684      * Sets the data for the Grid
59685      * accepts a Key => Value object of all the elements avaiable.
59686      * @param {Object} data  to appear in grid.
59687      */
59688     setSource : function(source){
59689         this.store.setSource(source);
59690         //this.autoSize();
59691     },
59692     /**
59693      * Gets all the data from the grid.
59694      * @return {Object} data  data stored in grid
59695      */
59696     getSource : function(){
59697         return this.store.getSource();
59698     }
59699 });/*
59700   
59701  * Licence LGPL
59702  
59703  */
59704  
59705 /**
59706  * @class Roo.grid.Calendar
59707  * @extends Roo.util.Grid
59708  * This class extends the Grid to provide a calendar widget
59709  * <br><br>Usage:<pre><code>
59710  var grid = new Roo.grid.Calendar("my-container-id", {
59711      ds: myDataStore,
59712      cm: myColModel,
59713      selModel: mySelectionModel,
59714      autoSizeColumns: true,
59715      monitorWindowResize: false,
59716      trackMouseOver: true
59717      eventstore : real data store..
59718  });
59719  // set any options
59720  grid.render();
59721   
59722   * @constructor
59723  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59724  * The container MUST have some type of size defined for the grid to fill. The container will be
59725  * automatically set to position relative if it isn't already.
59726  * @param {Object} config A config object that sets properties on this grid.
59727  */
59728 Roo.grid.Calendar = function(container, config){
59729         // initialize the container
59730         this.container = Roo.get(container);
59731         this.container.update("");
59732         this.container.setStyle("overflow", "hidden");
59733     this.container.addClass('x-grid-container');
59734
59735     this.id = this.container.id;
59736
59737     Roo.apply(this, config);
59738     // check and correct shorthanded configs
59739     
59740     var rows = [];
59741     var d =1;
59742     for (var r = 0;r < 6;r++) {
59743         
59744         rows[r]=[];
59745         for (var c =0;c < 7;c++) {
59746             rows[r][c]= '';
59747         }
59748     }
59749     if (this.eventStore) {
59750         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59751         this.eventStore.on('load',this.onLoad, this);
59752         this.eventStore.on('beforeload',this.clearEvents, this);
59753          
59754     }
59755     
59756     this.dataSource = new Roo.data.Store({
59757             proxy: new Roo.data.MemoryProxy(rows),
59758             reader: new Roo.data.ArrayReader({}, [
59759                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59760     });
59761
59762     this.dataSource.load();
59763     this.ds = this.dataSource;
59764     this.ds.xmodule = this.xmodule || false;
59765     
59766     
59767     var cellRender = function(v,x,r)
59768     {
59769         return String.format(
59770             '<div class="fc-day  fc-widget-content"><div>' +
59771                 '<div class="fc-event-container"></div>' +
59772                 '<div class="fc-day-number">{0}</div>'+
59773                 
59774                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59775             '</div></div>', v);
59776     
59777     }
59778     
59779     
59780     this.colModel = new Roo.grid.ColumnModel( [
59781         {
59782             xtype: 'ColumnModel',
59783             xns: Roo.grid,
59784             dataIndex : 'weekday0',
59785             header : 'Sunday',
59786             renderer : cellRender
59787         },
59788         {
59789             xtype: 'ColumnModel',
59790             xns: Roo.grid,
59791             dataIndex : 'weekday1',
59792             header : 'Monday',
59793             renderer : cellRender
59794         },
59795         {
59796             xtype: 'ColumnModel',
59797             xns: Roo.grid,
59798             dataIndex : 'weekday2',
59799             header : 'Tuesday',
59800             renderer : cellRender
59801         },
59802         {
59803             xtype: 'ColumnModel',
59804             xns: Roo.grid,
59805             dataIndex : 'weekday3',
59806             header : 'Wednesday',
59807             renderer : cellRender
59808         },
59809         {
59810             xtype: 'ColumnModel',
59811             xns: Roo.grid,
59812             dataIndex : 'weekday4',
59813             header : 'Thursday',
59814             renderer : cellRender
59815         },
59816         {
59817             xtype: 'ColumnModel',
59818             xns: Roo.grid,
59819             dataIndex : 'weekday5',
59820             header : 'Friday',
59821             renderer : cellRender
59822         },
59823         {
59824             xtype: 'ColumnModel',
59825             xns: Roo.grid,
59826             dataIndex : 'weekday6',
59827             header : 'Saturday',
59828             renderer : cellRender
59829         }
59830     ]);
59831     this.cm = this.colModel;
59832     this.cm.xmodule = this.xmodule || false;
59833  
59834         
59835           
59836     //this.selModel = new Roo.grid.CellSelectionModel();
59837     //this.sm = this.selModel;
59838     //this.selModel.init(this);
59839     
59840     
59841     if(this.width){
59842         this.container.setWidth(this.width);
59843     }
59844
59845     if(this.height){
59846         this.container.setHeight(this.height);
59847     }
59848     /** @private */
59849         this.addEvents({
59850         // raw events
59851         /**
59852          * @event click
59853          * The raw click event for the entire grid.
59854          * @param {Roo.EventObject} e
59855          */
59856         "click" : true,
59857         /**
59858          * @event dblclick
59859          * The raw dblclick event for the entire grid.
59860          * @param {Roo.EventObject} e
59861          */
59862         "dblclick" : true,
59863         /**
59864          * @event contextmenu
59865          * The raw contextmenu event for the entire grid.
59866          * @param {Roo.EventObject} e
59867          */
59868         "contextmenu" : true,
59869         /**
59870          * @event mousedown
59871          * The raw mousedown event for the entire grid.
59872          * @param {Roo.EventObject} e
59873          */
59874         "mousedown" : true,
59875         /**
59876          * @event mouseup
59877          * The raw mouseup event for the entire grid.
59878          * @param {Roo.EventObject} e
59879          */
59880         "mouseup" : true,
59881         /**
59882          * @event mouseover
59883          * The raw mouseover event for the entire grid.
59884          * @param {Roo.EventObject} e
59885          */
59886         "mouseover" : true,
59887         /**
59888          * @event mouseout
59889          * The raw mouseout event for the entire grid.
59890          * @param {Roo.EventObject} e
59891          */
59892         "mouseout" : true,
59893         /**
59894          * @event keypress
59895          * The raw keypress event for the entire grid.
59896          * @param {Roo.EventObject} e
59897          */
59898         "keypress" : true,
59899         /**
59900          * @event keydown
59901          * The raw keydown event for the entire grid.
59902          * @param {Roo.EventObject} e
59903          */
59904         "keydown" : true,
59905
59906         // custom events
59907
59908         /**
59909          * @event cellclick
59910          * Fires when a cell is clicked
59911          * @param {Grid} this
59912          * @param {Number} rowIndex
59913          * @param {Number} columnIndex
59914          * @param {Roo.EventObject} e
59915          */
59916         "cellclick" : true,
59917         /**
59918          * @event celldblclick
59919          * Fires when a cell is double clicked
59920          * @param {Grid} this
59921          * @param {Number} rowIndex
59922          * @param {Number} columnIndex
59923          * @param {Roo.EventObject} e
59924          */
59925         "celldblclick" : true,
59926         /**
59927          * @event rowclick
59928          * Fires when a row is clicked
59929          * @param {Grid} this
59930          * @param {Number} rowIndex
59931          * @param {Roo.EventObject} e
59932          */
59933         "rowclick" : true,
59934         /**
59935          * @event rowdblclick
59936          * Fires when a row is double clicked
59937          * @param {Grid} this
59938          * @param {Number} rowIndex
59939          * @param {Roo.EventObject} e
59940          */
59941         "rowdblclick" : true,
59942         /**
59943          * @event headerclick
59944          * Fires when a header is clicked
59945          * @param {Grid} this
59946          * @param {Number} columnIndex
59947          * @param {Roo.EventObject} e
59948          */
59949         "headerclick" : true,
59950         /**
59951          * @event headerdblclick
59952          * Fires when a header cell is double clicked
59953          * @param {Grid} this
59954          * @param {Number} columnIndex
59955          * @param {Roo.EventObject} e
59956          */
59957         "headerdblclick" : true,
59958         /**
59959          * @event rowcontextmenu
59960          * Fires when a row is right clicked
59961          * @param {Grid} this
59962          * @param {Number} rowIndex
59963          * @param {Roo.EventObject} e
59964          */
59965         "rowcontextmenu" : true,
59966         /**
59967          * @event cellcontextmenu
59968          * Fires when a cell is right clicked
59969          * @param {Grid} this
59970          * @param {Number} rowIndex
59971          * @param {Number} cellIndex
59972          * @param {Roo.EventObject} e
59973          */
59974          "cellcontextmenu" : true,
59975         /**
59976          * @event headercontextmenu
59977          * Fires when a header is right clicked
59978          * @param {Grid} this
59979          * @param {Number} columnIndex
59980          * @param {Roo.EventObject} e
59981          */
59982         "headercontextmenu" : true,
59983         /**
59984          * @event bodyscroll
59985          * Fires when the body element is scrolled
59986          * @param {Number} scrollLeft
59987          * @param {Number} scrollTop
59988          */
59989         "bodyscroll" : true,
59990         /**
59991          * @event columnresize
59992          * Fires when the user resizes a column
59993          * @param {Number} columnIndex
59994          * @param {Number} newSize
59995          */
59996         "columnresize" : true,
59997         /**
59998          * @event columnmove
59999          * Fires when the user moves a column
60000          * @param {Number} oldIndex
60001          * @param {Number} newIndex
60002          */
60003         "columnmove" : true,
60004         /**
60005          * @event startdrag
60006          * Fires when row(s) start being dragged
60007          * @param {Grid} this
60008          * @param {Roo.GridDD} dd The drag drop object
60009          * @param {event} e The raw browser event
60010          */
60011         "startdrag" : true,
60012         /**
60013          * @event enddrag
60014          * Fires when a drag operation is complete
60015          * @param {Grid} this
60016          * @param {Roo.GridDD} dd The drag drop object
60017          * @param {event} e The raw browser event
60018          */
60019         "enddrag" : true,
60020         /**
60021          * @event dragdrop
60022          * Fires when dragged row(s) are dropped on a valid DD target
60023          * @param {Grid} this
60024          * @param {Roo.GridDD} dd The drag drop object
60025          * @param {String} targetId The target drag drop object
60026          * @param {event} e The raw browser event
60027          */
60028         "dragdrop" : true,
60029         /**
60030          * @event dragover
60031          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60032          * @param {Grid} this
60033          * @param {Roo.GridDD} dd The drag drop object
60034          * @param {String} targetId The target drag drop object
60035          * @param {event} e The raw browser event
60036          */
60037         "dragover" : true,
60038         /**
60039          * @event dragenter
60040          *  Fires when the dragged row(s) first cross another DD target while being dragged
60041          * @param {Grid} this
60042          * @param {Roo.GridDD} dd The drag drop object
60043          * @param {String} targetId The target drag drop object
60044          * @param {event} e The raw browser event
60045          */
60046         "dragenter" : true,
60047         /**
60048          * @event dragout
60049          * Fires when the dragged row(s) leave another DD target while being dragged
60050          * @param {Grid} this
60051          * @param {Roo.GridDD} dd The drag drop object
60052          * @param {String} targetId The target drag drop object
60053          * @param {event} e The raw browser event
60054          */
60055         "dragout" : true,
60056         /**
60057          * @event rowclass
60058          * Fires when a row is rendered, so you can change add a style to it.
60059          * @param {GridView} gridview   The grid view
60060          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60061          */
60062         'rowclass' : true,
60063
60064         /**
60065          * @event render
60066          * Fires when the grid is rendered
60067          * @param {Grid} grid
60068          */
60069         'render' : true,
60070             /**
60071              * @event select
60072              * Fires when a date is selected
60073              * @param {DatePicker} this
60074              * @param {Date} date The selected date
60075              */
60076         'select': true,
60077         /**
60078              * @event monthchange
60079              * Fires when the displayed month changes 
60080              * @param {DatePicker} this
60081              * @param {Date} date The selected month
60082              */
60083         'monthchange': true,
60084         /**
60085              * @event evententer
60086              * Fires when mouse over an event
60087              * @param {Calendar} this
60088              * @param {event} Event
60089              */
60090         'evententer': true,
60091         /**
60092              * @event eventleave
60093              * Fires when the mouse leaves an
60094              * @param {Calendar} this
60095              * @param {event}
60096              */
60097         'eventleave': true,
60098         /**
60099              * @event eventclick
60100              * Fires when the mouse click an
60101              * @param {Calendar} this
60102              * @param {event}
60103              */
60104         'eventclick': true,
60105         /**
60106              * @event eventrender
60107              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60108              * @param {Calendar} this
60109              * @param {data} data to be modified
60110              */
60111         'eventrender': true
60112         
60113     });
60114
60115     Roo.grid.Grid.superclass.constructor.call(this);
60116     this.on('render', function() {
60117         this.view.el.addClass('x-grid-cal'); 
60118         
60119         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60120
60121     },this);
60122     
60123     if (!Roo.grid.Calendar.style) {
60124         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60125             
60126             
60127             '.x-grid-cal .x-grid-col' :  {
60128                 height: 'auto !important',
60129                 'vertical-align': 'top'
60130             },
60131             '.x-grid-cal  .fc-event-hori' : {
60132                 height: '14px'
60133             }
60134              
60135             
60136         }, Roo.id());
60137     }
60138
60139     
60140     
60141 };
60142 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60143     /**
60144      * @cfg {Store} eventStore The store that loads events.
60145      */
60146     eventStore : 25,
60147
60148      
60149     activeDate : false,
60150     startDay : 0,
60151     autoWidth : true,
60152     monitorWindowResize : false,
60153
60154     
60155     resizeColumns : function() {
60156         var col = (this.view.el.getWidth() / 7) - 3;
60157         // loop through cols, and setWidth
60158         for(var i =0 ; i < 7 ; i++){
60159             this.cm.setColumnWidth(i, col);
60160         }
60161     },
60162      setDate :function(date) {
60163         
60164         Roo.log('setDate?');
60165         
60166         this.resizeColumns();
60167         var vd = this.activeDate;
60168         this.activeDate = date;
60169 //        if(vd && this.el){
60170 //            var t = date.getTime();
60171 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60172 //                Roo.log('using add remove');
60173 //                
60174 //                this.fireEvent('monthchange', this, date);
60175 //                
60176 //                this.cells.removeClass("fc-state-highlight");
60177 //                this.cells.each(function(c){
60178 //                   if(c.dateValue == t){
60179 //                       c.addClass("fc-state-highlight");
60180 //                       setTimeout(function(){
60181 //                            try{c.dom.firstChild.focus();}catch(e){}
60182 //                       }, 50);
60183 //                       return false;
60184 //                   }
60185 //                   return true;
60186 //                });
60187 //                return;
60188 //            }
60189 //        }
60190         
60191         var days = date.getDaysInMonth();
60192         
60193         var firstOfMonth = date.getFirstDateOfMonth();
60194         var startingPos = firstOfMonth.getDay()-this.startDay;
60195         
60196         if(startingPos < this.startDay){
60197             startingPos += 7;
60198         }
60199         
60200         var pm = date.add(Date.MONTH, -1);
60201         var prevStart = pm.getDaysInMonth()-startingPos;
60202 //        
60203         
60204         
60205         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60206         
60207         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60208         //this.cells.addClassOnOver('fc-state-hover');
60209         
60210         var cells = this.cells.elements;
60211         var textEls = this.textNodes;
60212         
60213         //Roo.each(cells, function(cell){
60214         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60215         //});
60216         
60217         days += startingPos;
60218
60219         // convert everything to numbers so it's fast
60220         var day = 86400000;
60221         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60222         //Roo.log(d);
60223         //Roo.log(pm);
60224         //Roo.log(prevStart);
60225         
60226         var today = new Date().clearTime().getTime();
60227         var sel = date.clearTime().getTime();
60228         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60229         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60230         var ddMatch = this.disabledDatesRE;
60231         var ddText = this.disabledDatesText;
60232         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60233         var ddaysText = this.disabledDaysText;
60234         var format = this.format;
60235         
60236         var setCellClass = function(cal, cell){
60237             
60238             //Roo.log('set Cell Class');
60239             cell.title = "";
60240             var t = d.getTime();
60241             
60242             //Roo.log(d);
60243             
60244             
60245             cell.dateValue = t;
60246             if(t == today){
60247                 cell.className += " fc-today";
60248                 cell.className += " fc-state-highlight";
60249                 cell.title = cal.todayText;
60250             }
60251             if(t == sel){
60252                 // disable highlight in other month..
60253                 cell.className += " fc-state-highlight";
60254                 
60255             }
60256             // disabling
60257             if(t < min) {
60258                 //cell.className = " fc-state-disabled";
60259                 cell.title = cal.minText;
60260                 return;
60261             }
60262             if(t > max) {
60263                 //cell.className = " fc-state-disabled";
60264                 cell.title = cal.maxText;
60265                 return;
60266             }
60267             if(ddays){
60268                 if(ddays.indexOf(d.getDay()) != -1){
60269                     // cell.title = ddaysText;
60270                    // cell.className = " fc-state-disabled";
60271                 }
60272             }
60273             if(ddMatch && format){
60274                 var fvalue = d.dateFormat(format);
60275                 if(ddMatch.test(fvalue)){
60276                     cell.title = ddText.replace("%0", fvalue);
60277                    cell.className = " fc-state-disabled";
60278                 }
60279             }
60280             
60281             if (!cell.initialClassName) {
60282                 cell.initialClassName = cell.dom.className;
60283             }
60284             
60285             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60286         };
60287
60288         var i = 0;
60289         
60290         for(; i < startingPos; i++) {
60291             cells[i].dayName =  (++prevStart);
60292             Roo.log(textEls[i]);
60293             d.setDate(d.getDate()+1);
60294             
60295             //cells[i].className = "fc-past fc-other-month";
60296             setCellClass(this, cells[i]);
60297         }
60298         
60299         var intDay = 0;
60300         
60301         for(; i < days; i++){
60302             intDay = i - startingPos + 1;
60303             cells[i].dayName =  (intDay);
60304             d.setDate(d.getDate()+1);
60305             
60306             cells[i].className = ''; // "x-date-active";
60307             setCellClass(this, cells[i]);
60308         }
60309         var extraDays = 0;
60310         
60311         for(; i < 42; i++) {
60312             //textEls[i].innerHTML = (++extraDays);
60313             
60314             d.setDate(d.getDate()+1);
60315             cells[i].dayName = (++extraDays);
60316             cells[i].className = "fc-future fc-other-month";
60317             setCellClass(this, cells[i]);
60318         }
60319         
60320         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60321         
60322         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60323         
60324         // this will cause all the cells to mis
60325         var rows= [];
60326         var i =0;
60327         for (var r = 0;r < 6;r++) {
60328             for (var c =0;c < 7;c++) {
60329                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60330             }    
60331         }
60332         
60333         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60334         for(i=0;i<cells.length;i++) {
60335             
60336             this.cells.elements[i].dayName = cells[i].dayName ;
60337             this.cells.elements[i].className = cells[i].className;
60338             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60339             this.cells.elements[i].title = cells[i].title ;
60340             this.cells.elements[i].dateValue = cells[i].dateValue ;
60341         }
60342         
60343         
60344         
60345         
60346         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60347         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60348         
60349         ////if(totalRows != 6){
60350             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60351            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60352        // }
60353         
60354         this.fireEvent('monthchange', this, date);
60355         
60356         
60357     },
60358  /**
60359      * Returns the grid's SelectionModel.
60360      * @return {SelectionModel}
60361      */
60362     getSelectionModel : function(){
60363         if(!this.selModel){
60364             this.selModel = new Roo.grid.CellSelectionModel();
60365         }
60366         return this.selModel;
60367     },
60368
60369     load: function() {
60370         this.eventStore.load()
60371         
60372         
60373         
60374     },
60375     
60376     findCell : function(dt) {
60377         dt = dt.clearTime().getTime();
60378         var ret = false;
60379         this.cells.each(function(c){
60380             //Roo.log("check " +c.dateValue + '?=' + dt);
60381             if(c.dateValue == dt){
60382                 ret = c;
60383                 return false;
60384             }
60385             return true;
60386         });
60387         
60388         return ret;
60389     },
60390     
60391     findCells : function(rec) {
60392         var s = rec.data.start_dt.clone().clearTime().getTime();
60393        // Roo.log(s);
60394         var e= rec.data.end_dt.clone().clearTime().getTime();
60395        // Roo.log(e);
60396         var ret = [];
60397         this.cells.each(function(c){
60398              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60399             
60400             if(c.dateValue > e){
60401                 return ;
60402             }
60403             if(c.dateValue < s){
60404                 return ;
60405             }
60406             ret.push(c);
60407         });
60408         
60409         return ret;    
60410     },
60411     
60412     findBestRow: function(cells)
60413     {
60414         var ret = 0;
60415         
60416         for (var i =0 ; i < cells.length;i++) {
60417             ret  = Math.max(cells[i].rows || 0,ret);
60418         }
60419         return ret;
60420         
60421     },
60422     
60423     
60424     addItem : function(rec)
60425     {
60426         // look for vertical location slot in
60427         var cells = this.findCells(rec);
60428         
60429         rec.row = this.findBestRow(cells);
60430         
60431         // work out the location.
60432         
60433         var crow = false;
60434         var rows = [];
60435         for(var i =0; i < cells.length; i++) {
60436             if (!crow) {
60437                 crow = {
60438                     start : cells[i],
60439                     end :  cells[i]
60440                 };
60441                 continue;
60442             }
60443             if (crow.start.getY() == cells[i].getY()) {
60444                 // on same row.
60445                 crow.end = cells[i];
60446                 continue;
60447             }
60448             // different row.
60449             rows.push(crow);
60450             crow = {
60451                 start: cells[i],
60452                 end : cells[i]
60453             };
60454             
60455         }
60456         
60457         rows.push(crow);
60458         rec.els = [];
60459         rec.rows = rows;
60460         rec.cells = cells;
60461         for (var i = 0; i < cells.length;i++) {
60462             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60463             
60464         }
60465         
60466         
60467     },
60468     
60469     clearEvents: function() {
60470         
60471         if (!this.eventStore.getCount()) {
60472             return;
60473         }
60474         // reset number of rows in cells.
60475         Roo.each(this.cells.elements, function(c){
60476             c.rows = 0;
60477         });
60478         
60479         this.eventStore.each(function(e) {
60480             this.clearEvent(e);
60481         },this);
60482         
60483     },
60484     
60485     clearEvent : function(ev)
60486     {
60487         if (ev.els) {
60488             Roo.each(ev.els, function(el) {
60489                 el.un('mouseenter' ,this.onEventEnter, this);
60490                 el.un('mouseleave' ,this.onEventLeave, this);
60491                 el.remove();
60492             },this);
60493             ev.els = [];
60494         }
60495     },
60496     
60497     
60498     renderEvent : function(ev,ctr) {
60499         if (!ctr) {
60500              ctr = this.view.el.select('.fc-event-container',true).first();
60501         }
60502         
60503          
60504         this.clearEvent(ev);
60505             //code
60506        
60507         
60508         
60509         ev.els = [];
60510         var cells = ev.cells;
60511         var rows = ev.rows;
60512         this.fireEvent('eventrender', this, ev);
60513         
60514         for(var i =0; i < rows.length; i++) {
60515             
60516             cls = '';
60517             if (i == 0) {
60518                 cls += ' fc-event-start';
60519             }
60520             if ((i+1) == rows.length) {
60521                 cls += ' fc-event-end';
60522             }
60523             
60524             //Roo.log(ev.data);
60525             // how many rows should it span..
60526             var cg = this.eventTmpl.append(ctr,Roo.apply({
60527                 fccls : cls
60528                 
60529             }, ev.data) , true);
60530             
60531             
60532             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60533             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60534             cg.on('click', this.onEventClick, this, ev);
60535             
60536             ev.els.push(cg);
60537             
60538             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60539             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60540             //Roo.log(cg);
60541              
60542             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60543             cg.setWidth(ebox.right - sbox.x -2);
60544         }
60545     },
60546     
60547     renderEvents: function()
60548     {   
60549         // first make sure there is enough space..
60550         
60551         if (!this.eventTmpl) {
60552             this.eventTmpl = new Roo.Template(
60553                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60554                     '<div class="fc-event-inner">' +
60555                         '<span class="fc-event-time">{time}</span>' +
60556                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60557                     '</div>' +
60558                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60559                 '</div>'
60560             );
60561                 
60562         }
60563                
60564         
60565         
60566         this.cells.each(function(c) {
60567             //Roo.log(c.select('.fc-day-content div',true).first());
60568             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60569         });
60570         
60571         var ctr = this.view.el.select('.fc-event-container',true).first();
60572         
60573         var cls;
60574         this.eventStore.each(function(ev){
60575             
60576             this.renderEvent(ev);
60577              
60578              
60579         }, this);
60580         this.view.layout();
60581         
60582     },
60583     
60584     onEventEnter: function (e, el,event,d) {
60585         this.fireEvent('evententer', this, el, event);
60586     },
60587     
60588     onEventLeave: function (e, el,event,d) {
60589         this.fireEvent('eventleave', this, el, event);
60590     },
60591     
60592     onEventClick: function (e, el,event,d) {
60593         this.fireEvent('eventclick', this, el, event);
60594     },
60595     
60596     onMonthChange: function () {
60597         this.store.load();
60598     },
60599     
60600     onLoad: function () {
60601         
60602         //Roo.log('calendar onload');
60603 //         
60604         if(this.eventStore.getCount() > 0){
60605             
60606            
60607             
60608             this.eventStore.each(function(d){
60609                 
60610                 
60611                 // FIXME..
60612                 var add =   d.data;
60613                 if (typeof(add.end_dt) == 'undefined')  {
60614                     Roo.log("Missing End time in calendar data: ");
60615                     Roo.log(d);
60616                     return;
60617                 }
60618                 if (typeof(add.start_dt) == 'undefined')  {
60619                     Roo.log("Missing Start time in calendar data: ");
60620                     Roo.log(d);
60621                     return;
60622                 }
60623                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60624                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60625                 add.id = add.id || d.id;
60626                 add.title = add.title || '??';
60627                 
60628                 this.addItem(d);
60629                 
60630              
60631             },this);
60632         }
60633         
60634         this.renderEvents();
60635     }
60636     
60637
60638 });
60639 /*
60640  grid : {
60641                 xtype: 'Grid',
60642                 xns: Roo.grid,
60643                 listeners : {
60644                     render : function ()
60645                     {
60646                         _this.grid = this;
60647                         
60648                         if (!this.view.el.hasClass('course-timesheet')) {
60649                             this.view.el.addClass('course-timesheet');
60650                         }
60651                         if (this.tsStyle) {
60652                             this.ds.load({});
60653                             return; 
60654                         }
60655                         Roo.log('width');
60656                         Roo.log(_this.grid.view.el.getWidth());
60657                         
60658                         
60659                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60660                             '.course-timesheet .x-grid-row' : {
60661                                 height: '80px'
60662                             },
60663                             '.x-grid-row td' : {
60664                                 'vertical-align' : 0
60665                             },
60666                             '.course-edit-link' : {
60667                                 'color' : 'blue',
60668                                 'text-overflow' : 'ellipsis',
60669                                 'overflow' : 'hidden',
60670                                 'white-space' : 'nowrap',
60671                                 'cursor' : 'pointer'
60672                             },
60673                             '.sub-link' : {
60674                                 'color' : 'green'
60675                             },
60676                             '.de-act-sup-link' : {
60677                                 'color' : 'purple',
60678                                 'text-decoration' : 'line-through'
60679                             },
60680                             '.de-act-link' : {
60681                                 'color' : 'red',
60682                                 'text-decoration' : 'line-through'
60683                             },
60684                             '.course-timesheet .course-highlight' : {
60685                                 'border-top-style': 'dashed !important',
60686                                 'border-bottom-bottom': 'dashed !important'
60687                             },
60688                             '.course-timesheet .course-item' : {
60689                                 'font-family'   : 'tahoma, arial, helvetica',
60690                                 'font-size'     : '11px',
60691                                 'overflow'      : 'hidden',
60692                                 'padding-left'  : '10px',
60693                                 'padding-right' : '10px',
60694                                 'padding-top' : '10px' 
60695                             }
60696                             
60697                         }, Roo.id());
60698                                 this.ds.load({});
60699                     }
60700                 },
60701                 autoWidth : true,
60702                 monitorWindowResize : false,
60703                 cellrenderer : function(v,x,r)
60704                 {
60705                     return v;
60706                 },
60707                 sm : {
60708                     xtype: 'CellSelectionModel',
60709                     xns: Roo.grid
60710                 },
60711                 dataSource : {
60712                     xtype: 'Store',
60713                     xns: Roo.data,
60714                     listeners : {
60715                         beforeload : function (_self, options)
60716                         {
60717                             options.params = options.params || {};
60718                             options.params._month = _this.monthField.getValue();
60719                             options.params.limit = 9999;
60720                             options.params['sort'] = 'when_dt';    
60721                             options.params['dir'] = 'ASC';    
60722                             this.proxy.loadResponse = this.loadResponse;
60723                             Roo.log("load?");
60724                             //this.addColumns();
60725                         },
60726                         load : function (_self, records, options)
60727                         {
60728                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60729                                 // if you click on the translation.. you can edit it...
60730                                 var el = Roo.get(this);
60731                                 var id = el.dom.getAttribute('data-id');
60732                                 var d = el.dom.getAttribute('data-date');
60733                                 var t = el.dom.getAttribute('data-time');
60734                                 //var id = this.child('span').dom.textContent;
60735                                 
60736                                 //Roo.log(this);
60737                                 Pman.Dialog.CourseCalendar.show({
60738                                     id : id,
60739                                     when_d : d,
60740                                     when_t : t,
60741                                     productitem_active : id ? 1 : 0
60742                                 }, function() {
60743                                     _this.grid.ds.load({});
60744                                 });
60745                            
60746                            });
60747                            
60748                            _this.panel.fireEvent('resize', [ '', '' ]);
60749                         }
60750                     },
60751                     loadResponse : function(o, success, response){
60752                             // this is overridden on before load..
60753                             
60754                             Roo.log("our code?");       
60755                             //Roo.log(success);
60756                             //Roo.log(response)
60757                             delete this.activeRequest;
60758                             if(!success){
60759                                 this.fireEvent("loadexception", this, o, response);
60760                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60761                                 return;
60762                             }
60763                             var result;
60764                             try {
60765                                 result = o.reader.read(response);
60766                             }catch(e){
60767                                 Roo.log("load exception?");
60768                                 this.fireEvent("loadexception", this, o, response, e);
60769                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60770                                 return;
60771                             }
60772                             Roo.log("ready...");        
60773                             // loop through result.records;
60774                             // and set this.tdate[date] = [] << array of records..
60775                             _this.tdata  = {};
60776                             Roo.each(result.records, function(r){
60777                                 //Roo.log(r.data);
60778                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60779                                     _this.tdata[r.data.when_dt.format('j')] = [];
60780                                 }
60781                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60782                             });
60783                             
60784                             //Roo.log(_this.tdata);
60785                             
60786                             result.records = [];
60787                             result.totalRecords = 6;
60788                     
60789                             // let's generate some duumy records for the rows.
60790                             //var st = _this.dateField.getValue();
60791                             
60792                             // work out monday..
60793                             //st = st.add(Date.DAY, -1 * st.format('w'));
60794                             
60795                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60796                             
60797                             var firstOfMonth = date.getFirstDayOfMonth();
60798                             var days = date.getDaysInMonth();
60799                             var d = 1;
60800                             var firstAdded = false;
60801                             for (var i = 0; i < result.totalRecords ; i++) {
60802                                 //var d= st.add(Date.DAY, i);
60803                                 var row = {};
60804                                 var added = 0;
60805                                 for(var w = 0 ; w < 7 ; w++){
60806                                     if(!firstAdded && firstOfMonth != w){
60807                                         continue;
60808                                     }
60809                                     if(d > days){
60810                                         continue;
60811                                     }
60812                                     firstAdded = true;
60813                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60814                                     row['weekday'+w] = String.format(
60815                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60816                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60817                                                     d,
60818                                                     date.format('Y-m-')+dd
60819                                                 );
60820                                     added++;
60821                                     if(typeof(_this.tdata[d]) != 'undefined'){
60822                                         Roo.each(_this.tdata[d], function(r){
60823                                             var is_sub = '';
60824                                             var deactive = '';
60825                                             var id = r.id;
60826                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60827                                             if(r.parent_id*1>0){
60828                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60829                                                 id = r.parent_id;
60830                                             }
60831                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60832                                                 deactive = 'de-act-link';
60833                                             }
60834                                             
60835                                             row['weekday'+w] += String.format(
60836                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60837                                                     id, //0
60838                                                     r.product_id_name, //1
60839                                                     r.when_dt.format('h:ia'), //2
60840                                                     is_sub, //3
60841                                                     deactive, //4
60842                                                     desc // 5
60843                                             );
60844                                         });
60845                                     }
60846                                     d++;
60847                                 }
60848                                 
60849                                 // only do this if something added..
60850                                 if(added > 0){ 
60851                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60852                                 }
60853                                 
60854                                 
60855                                 // push it twice. (second one with an hour..
60856                                 
60857                             }
60858                             //Roo.log(result);
60859                             this.fireEvent("load", this, o, o.request.arg);
60860                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60861                         },
60862                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60863                     proxy : {
60864                         xtype: 'HttpProxy',
60865                         xns: Roo.data,
60866                         method : 'GET',
60867                         url : baseURL + '/Roo/Shop_course.php'
60868                     },
60869                     reader : {
60870                         xtype: 'JsonReader',
60871                         xns: Roo.data,
60872                         id : 'id',
60873                         fields : [
60874                             {
60875                                 'name': 'id',
60876                                 'type': 'int'
60877                             },
60878                             {
60879                                 'name': 'when_dt',
60880                                 'type': 'string'
60881                             },
60882                             {
60883                                 'name': 'end_dt',
60884                                 'type': 'string'
60885                             },
60886                             {
60887                                 'name': 'parent_id',
60888                                 'type': 'int'
60889                             },
60890                             {
60891                                 'name': 'product_id',
60892                                 'type': 'int'
60893                             },
60894                             {
60895                                 'name': 'productitem_id',
60896                                 'type': 'int'
60897                             },
60898                             {
60899                                 'name': 'guid',
60900                                 'type': 'int'
60901                             }
60902                         ]
60903                     }
60904                 },
60905                 toolbar : {
60906                     xtype: 'Toolbar',
60907                     xns: Roo,
60908                     items : [
60909                         {
60910                             xtype: 'Button',
60911                             xns: Roo.Toolbar,
60912                             listeners : {
60913                                 click : function (_self, e)
60914                                 {
60915                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60916                                     sd.setMonth(sd.getMonth()-1);
60917                                     _this.monthField.setValue(sd.format('Y-m-d'));
60918                                     _this.grid.ds.load({});
60919                                 }
60920                             },
60921                             text : "Back"
60922                         },
60923                         {
60924                             xtype: 'Separator',
60925                             xns: Roo.Toolbar
60926                         },
60927                         {
60928                             xtype: 'MonthField',
60929                             xns: Roo.form,
60930                             listeners : {
60931                                 render : function (_self)
60932                                 {
60933                                     _this.monthField = _self;
60934                                    // _this.monthField.set  today
60935                                 },
60936                                 select : function (combo, date)
60937                                 {
60938                                     _this.grid.ds.load({});
60939                                 }
60940                             },
60941                             value : (function() { return new Date(); })()
60942                         },
60943                         {
60944                             xtype: 'Separator',
60945                             xns: Roo.Toolbar
60946                         },
60947                         {
60948                             xtype: 'TextItem',
60949                             xns: Roo.Toolbar,
60950                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
60951                         },
60952                         {
60953                             xtype: 'Fill',
60954                             xns: Roo.Toolbar
60955                         },
60956                         {
60957                             xtype: 'Button',
60958                             xns: Roo.Toolbar,
60959                             listeners : {
60960                                 click : function (_self, e)
60961                                 {
60962                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60963                                     sd.setMonth(sd.getMonth()+1);
60964                                     _this.monthField.setValue(sd.format('Y-m-d'));
60965                                     _this.grid.ds.load({});
60966                                 }
60967                             },
60968                             text : "Next"
60969                         }
60970                     ]
60971                 },
60972                  
60973             }
60974         };
60975         
60976         *//*
60977  * Based on:
60978  * Ext JS Library 1.1.1
60979  * Copyright(c) 2006-2007, Ext JS, LLC.
60980  *
60981  * Originally Released Under LGPL - original licence link has changed is not relivant.
60982  *
60983  * Fork - LGPL
60984  * <script type="text/javascript">
60985  */
60986  
60987 /**
60988  * @class Roo.LoadMask
60989  * A simple utility class for generically masking elements while loading data.  If the element being masked has
60990  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
60991  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
60992  * element's UpdateManager load indicator and will be destroyed after the initial load.
60993  * @constructor
60994  * Create a new LoadMask
60995  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
60996  * @param {Object} config The config object
60997  */
60998 Roo.LoadMask = function(el, config){
60999     this.el = Roo.get(el);
61000     Roo.apply(this, config);
61001     if(this.store){
61002         this.store.on('beforeload', this.onBeforeLoad, this);
61003         this.store.on('load', this.onLoad, this);
61004         this.store.on('loadexception', this.onLoadException, this);
61005         this.removeMask = false;
61006     }else{
61007         var um = this.el.getUpdateManager();
61008         um.showLoadIndicator = false; // disable the default indicator
61009         um.on('beforeupdate', this.onBeforeLoad, this);
61010         um.on('update', this.onLoad, this);
61011         um.on('failure', this.onLoad, this);
61012         this.removeMask = true;
61013     }
61014 };
61015
61016 Roo.LoadMask.prototype = {
61017     /**
61018      * @cfg {Boolean} removeMask
61019      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61020      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61021      */
61022     /**
61023      * @cfg {String} msg
61024      * The text to display in a centered loading message box (defaults to 'Loading...')
61025      */
61026     msg : 'Loading...',
61027     /**
61028      * @cfg {String} msgCls
61029      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61030      */
61031     msgCls : 'x-mask-loading',
61032
61033     /**
61034      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61035      * @type Boolean
61036      */
61037     disabled: false,
61038
61039     /**
61040      * Disables the mask to prevent it from being displayed
61041      */
61042     disable : function(){
61043        this.disabled = true;
61044     },
61045
61046     /**
61047      * Enables the mask so that it can be displayed
61048      */
61049     enable : function(){
61050         this.disabled = false;
61051     },
61052     
61053     onLoadException : function()
61054     {
61055         Roo.log(arguments);
61056         
61057         if (typeof(arguments[3]) != 'undefined') {
61058             Roo.MessageBox.alert("Error loading",arguments[3]);
61059         } 
61060         /*
61061         try {
61062             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61063                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61064             }   
61065         } catch(e) {
61066             
61067         }
61068         */
61069     
61070         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61071     },
61072     // private
61073     onLoad : function()
61074     {
61075         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61076     },
61077
61078     // private
61079     onBeforeLoad : function(){
61080         if(!this.disabled){
61081             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61082         }
61083     },
61084
61085     // private
61086     destroy : function(){
61087         if(this.store){
61088             this.store.un('beforeload', this.onBeforeLoad, this);
61089             this.store.un('load', this.onLoad, this);
61090             this.store.un('loadexception', this.onLoadException, this);
61091         }else{
61092             var um = this.el.getUpdateManager();
61093             um.un('beforeupdate', this.onBeforeLoad, this);
61094             um.un('update', this.onLoad, this);
61095             um.un('failure', this.onLoad, this);
61096         }
61097     }
61098 };/*
61099  * Based on:
61100  * Ext JS Library 1.1.1
61101  * Copyright(c) 2006-2007, Ext JS, LLC.
61102  *
61103  * Originally Released Under LGPL - original licence link has changed is not relivant.
61104  *
61105  * Fork - LGPL
61106  * <script type="text/javascript">
61107  */
61108
61109
61110 /**
61111  * @class Roo.XTemplate
61112  * @extends Roo.Template
61113  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61114 <pre><code>
61115 var t = new Roo.XTemplate(
61116         '&lt;select name="{name}"&gt;',
61117                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61118         '&lt;/select&gt;'
61119 );
61120  
61121 // then append, applying the master template values
61122  </code></pre>
61123  *
61124  * Supported features:
61125  *
61126  *  Tags:
61127
61128 <pre><code>
61129       {a_variable} - output encoded.
61130       {a_variable.format:("Y-m-d")} - call a method on the variable
61131       {a_variable:raw} - unencoded output
61132       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61133       {a_variable:this.method_on_template(...)} - call a method on the template object.
61134  
61135 </code></pre>
61136  *  The tpl tag:
61137 <pre><code>
61138         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61139         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61140         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61141         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61142   
61143         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61144         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61145 </code></pre>
61146  *      
61147  */
61148 Roo.XTemplate = function()
61149 {
61150     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61151     if (this.html) {
61152         this.compile();
61153     }
61154 };
61155
61156
61157 Roo.extend(Roo.XTemplate, Roo.Template, {
61158
61159     /**
61160      * The various sub templates
61161      */
61162     tpls : false,
61163     /**
61164      *
61165      * basic tag replacing syntax
61166      * WORD:WORD()
61167      *
61168      * // you can fake an object call by doing this
61169      *  x.t:(test,tesT) 
61170      * 
61171      */
61172     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61173
61174     /**
61175      * compile the template
61176      *
61177      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61178      *
61179      */
61180     compile: function()
61181     {
61182         var s = this.html;
61183      
61184         s = ['<tpl>', s, '</tpl>'].join('');
61185     
61186         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61187             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61188             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61189             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61190             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61191             m,
61192             id     = 0,
61193             tpls   = [];
61194     
61195         while(true == !!(m = s.match(re))){
61196             var forMatch   = m[0].match(nameRe),
61197                 ifMatch   = m[0].match(ifRe),
61198                 execMatch   = m[0].match(execRe),
61199                 namedMatch   = m[0].match(namedRe),
61200                 
61201                 exp  = null, 
61202                 fn   = null,
61203                 exec = null,
61204                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61205                 
61206             if (ifMatch) {
61207                 // if - puts fn into test..
61208                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61209                 if(exp){
61210                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61211                 }
61212             }
61213             
61214             if (execMatch) {
61215                 // exec - calls a function... returns empty if true is  returned.
61216                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61217                 if(exp){
61218                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61219                 }
61220             }
61221             
61222             
61223             if (name) {
61224                 // for = 
61225                 switch(name){
61226                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61227                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61228                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61229                 }
61230             }
61231             var uid = namedMatch ? namedMatch[1] : id;
61232             
61233             
61234             tpls.push({
61235                 id:     namedMatch ? namedMatch[1] : id,
61236                 target: name,
61237                 exec:   exec,
61238                 test:   fn,
61239                 body:   m[1] || ''
61240             });
61241             if (namedMatch) {
61242                 s = s.replace(m[0], '');
61243             } else { 
61244                 s = s.replace(m[0], '{xtpl'+ id + '}');
61245             }
61246             ++id;
61247         }
61248         this.tpls = [];
61249         for(var i = tpls.length-1; i >= 0; --i){
61250             this.compileTpl(tpls[i]);
61251             this.tpls[tpls[i].id] = tpls[i];
61252         }
61253         this.master = tpls[tpls.length-1];
61254         return this;
61255     },
61256     /**
61257      * same as applyTemplate, except it's done to one of the subTemplates
61258      * when using named templates, you can do:
61259      *
61260      * var str = pl.applySubTemplate('your-name', values);
61261      *
61262      * 
61263      * @param {Number} id of the template
61264      * @param {Object} values to apply to template
61265      * @param {Object} parent (normaly the instance of this object)
61266      */
61267     applySubTemplate : function(id, values, parent)
61268     {
61269         
61270         
61271         var t = this.tpls[id];
61272         
61273         
61274         try { 
61275             if(t.test && !t.test.call(this, values, parent)){
61276                 return '';
61277             }
61278         } catch(e) {
61279             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61280             Roo.log(e.toString());
61281             Roo.log(t.test);
61282             return ''
61283         }
61284         try { 
61285             
61286             if(t.exec && t.exec.call(this, values, parent)){
61287                 return '';
61288             }
61289         } catch(e) {
61290             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61291             Roo.log(e.toString());
61292             Roo.log(t.exec);
61293             return ''
61294         }
61295         try {
61296             var vs = t.target ? t.target.call(this, values, parent) : values;
61297             parent = t.target ? values : parent;
61298             if(t.target && vs instanceof Array){
61299                 var buf = [];
61300                 for(var i = 0, len = vs.length; i < len; i++){
61301                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61302                 }
61303                 return buf.join('');
61304             }
61305             return t.compiled.call(this, vs, parent);
61306         } catch (e) {
61307             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61308             Roo.log(e.toString());
61309             Roo.log(t.compiled);
61310             return '';
61311         }
61312     },
61313
61314     compileTpl : function(tpl)
61315     {
61316         var fm = Roo.util.Format;
61317         var useF = this.disableFormats !== true;
61318         var sep = Roo.isGecko ? "+" : ",";
61319         var undef = function(str) {
61320             Roo.log("Property not found :"  + str);
61321             return '';
61322         };
61323         
61324         var fn = function(m, name, format, args)
61325         {
61326             //Roo.log(arguments);
61327             args = args ? args.replace(/\\'/g,"'") : args;
61328             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61329             if (typeof(format) == 'undefined') {
61330                 format= 'htmlEncode';
61331             }
61332             if (format == 'raw' ) {
61333                 format = false;
61334             }
61335             
61336             if(name.substr(0, 4) == 'xtpl'){
61337                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61338             }
61339             
61340             // build an array of options to determine if value is undefined..
61341             
61342             // basically get 'xxxx.yyyy' then do
61343             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61344             //    (function () { Roo.log("Property not found"); return ''; })() :
61345             //    ......
61346             
61347             var udef_ar = [];
61348             var lookfor = '';
61349             Roo.each(name.split('.'), function(st) {
61350                 lookfor += (lookfor.length ? '.': '') + st;
61351                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61352             });
61353             
61354             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61355             
61356             
61357             if(format && useF){
61358                 
61359                 args = args ? ',' + args : "";
61360                  
61361                 if(format.substr(0, 5) != "this."){
61362                     format = "fm." + format + '(';
61363                 }else{
61364                     format = 'this.call("'+ format.substr(5) + '", ';
61365                     args = ", values";
61366                 }
61367                 
61368                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61369             }
61370              
61371             if (args.length) {
61372                 // called with xxyx.yuu:(test,test)
61373                 // change to ()
61374                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61375             }
61376             // raw.. - :raw modifier..
61377             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61378             
61379         };
61380         var body;
61381         // branched to use + in gecko and [].join() in others
61382         if(Roo.isGecko){
61383             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61384                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61385                     "';};};";
61386         }else{
61387             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61388             body.push(tpl.body.replace(/(\r\n|\n)/g,
61389                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61390             body.push("'].join('');};};");
61391             body = body.join('');
61392         }
61393         
61394         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61395        
61396         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61397         eval(body);
61398         
61399         return this;
61400     },
61401
61402     applyTemplate : function(values){
61403         return this.master.compiled.call(this, values, {});
61404         //var s = this.subs;
61405     },
61406
61407     apply : function(){
61408         return this.applyTemplate.apply(this, arguments);
61409     }
61410
61411  });
61412
61413 Roo.XTemplate.from = function(el){
61414     el = Roo.getDom(el);
61415     return new Roo.XTemplate(el.value || el.innerHTML);
61416 };