db52ac5e132603786f727a1fcaa48192f884a956
[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          /**
16855          * eval:var:noop
16856     */
16857     var noop = function () {}
16858     noop.exec = noop;
16859     
16860          /**
16861          * eval:var:merge
16862     */
16863     var merge = function (obj) {
16864       var i = 1
16865         , target
16866         , key;
16867     
16868       for (; i < arguments.length; i++) {
16869         target = arguments[i];
16870         for (key in target) {
16871           if (Object.prototype.hasOwnProperty.call(target, key)) {
16872             obj[key] = target[key];
16873           }
16874         }
16875       }
16876     
16877       return obj;
16878     }
16879     
16880     
16881     /**
16882      * Block-Level Grammar
16883      */
16884     
16885     
16886     
16887     
16888     var block = {
16889       newline: /^\n+/,
16890       code: /^( {4}[^\n]+\n*)+/,
16891       fences: noop,
16892       hr: /^( *[-*_]){3,} *(?:\n+|$)/,
16893       heading: /^ *(#{1,6}) *([^\n]+?) *#* *(?:\n+|$)/,
16894       nptable: noop,
16895       lheading: /^([^\n]+)\n *(=|-){2,} *(?:\n+|$)/,
16896       blockquote: /^( *>[^\n]+(\n(?!def)[^\n]+)*\n*)+/,
16897       list: /^( *)(bull) [\s\S]+?(?:hr|def|\n{2,}(?! )(?!\1bull )\n*|\s*$)/,
16898       html: /^ *(?:comment *(?:\n|\s*$)|closed *(?:\n{2,}|\s*$)|closing *(?:\n{2,}|\s*$))/,
16899       def: /^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +["(]([^\n]+)[")])? *(?:\n+|$)/,
16900       table: noop,
16901       paragraph: /^((?:[^\n]+\n?(?!hr|heading|lheading|blockquote|tag|def))+)\n*/,
16902       text: /^[^\n]+/
16903     };
16904     
16905     block.bullet = /(?:[*+-]|\d+\.)/;
16906     block.item = /^( *)(bull) [^\n]*(?:\n(?!\1bull )[^\n]*)*/;
16907     block.item = replace(block.item, 'gm')
16908       (/bull/g, block.bullet)
16909       ();
16910     
16911     block.list = replace(block.list)
16912       (/bull/g, block.bullet)
16913       ('hr', '\\n+(?=\\1?(?:[-*_] *){3,}(?:\\n+|$))')
16914       ('def', '\\n+(?=' + block.def.source + ')')
16915       ();
16916     
16917     block.blockquote = replace(block.blockquote)
16918       ('def', block.def)
16919       ();
16920     
16921     block._tag = '(?!(?:'
16922       + 'a|em|strong|small|s|cite|q|dfn|abbr|data|time|code'
16923       + '|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo'
16924       + '|span|br|wbr|ins|del|img)\\b)\\w+(?!:/|[^\\w\\s@]*@)\\b';
16925     
16926     block.html = replace(block.html)
16927       ('comment', /<!--[\s\S]*?-->/)
16928       ('closed', /<(tag)[\s\S]+?<\/\1>/)
16929       ('closing', /<tag(?:"[^"]*"|'[^']*'|[^'">])*?>/)
16930       (/tag/g, block._tag)
16931       ();
16932     
16933     block.paragraph = replace(block.paragraph)
16934       ('hr', block.hr)
16935       ('heading', block.heading)
16936       ('lheading', block.lheading)
16937       ('blockquote', block.blockquote)
16938       ('tag', '<' + block._tag)
16939       ('def', block.def)
16940       ();
16941     
16942     /**
16943      * Normal Block Grammar
16944      */
16945     
16946     block.normal = merge({}, block);
16947     
16948     /**
16949      * GFM Block Grammar
16950      */
16951     
16952     block.gfm = merge({}, block.normal, {
16953       fences: /^ *(`{3,}|~{3,})[ \.]*(\S+)? *\n([\s\S]*?)\s*\1 *(?:\n+|$)/,
16954       paragraph: /^/,
16955       heading: /^ *(#{1,6}) +([^\n]+?) *#* *(?:\n+|$)/
16956     });
16957     
16958     block.gfm.paragraph = replace(block.paragraph)
16959       ('(?!', '(?!'
16960         + block.gfm.fences.source.replace('\\1', '\\2') + '|'
16961         + block.list.source.replace('\\1', '\\3') + '|')
16962       ();
16963     
16964     /**
16965      * GFM + Tables Block Grammar
16966      */
16967     
16968     block.tables = merge({}, block.gfm, {
16969       nptable: /^ *(\S.*\|.*)\n *([-:]+ *\|[-| :]*)\n((?:.*\|.*(?:\n|$))*)\n*/,
16970       table: /^ *\|(.+)\n *\|( *[-:]+[-| :]*)\n((?: *\|.*(?:\n|$))*)\n*/
16971     });
16972     
16973     /**
16974      * Block Lexer
16975      */
16976     
16977     var Lexer = function (options) {
16978       this.tokens = [];
16979       this.tokens.links = {};
16980       this.options = options || marked.defaults;
16981       this.rules = block.normal;
16982     
16983       if (this.options.gfm) {
16984         if (this.options.tables) {
16985           this.rules = block.tables;
16986         } else {
16987           this.rules = block.gfm;
16988         }
16989       }
16990     }
16991     
16992     /**
16993      * Expose Block Rules
16994      */
16995     
16996     Lexer.rules = block;
16997     
16998     /**
16999      * Static Lex Method
17000      */
17001     
17002     Lexer.lex = function(src, options) {
17003       var lexer = new Lexer(options);
17004       return lexer.lex(src);
17005     };
17006     
17007     /**
17008      * Preprocessing
17009      */
17010     
17011     Lexer.prototype.lex = function(src) {
17012       src = src
17013         .replace(/\r\n|\r/g, '\n')
17014         .replace(/\t/g, '    ')
17015         .replace(/\u00a0/g, ' ')
17016         .replace(/\u2424/g, '\n');
17017     
17018       return this.token(src, true);
17019     };
17020     
17021     /**
17022      * Lexing
17023      */
17024     
17025     Lexer.prototype.token = function(src, top, bq) {
17026       var src = src.replace(/^ +$/gm, '')
17027         , next
17028         , loose
17029         , cap
17030         , bull
17031         , b
17032         , item
17033         , space
17034         , i
17035         , l;
17036     
17037       while (src) {
17038         // newline
17039         if (cap = this.rules.newline.exec(src)) {
17040           src = src.substring(cap[0].length);
17041           if (cap[0].length > 1) {
17042             this.tokens.push({
17043               type: 'space'
17044             });
17045           }
17046         }
17047     
17048         // code
17049         if (cap = this.rules.code.exec(src)) {
17050           src = src.substring(cap[0].length);
17051           cap = cap[0].replace(/^ {4}/gm, '');
17052           this.tokens.push({
17053             type: 'code',
17054             text: !this.options.pedantic
17055               ? cap.replace(/\n+$/, '')
17056               : cap
17057           });
17058           continue;
17059         }
17060     
17061         // fences (gfm)
17062         if (cap = this.rules.fences.exec(src)) {
17063           src = src.substring(cap[0].length);
17064           this.tokens.push({
17065             type: 'code',
17066             lang: cap[2],
17067             text: cap[3] || ''
17068           });
17069           continue;
17070         }
17071     
17072         // heading
17073         if (cap = this.rules.heading.exec(src)) {
17074           src = src.substring(cap[0].length);
17075           this.tokens.push({
17076             type: 'heading',
17077             depth: cap[1].length,
17078             text: cap[2]
17079           });
17080           continue;
17081         }
17082     
17083         // table no leading pipe (gfm)
17084         if (top && (cap = this.rules.nptable.exec(src))) {
17085           src = src.substring(cap[0].length);
17086     
17087           item = {
17088             type: 'table',
17089             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17090             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17091             cells: cap[3].replace(/\n$/, '').split('\n')
17092           };
17093     
17094           for (i = 0; i < item.align.length; i++) {
17095             if (/^ *-+: *$/.test(item.align[i])) {
17096               item.align[i] = 'right';
17097             } else if (/^ *:-+: *$/.test(item.align[i])) {
17098               item.align[i] = 'center';
17099             } else if (/^ *:-+ *$/.test(item.align[i])) {
17100               item.align[i] = 'left';
17101             } else {
17102               item.align[i] = null;
17103             }
17104           }
17105     
17106           for (i = 0; i < item.cells.length; i++) {
17107             item.cells[i] = item.cells[i].split(/ *\| */);
17108           }
17109     
17110           this.tokens.push(item);
17111     
17112           continue;
17113         }
17114     
17115         // lheading
17116         if (cap = this.rules.lheading.exec(src)) {
17117           src = src.substring(cap[0].length);
17118           this.tokens.push({
17119             type: 'heading',
17120             depth: cap[2] === '=' ? 1 : 2,
17121             text: cap[1]
17122           });
17123           continue;
17124         }
17125     
17126         // hr
17127         if (cap = this.rules.hr.exec(src)) {
17128           src = src.substring(cap[0].length);
17129           this.tokens.push({
17130             type: 'hr'
17131           });
17132           continue;
17133         }
17134     
17135         // blockquote
17136         if (cap = this.rules.blockquote.exec(src)) {
17137           src = src.substring(cap[0].length);
17138     
17139           this.tokens.push({
17140             type: 'blockquote_start'
17141           });
17142     
17143           cap = cap[0].replace(/^ *> ?/gm, '');
17144     
17145           // Pass `top` to keep the current
17146           // "toplevel" state. This is exactly
17147           // how markdown.pl works.
17148           this.token(cap, top, true);
17149     
17150           this.tokens.push({
17151             type: 'blockquote_end'
17152           });
17153     
17154           continue;
17155         }
17156     
17157         // list
17158         if (cap = this.rules.list.exec(src)) {
17159           src = src.substring(cap[0].length);
17160           bull = cap[2];
17161     
17162           this.tokens.push({
17163             type: 'list_start',
17164             ordered: bull.length > 1
17165           });
17166     
17167           // Get each top-level item.
17168           cap = cap[0].match(this.rules.item);
17169     
17170           next = false;
17171           l = cap.length;
17172           i = 0;
17173     
17174           for (; i < l; i++) {
17175             item = cap[i];
17176     
17177             // Remove the list item's bullet
17178             // so it is seen as the next token.
17179             space = item.length;
17180             item = item.replace(/^ *([*+-]|\d+\.) +/, '');
17181     
17182             // Outdent whatever the
17183             // list item contains. Hacky.
17184             if (~item.indexOf('\n ')) {
17185               space -= item.length;
17186               item = !this.options.pedantic
17187                 ? item.replace(new RegExp('^ {1,' + space + '}', 'gm'), '')
17188                 : item.replace(/^ {1,4}/gm, '');
17189             }
17190     
17191             // Determine whether the next list item belongs here.
17192             // Backpedal if it does not belong in this list.
17193             if (this.options.smartLists && i !== l - 1) {
17194               b = block.bullet.exec(cap[i + 1])[0];
17195               if (bull !== b && !(bull.length > 1 && b.length > 1)) {
17196                 src = cap.slice(i + 1).join('\n') + src;
17197                 i = l - 1;
17198               }
17199             }
17200     
17201             // Determine whether item is loose or not.
17202             // Use: /(^|\n)(?! )[^\n]+\n\n(?!\s*$)/
17203             // for discount behavior.
17204             loose = next || /\n\n(?!\s*$)/.test(item);
17205             if (i !== l - 1) {
17206               next = item.charAt(item.length - 1) === '\n';
17207               if (!loose) { loose = next; }
17208             }
17209     
17210             this.tokens.push({
17211               type: loose
17212                 ? 'loose_item_start'
17213                 : 'list_item_start'
17214             });
17215     
17216             // Recurse.
17217             this.token(item, false, bq);
17218     
17219             this.tokens.push({
17220               type: 'list_item_end'
17221             });
17222           }
17223     
17224           this.tokens.push({
17225             type: 'list_end'
17226           });
17227     
17228           continue;
17229         }
17230     
17231         // html
17232         if (cap = this.rules.html.exec(src)) {
17233           src = src.substring(cap[0].length);
17234           this.tokens.push({
17235             type: this.options.sanitize
17236               ? 'paragraph'
17237               : 'html',
17238             pre: !this.options.sanitizer
17239               && (cap[1] === 'pre' || cap[1] === 'script' || cap[1] === 'style'),
17240             text: cap[0]
17241           });
17242           continue;
17243         }
17244     
17245         // def
17246         if ((!bq && top) && (cap = this.rules.def.exec(src))) {
17247           src = src.substring(cap[0].length);
17248           this.tokens.links[cap[1].toLowerCase()] = {
17249             href: cap[2],
17250             title: cap[3]
17251           };
17252           continue;
17253         }
17254     
17255         // table (gfm)
17256         if (top && (cap = this.rules.table.exec(src))) {
17257           src = src.substring(cap[0].length);
17258     
17259           item = {
17260             type: 'table',
17261             header: cap[1].replace(/^ *| *\| *$/g, '').split(/ *\| */),
17262             align: cap[2].replace(/^ *|\| *$/g, '').split(/ *\| */),
17263             cells: cap[3].replace(/(?: *\| *)?\n$/, '').split('\n')
17264           };
17265     
17266           for (i = 0; i < item.align.length; i++) {
17267             if (/^ *-+: *$/.test(item.align[i])) {
17268               item.align[i] = 'right';
17269             } else if (/^ *:-+: *$/.test(item.align[i])) {
17270               item.align[i] = 'center';
17271             } else if (/^ *:-+ *$/.test(item.align[i])) {
17272               item.align[i] = 'left';
17273             } else {
17274               item.align[i] = null;
17275             }
17276           }
17277     
17278           for (i = 0; i < item.cells.length; i++) {
17279             item.cells[i] = item.cells[i]
17280               .replace(/^ *\| *| *\| *$/g, '')
17281               .split(/ *\| */);
17282           }
17283     
17284           this.tokens.push(item);
17285     
17286           continue;
17287         }
17288     
17289         // top-level paragraph
17290         if (top && (cap = this.rules.paragraph.exec(src))) {
17291           src = src.substring(cap[0].length);
17292           this.tokens.push({
17293             type: 'paragraph',
17294             text: cap[1].charAt(cap[1].length - 1) === '\n'
17295               ? cap[1].slice(0, -1)
17296               : cap[1]
17297           });
17298           continue;
17299         }
17300     
17301         // text
17302         if (cap = this.rules.text.exec(src)) {
17303           // Top-level should never reach here.
17304           src = src.substring(cap[0].length);
17305           this.tokens.push({
17306             type: 'text',
17307             text: cap[0]
17308           });
17309           continue;
17310         }
17311     
17312         if (src) {
17313           throw new
17314             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17315         }
17316       }
17317     
17318       return this.tokens;
17319     };
17320     
17321     /**
17322      * Inline-Level Grammar
17323      */
17324     
17325     var inline = {
17326       escape: /^\\([\\`*{}\[\]()#+\-.!_>])/,
17327       autolink: /^<([^ >]+(@|:\/)[^ >]+)>/,
17328       url: noop,
17329       tag: /^<!--[\s\S]*?-->|^<\/?\w+(?:"[^"]*"|'[^']*'|[^'">])*?>/,
17330       link: /^!?\[(inside)\]\(href\)/,
17331       reflink: /^!?\[(inside)\]\s*\[([^\]]*)\]/,
17332       nolink: /^!?\[((?:\[[^\]]*\]|[^\[\]])*)\]/,
17333       strong: /^__([\s\S]+?)__(?!_)|^\*\*([\s\S]+?)\*\*(?!\*)/,
17334       em: /^\b_((?:[^_]|__)+?)_\b|^\*((?:\*\*|[\s\S])+?)\*(?!\*)/,
17335       code: /^(`+)\s*([\s\S]*?[^`])\s*\1(?!`)/,
17336       br: /^ {2,}\n(?!\s*$)/,
17337       del: noop,
17338       text: /^[\s\S]+?(?=[\\<!\[_*`]| {2,}\n|$)/
17339     };
17340     
17341     inline._inside = /(?:\[[^\]]*\]|[^\[\]]|\](?=[^\[]*\]))*/;
17342     inline._href = /\s*<?([\s\S]*?)>?(?:\s+['"]([\s\S]*?)['"])?\s*/;
17343     
17344     inline.link = replace(inline.link)
17345       ('inside', inline._inside)
17346       ('href', inline._href)
17347       ();
17348     
17349     inline.reflink = replace(inline.reflink)
17350       ('inside', inline._inside)
17351       ();
17352     
17353     /**
17354      * Normal Inline Grammar
17355      */
17356     
17357     inline.normal = merge({}, inline);
17358     
17359     /**
17360      * Pedantic Inline Grammar
17361      */
17362     
17363     inline.pedantic = merge({}, inline.normal, {
17364       strong: /^__(?=\S)([\s\S]*?\S)__(?!_)|^\*\*(?=\S)([\s\S]*?\S)\*\*(?!\*)/,
17365       em: /^_(?=\S)([\s\S]*?\S)_(?!_)|^\*(?=\S)([\s\S]*?\S)\*(?!\*)/
17366     });
17367     
17368     /**
17369      * GFM Inline Grammar
17370      */
17371     
17372     inline.gfm = merge({}, inline.normal, {
17373       escape: replace(inline.escape)('])', '~|])')(),
17374       url: /^(https?:\/\/[^\s<]+[^<.,:;"')\]\s])/,
17375       del: /^~~(?=\S)([\s\S]*?\S)~~/,
17376       text: replace(inline.text)
17377         (']|', '~]|')
17378         ('|', '|https?://|')
17379         ()
17380     });
17381     
17382     /**
17383      * GFM + Line Breaks Inline Grammar
17384      */
17385     
17386     inline.breaks = merge({}, inline.gfm, {
17387       br: replace(inline.br)('{2,}', '*')(),
17388       text: replace(inline.gfm.text)('{2,}', '*')()
17389     });
17390     
17391     /**
17392      * Inline Lexer & Compiler
17393      */
17394     
17395     var InlineLexer  = function (links, options) {
17396       this.options = options || marked.defaults;
17397       this.links = links;
17398       this.rules = inline.normal;
17399       this.renderer = this.options.renderer || new Renderer;
17400       this.renderer.options = this.options;
17401     
17402       if (!this.links) {
17403         throw new
17404           Error('Tokens array requires a `links` property.');
17405       }
17406     
17407       if (this.options.gfm) {
17408         if (this.options.breaks) {
17409           this.rules = inline.breaks;
17410         } else {
17411           this.rules = inline.gfm;
17412         }
17413       } else if (this.options.pedantic) {
17414         this.rules = inline.pedantic;
17415       }
17416     }
17417     
17418     /**
17419      * Expose Inline Rules
17420      */
17421     
17422     InlineLexer.rules = inline;
17423     
17424     /**
17425      * Static Lexing/Compiling Method
17426      */
17427     
17428     InlineLexer.output = function(src, links, options) {
17429       var inline = new InlineLexer(links, options);
17430       return inline.output(src);
17431     };
17432     
17433     /**
17434      * Lexing/Compiling
17435      */
17436     
17437     InlineLexer.prototype.output = function(src) {
17438       var out = ''
17439         , link
17440         , text
17441         , href
17442         , cap;
17443     
17444       while (src) {
17445         // escape
17446         if (cap = this.rules.escape.exec(src)) {
17447           src = src.substring(cap[0].length);
17448           out += cap[1];
17449           continue;
17450         }
17451     
17452         // autolink
17453         if (cap = this.rules.autolink.exec(src)) {
17454           src = src.substring(cap[0].length);
17455           if (cap[2] === '@') {
17456             text = cap[1].charAt(6) === ':'
17457               ? this.mangle(cap[1].substring(7))
17458               : this.mangle(cap[1]);
17459             href = this.mangle('mailto:') + text;
17460           } else {
17461             text = escape(cap[1]);
17462             href = text;
17463           }
17464           out += this.renderer.link(href, null, text);
17465           continue;
17466         }
17467     
17468         // url (gfm)
17469         if (!this.inLink && (cap = this.rules.url.exec(src))) {
17470           src = src.substring(cap[0].length);
17471           text = escape(cap[1]);
17472           href = text;
17473           out += this.renderer.link(href, null, text);
17474           continue;
17475         }
17476     
17477         // tag
17478         if (cap = this.rules.tag.exec(src)) {
17479           if (!this.inLink && /^<a /i.test(cap[0])) {
17480             this.inLink = true;
17481           } else if (this.inLink && /^<\/a>/i.test(cap[0])) {
17482             this.inLink = false;
17483           }
17484           src = src.substring(cap[0].length);
17485           out += this.options.sanitize
17486             ? this.options.sanitizer
17487               ? this.options.sanitizer(cap[0])
17488               : escape(cap[0])
17489             : cap[0];
17490           continue;
17491         }
17492     
17493         // link
17494         if (cap = this.rules.link.exec(src)) {
17495           src = src.substring(cap[0].length);
17496           this.inLink = true;
17497           out += this.outputLink(cap, {
17498             href: cap[2],
17499             title: cap[3]
17500           });
17501           this.inLink = false;
17502           continue;
17503         }
17504     
17505         // reflink, nolink
17506         if ((cap = this.rules.reflink.exec(src))
17507             || (cap = this.rules.nolink.exec(src))) {
17508           src = src.substring(cap[0].length);
17509           link = (cap[2] || cap[1]).replace(/\s+/g, ' ');
17510           link = this.links[link.toLowerCase()];
17511           if (!link || !link.href) {
17512             out += cap[0].charAt(0);
17513             src = cap[0].substring(1) + src;
17514             continue;
17515           }
17516           this.inLink = true;
17517           out += this.outputLink(cap, link);
17518           this.inLink = false;
17519           continue;
17520         }
17521     
17522         // strong
17523         if (cap = this.rules.strong.exec(src)) {
17524           src = src.substring(cap[0].length);
17525           out += this.renderer.strong(this.output(cap[2] || cap[1]));
17526           continue;
17527         }
17528     
17529         // em
17530         if (cap = this.rules.em.exec(src)) {
17531           src = src.substring(cap[0].length);
17532           out += this.renderer.em(this.output(cap[2] || cap[1]));
17533           continue;
17534         }
17535     
17536         // code
17537         if (cap = this.rules.code.exec(src)) {
17538           src = src.substring(cap[0].length);
17539           out += this.renderer.codespan(escape(cap[2], true));
17540           continue;
17541         }
17542     
17543         // br
17544         if (cap = this.rules.br.exec(src)) {
17545           src = src.substring(cap[0].length);
17546           out += this.renderer.br();
17547           continue;
17548         }
17549     
17550         // del (gfm)
17551         if (cap = this.rules.del.exec(src)) {
17552           src = src.substring(cap[0].length);
17553           out += this.renderer.del(this.output(cap[1]));
17554           continue;
17555         }
17556     
17557         // text
17558         if (cap = this.rules.text.exec(src)) {
17559           src = src.substring(cap[0].length);
17560           out += this.renderer.text(escape(this.smartypants(cap[0])));
17561           continue;
17562         }
17563     
17564         if (src) {
17565           throw new
17566             Error('Infinite loop on byte: ' + src.charCodeAt(0));
17567         }
17568       }
17569     
17570       return out;
17571     };
17572     
17573     /**
17574      * Compile Link
17575      */
17576     
17577     InlineLexer.prototype.outputLink = function(cap, link) {
17578       var href = escape(link.href)
17579         , title = link.title ? escape(link.title) : null;
17580     
17581       return cap[0].charAt(0) !== '!'
17582         ? this.renderer.link(href, title, this.output(cap[1]))
17583         : this.renderer.image(href, title, escape(cap[1]));
17584     };
17585     
17586     /**
17587      * Smartypants Transformations
17588      */
17589     
17590     InlineLexer.prototype.smartypants = function(text) {
17591       if (!this.options.smartypants)  { return text; }
17592       return text
17593         // em-dashes
17594         .replace(/---/g, '\u2014')
17595         // en-dashes
17596         .replace(/--/g, '\u2013')
17597         // opening singles
17598         .replace(/(^|[-\u2014/(\[{"\s])'/g, '$1\u2018')
17599         // closing singles & apostrophes
17600         .replace(/'/g, '\u2019')
17601         // opening doubles
17602         .replace(/(^|[-\u2014/(\[{\u2018\s])"/g, '$1\u201c')
17603         // closing doubles
17604         .replace(/"/g, '\u201d')
17605         // ellipses
17606         .replace(/\.{3}/g, '\u2026');
17607     };
17608     
17609     /**
17610      * Mangle Links
17611      */
17612     
17613     InlineLexer.prototype.mangle = function(text) {
17614       if (!this.options.mangle) { return text; }
17615       var out = ''
17616         , l = text.length
17617         , i = 0
17618         , ch;
17619     
17620       for (; i < l; i++) {
17621         ch = text.charCodeAt(i);
17622         if (Math.random() > 0.5) {
17623           ch = 'x' + ch.toString(16);
17624         }
17625         out += '&#' + ch + ';';
17626       }
17627     
17628       return out;
17629     };
17630     
17631     /**
17632      * Renderer
17633      */
17634     
17635      /**
17636          * eval:var:Renderer
17637     */
17638     
17639     var Renderer   = function (options) {
17640       this.options = options || {};
17641     }
17642     
17643     Renderer.prototype.code = function(code, lang, escaped) {
17644       if (this.options.highlight) {
17645         var out = this.options.highlight(code, lang);
17646         if (out != null && out !== code) {
17647           escaped = true;
17648           code = out;
17649         }
17650       } else {
17651             // hack!!! - it's already escapeD?
17652             escaped = true;
17653       }
17654     
17655       if (!lang) {
17656         return '<pre><code>'
17657           + (escaped ? code : escape(code, true))
17658           + '\n</code></pre>';
17659       }
17660     
17661       return '<pre><code class="'
17662         + this.options.langPrefix
17663         + escape(lang, true)
17664         + '">'
17665         + (escaped ? code : escape(code, true))
17666         + '\n</code></pre>\n';
17667     };
17668     
17669     Renderer.prototype.blockquote = function(quote) {
17670       return '<blockquote>\n' + quote + '</blockquote>\n';
17671     };
17672     
17673     Renderer.prototype.html = function(html) {
17674       return html;
17675     };
17676     
17677     Renderer.prototype.heading = function(text, level, raw) {
17678       return '<h'
17679         + level
17680         + ' id="'
17681         + this.options.headerPrefix
17682         + raw.toLowerCase().replace(/[^\w]+/g, '-')
17683         + '">'
17684         + text
17685         + '</h'
17686         + level
17687         + '>\n';
17688     };
17689     
17690     Renderer.prototype.hr = function() {
17691       return this.options.xhtml ? '<hr/>\n' : '<hr>\n';
17692     };
17693     
17694     Renderer.prototype.list = function(body, ordered) {
17695       var type = ordered ? 'ol' : 'ul';
17696       return '<' + type + '>\n' + body + '</' + type + '>\n';
17697     };
17698     
17699     Renderer.prototype.listitem = function(text) {
17700       return '<li>' + text + '</li>\n';
17701     };
17702     
17703     Renderer.prototype.paragraph = function(text) {
17704       return '<p>' + text + '</p>\n';
17705     };
17706     
17707     Renderer.prototype.table = function(header, body) {
17708       return '<table class="table table-striped">\n'
17709         + '<thead>\n'
17710         + header
17711         + '</thead>\n'
17712         + '<tbody>\n'
17713         + body
17714         + '</tbody>\n'
17715         + '</table>\n';
17716     };
17717     
17718     Renderer.prototype.tablerow = function(content) {
17719       return '<tr>\n' + content + '</tr>\n';
17720     };
17721     
17722     Renderer.prototype.tablecell = function(content, flags) {
17723       var type = flags.header ? 'th' : 'td';
17724       var tag = flags.align
17725         ? '<' + type + ' style="text-align:' + flags.align + '">'
17726         : '<' + type + '>';
17727       return tag + content + '</' + type + '>\n';
17728     };
17729     
17730     // span level renderer
17731     Renderer.prototype.strong = function(text) {
17732       return '<strong>' + text + '</strong>';
17733     };
17734     
17735     Renderer.prototype.em = function(text) {
17736       return '<em>' + text + '</em>';
17737     };
17738     
17739     Renderer.prototype.codespan = function(text) {
17740       return '<code>' + text + '</code>';
17741     };
17742     
17743     Renderer.prototype.br = function() {
17744       return this.options.xhtml ? '<br/>' : '<br>';
17745     };
17746     
17747     Renderer.prototype.del = function(text) {
17748       return '<del>' + text + '</del>';
17749     };
17750     
17751     Renderer.prototype.link = function(href, title, text) {
17752       if (this.options.sanitize) {
17753         try {
17754           var prot = decodeURIComponent(unescape(href))
17755             .replace(/[^\w:]/g, '')
17756             .toLowerCase();
17757         } catch (e) {
17758           return '';
17759         }
17760         if (prot.indexOf('javascript:') === 0 || prot.indexOf('vbscript:') === 0) {
17761           return '';
17762         }
17763       }
17764       var out = '<a href="' + href + '"';
17765       if (title) {
17766         out += ' title="' + title + '"';
17767       }
17768       out += '>' + text + '</a>';
17769       return out;
17770     };
17771     
17772     Renderer.prototype.image = function(href, title, text) {
17773       var out = '<img src="' + href + '" alt="' + text + '"';
17774       if (title) {
17775         out += ' title="' + title + '"';
17776       }
17777       out += this.options.xhtml ? '/>' : '>';
17778       return out;
17779     };
17780     
17781     Renderer.prototype.text = function(text) {
17782       return text;
17783     };
17784     
17785     /**
17786      * Parsing & Compiling
17787      */
17788          /**
17789          * eval:var:Parser
17790     */
17791     
17792     var Parser= function (options) {
17793       this.tokens = [];
17794       this.token = null;
17795       this.options = options || marked.defaults;
17796       this.options.renderer = this.options.renderer || new Renderer;
17797       this.renderer = this.options.renderer;
17798       this.renderer.options = this.options;
17799     }
17800     
17801     /**
17802      * Static Parse Method
17803      */
17804     
17805     Parser.parse = function(src, options, renderer) {
17806       var parser = new Parser(options, renderer);
17807       return parser.parse(src);
17808     };
17809     
17810     /**
17811      * Parse Loop
17812      */
17813     
17814     Parser.prototype.parse = function(src) {
17815       this.inline = new InlineLexer(src.links, this.options, this.renderer);
17816       this.tokens = src.reverse();
17817     
17818       var out = '';
17819       while (this.next()) {
17820         out += this.tok();
17821       }
17822     
17823       return out;
17824     };
17825     
17826     /**
17827      * Next Token
17828      */
17829     
17830     Parser.prototype.next = function() {
17831       return this.token = this.tokens.pop();
17832     };
17833     
17834     /**
17835      * Preview Next Token
17836      */
17837     
17838     Parser.prototype.peek = function() {
17839       return this.tokens[this.tokens.length - 1] || 0;
17840     };
17841     
17842     /**
17843      * Parse Text Tokens
17844      */
17845     
17846     Parser.prototype.parseText = function() {
17847       var body = this.token.text;
17848     
17849       while (this.peek().type === 'text') {
17850         body += '\n' + this.next().text;
17851       }
17852     
17853       return this.inline.output(body);
17854     };
17855     
17856     /**
17857      * Parse Current Token
17858      */
17859     
17860     Parser.prototype.tok = function() {
17861       switch (this.token.type) {
17862         case 'space': {
17863           return '';
17864         }
17865         case 'hr': {
17866           return this.renderer.hr();
17867         }
17868         case 'heading': {
17869           return this.renderer.heading(
17870             this.inline.output(this.token.text),
17871             this.token.depth,
17872             this.token.text);
17873         }
17874         case 'code': {
17875           return this.renderer.code(this.token.text,
17876             this.token.lang,
17877             this.token.escaped);
17878         }
17879         case 'table': {
17880           var header = ''
17881             , body = ''
17882             , i
17883             , row
17884             , cell
17885             , flags
17886             , j;
17887     
17888           // header
17889           cell = '';
17890           for (i = 0; i < this.token.header.length; i++) {
17891             flags = { header: true, align: this.token.align[i] };
17892             cell += this.renderer.tablecell(
17893               this.inline.output(this.token.header[i]),
17894               { header: true, align: this.token.align[i] }
17895             );
17896           }
17897           header += this.renderer.tablerow(cell);
17898     
17899           for (i = 0; i < this.token.cells.length; i++) {
17900             row = this.token.cells[i];
17901     
17902             cell = '';
17903             for (j = 0; j < row.length; j++) {
17904               cell += this.renderer.tablecell(
17905                 this.inline.output(row[j]),
17906                 { header: false, align: this.token.align[j] }
17907               );
17908             }
17909     
17910             body += this.renderer.tablerow(cell);
17911           }
17912           return this.renderer.table(header, body);
17913         }
17914         case 'blockquote_start': {
17915           var body = '';
17916     
17917           while (this.next().type !== 'blockquote_end') {
17918             body += this.tok();
17919           }
17920     
17921           return this.renderer.blockquote(body);
17922         }
17923         case 'list_start': {
17924           var body = ''
17925             , ordered = this.token.ordered;
17926     
17927           while (this.next().type !== 'list_end') {
17928             body += this.tok();
17929           }
17930     
17931           return this.renderer.list(body, ordered);
17932         }
17933         case 'list_item_start': {
17934           var body = '';
17935     
17936           while (this.next().type !== 'list_item_end') {
17937             body += this.token.type === 'text'
17938               ? this.parseText()
17939               : this.tok();
17940           }
17941     
17942           return this.renderer.listitem(body);
17943         }
17944         case 'loose_item_start': {
17945           var body = '';
17946     
17947           while (this.next().type !== 'list_item_end') {
17948             body += this.tok();
17949           }
17950     
17951           return this.renderer.listitem(body);
17952         }
17953         case 'html': {
17954           var html = !this.token.pre && !this.options.pedantic
17955             ? this.inline.output(this.token.text)
17956             : this.token.text;
17957           return this.renderer.html(html);
17958         }
17959         case 'paragraph': {
17960           return this.renderer.paragraph(this.inline.output(this.token.text));
17961         }
17962         case 'text': {
17963           return this.renderer.paragraph(this.parseText());
17964         }
17965       }
17966     };
17967   
17968     
17969     /**
17970      * Marked
17971      */
17972          /**
17973          * eval:var:marked
17974     */
17975     var marked = function (src, opt, callback) {
17976       if (callback || typeof opt === 'function') {
17977         if (!callback) {
17978           callback = opt;
17979           opt = null;
17980         }
17981     
17982         opt = merge({}, marked.defaults, opt || {});
17983     
17984         var highlight = opt.highlight
17985           , tokens
17986           , pending
17987           , i = 0;
17988     
17989         try {
17990           tokens = Lexer.lex(src, opt)
17991         } catch (e) {
17992           return callback(e);
17993         }
17994     
17995         pending = tokens.length;
17996          /**
17997          * eval:var:done
17998     */
17999         var done = function(err) {
18000           if (err) {
18001             opt.highlight = highlight;
18002             return callback(err);
18003           }
18004     
18005           var out;
18006     
18007           try {
18008             out = Parser.parse(tokens, opt);
18009           } catch (e) {
18010             err = e;
18011           }
18012     
18013           opt.highlight = highlight;
18014     
18015           return err
18016             ? callback(err)
18017             : callback(null, out);
18018         };
18019     
18020         if (!highlight || highlight.length < 3) {
18021           return done();
18022         }
18023     
18024         delete opt.highlight;
18025     
18026         if (!pending) { return done(); }
18027     
18028         for (; i < tokens.length; i++) {
18029           (function(token) {
18030             if (token.type !== 'code') {
18031               return --pending || done();
18032             }
18033             return highlight(token.text, token.lang, function(err, code) {
18034               if (err) { return done(err); }
18035               if (code == null || code === token.text) {
18036                 return --pending || done();
18037               }
18038               token.text = code;
18039               token.escaped = true;
18040               --pending || done();
18041             });
18042           })(tokens[i]);
18043         }
18044     
18045         return;
18046       }
18047       try {
18048         if (opt) { opt = merge({}, marked.defaults, opt); }
18049         return Parser.parse(Lexer.lex(src, opt), opt);
18050       } catch (e) {
18051         e.message += '\nPlease report this to https://github.com/chjj/marked.';
18052         if ((opt || marked.defaults).silent) {
18053           return '<p>An error occured:</p><pre>'
18054             + escape(e.message + '', true)
18055             + '</pre>';
18056         }
18057         throw e;
18058       }
18059     }
18060     
18061     /**
18062      * Options
18063      */
18064     
18065     marked.options =
18066     marked.setOptions = function(opt) {
18067       merge(marked.defaults, opt);
18068       return marked;
18069     };
18070     
18071     marked.defaults = {
18072       gfm: true,
18073       tables: true,
18074       breaks: false,
18075       pedantic: false,
18076       sanitize: false,
18077       sanitizer: null,
18078       mangle: true,
18079       smartLists: false,
18080       silent: false,
18081       highlight: null,
18082       langPrefix: 'lang-',
18083       smartypants: false,
18084       headerPrefix: '',
18085       renderer: new Renderer,
18086       xhtml: false
18087     };
18088     
18089     /**
18090      * Expose
18091      */
18092     
18093     marked.Parser = Parser;
18094     marked.parser = Parser.parse;
18095     
18096     marked.Renderer = Renderer;
18097     
18098     marked.Lexer = Lexer;
18099     marked.lexer = Lexer.lex;
18100     
18101     marked.InlineLexer = InlineLexer;
18102     marked.inlineLexer = InlineLexer.output;
18103     
18104     marked.parse = marked;
18105     
18106     Roo.Markdown.marked = marked;
18107
18108 })();/*
18109  * Based on:
18110  * Ext JS Library 1.1.1
18111  * Copyright(c) 2006-2007, Ext JS, LLC.
18112  *
18113  * Originally Released Under LGPL - original licence link has changed is not relivant.
18114  *
18115  * Fork - LGPL
18116  * <script type="text/javascript">
18117  */
18118
18119
18120
18121 /*
18122  * These classes are derivatives of the similarly named classes in the YUI Library.
18123  * The original license:
18124  * Copyright (c) 2006, Yahoo! Inc. All rights reserved.
18125  * Code licensed under the BSD License:
18126  * http://developer.yahoo.net/yui/license.txt
18127  */
18128
18129 (function() {
18130
18131 var Event=Roo.EventManager;
18132 var Dom=Roo.lib.Dom;
18133
18134 /**
18135  * @class Roo.dd.DragDrop
18136  * @extends Roo.util.Observable
18137  * Defines the interface and base operation of items that that can be
18138  * dragged or can be drop targets.  It was designed to be extended, overriding
18139  * the event handlers for startDrag, onDrag, onDragOver and onDragOut.
18140  * Up to three html elements can be associated with a DragDrop instance:
18141  * <ul>
18142  * <li>linked element: the element that is passed into the constructor.
18143  * This is the element which defines the boundaries for interaction with
18144  * other DragDrop objects.</li>
18145  * <li>handle element(s): The drag operation only occurs if the element that
18146  * was clicked matches a handle element.  By default this is the linked
18147  * element, but there are times that you will want only a portion of the
18148  * linked element to initiate the drag operation, and the setHandleElId()
18149  * method provides a way to define this.</li>
18150  * <li>drag element: this represents the element that would be moved along
18151  * with the cursor during a drag operation.  By default, this is the linked
18152  * element itself as in {@link Roo.dd.DD}.  setDragElId() lets you define
18153  * a separate element that would be moved, as in {@link Roo.dd.DDProxy}.
18154  * </li>
18155  * </ul>
18156  * This class should not be instantiated until the onload event to ensure that
18157  * the associated elements are available.
18158  * The following would define a DragDrop obj that would interact with any
18159  * other DragDrop obj in the "group1" group:
18160  * <pre>
18161  *  dd = new Roo.dd.DragDrop("div1", "group1");
18162  * </pre>
18163  * Since none of the event handlers have been implemented, nothing would
18164  * actually happen if you were to run the code above.  Normally you would
18165  * override this class or one of the default implementations, but you can
18166  * also override the methods you want on an instance of the class...
18167  * <pre>
18168  *  dd.onDragDrop = function(e, id) {
18169  *  &nbsp;&nbsp;alert("dd was dropped on " + id);
18170  *  }
18171  * </pre>
18172  * @constructor
18173  * @param {String} id of the element that is linked to this instance
18174  * @param {String} sGroup the group of related DragDrop objects
18175  * @param {object} config an object containing configurable attributes
18176  *                Valid properties for DragDrop:
18177  *                    padding, isTarget, maintainOffset, primaryButtonOnly
18178  */
18179 Roo.dd.DragDrop = function(id, sGroup, config) {
18180     if (id) {
18181         this.init(id, sGroup, config);
18182     }
18183     
18184 };
18185
18186 Roo.extend(Roo.dd.DragDrop, Roo.util.Observable , {
18187
18188     /**
18189      * The id of the element associated with this object.  This is what we
18190      * refer to as the "linked element" because the size and position of
18191      * this element is used to determine when the drag and drop objects have
18192      * interacted.
18193      * @property id
18194      * @type String
18195      */
18196     id: null,
18197
18198     /**
18199      * Configuration attributes passed into the constructor
18200      * @property config
18201      * @type object
18202      */
18203     config: null,
18204
18205     /**
18206      * The id of the element that will be dragged.  By default this is same
18207      * as the linked element , but could be changed to another element. Ex:
18208      * Roo.dd.DDProxy
18209      * @property dragElId
18210      * @type String
18211      * @private
18212      */
18213     dragElId: null,
18214
18215     /**
18216      * the id of the element that initiates the drag operation.  By default
18217      * this is the linked element, but could be changed to be a child of this
18218      * element.  This lets us do things like only starting the drag when the
18219      * header element within the linked html element is clicked.
18220      * @property handleElId
18221      * @type String
18222      * @private
18223      */
18224     handleElId: null,
18225
18226     /**
18227      * An associative array of HTML tags that will be ignored if clicked.
18228      * @property invalidHandleTypes
18229      * @type {string: string}
18230      */
18231     invalidHandleTypes: null,
18232
18233     /**
18234      * An associative array of ids for elements that will be ignored if clicked
18235      * @property invalidHandleIds
18236      * @type {string: string}
18237      */
18238     invalidHandleIds: null,
18239
18240     /**
18241      * An indexted array of css class names for elements that will be ignored
18242      * if clicked.
18243      * @property invalidHandleClasses
18244      * @type string[]
18245      */
18246     invalidHandleClasses: null,
18247
18248     /**
18249      * The linked element's absolute X position at the time the drag was
18250      * started
18251      * @property startPageX
18252      * @type int
18253      * @private
18254      */
18255     startPageX: 0,
18256
18257     /**
18258      * The linked element's absolute X position at the time the drag was
18259      * started
18260      * @property startPageY
18261      * @type int
18262      * @private
18263      */
18264     startPageY: 0,
18265
18266     /**
18267      * The group defines a logical collection of DragDrop objects that are
18268      * related.  Instances only get events when interacting with other
18269      * DragDrop object in the same group.  This lets us define multiple
18270      * groups using a single DragDrop subclass if we want.
18271      * @property groups
18272      * @type {string: string}
18273      */
18274     groups: null,
18275
18276     /**
18277      * Individual drag/drop instances can be locked.  This will prevent
18278      * onmousedown start drag.
18279      * @property locked
18280      * @type boolean
18281      * @private
18282      */
18283     locked: false,
18284
18285     /**
18286      * Lock this instance
18287      * @method lock
18288      */
18289     lock: function() { this.locked = true; },
18290
18291     /**
18292      * Unlock this instace
18293      * @method unlock
18294      */
18295     unlock: function() { this.locked = false; },
18296
18297     /**
18298      * By default, all insances can be a drop target.  This can be disabled by
18299      * setting isTarget to false.
18300      * @method isTarget
18301      * @type boolean
18302      */
18303     isTarget: true,
18304
18305     /**
18306      * The padding configured for this drag and drop object for calculating
18307      * the drop zone intersection with this object.
18308      * @method padding
18309      * @type int[]
18310      */
18311     padding: null,
18312
18313     /**
18314      * Cached reference to the linked element
18315      * @property _domRef
18316      * @private
18317      */
18318     _domRef: null,
18319
18320     /**
18321      * Internal typeof flag
18322      * @property __ygDragDrop
18323      * @private
18324      */
18325     __ygDragDrop: true,
18326
18327     /**
18328      * Set to true when horizontal contraints are applied
18329      * @property constrainX
18330      * @type boolean
18331      * @private
18332      */
18333     constrainX: false,
18334
18335     /**
18336      * Set to true when vertical contraints are applied
18337      * @property constrainY
18338      * @type boolean
18339      * @private
18340      */
18341     constrainY: false,
18342
18343     /**
18344      * The left constraint
18345      * @property minX
18346      * @type int
18347      * @private
18348      */
18349     minX: 0,
18350
18351     /**
18352      * The right constraint
18353      * @property maxX
18354      * @type int
18355      * @private
18356      */
18357     maxX: 0,
18358
18359     /**
18360      * The up constraint
18361      * @property minY
18362      * @type int
18363      * @type int
18364      * @private
18365      */
18366     minY: 0,
18367
18368     /**
18369      * The down constraint
18370      * @property maxY
18371      * @type int
18372      * @private
18373      */
18374     maxY: 0,
18375
18376     /**
18377      * Maintain offsets when we resetconstraints.  Set to true when you want
18378      * the position of the element relative to its parent to stay the same
18379      * when the page changes
18380      *
18381      * @property maintainOffset
18382      * @type boolean
18383      */
18384     maintainOffset: false,
18385
18386     /**
18387      * Array of pixel locations the element will snap to if we specified a
18388      * horizontal graduation/interval.  This array is generated automatically
18389      * when you define a tick interval.
18390      * @property xTicks
18391      * @type int[]
18392      */
18393     xTicks: null,
18394
18395     /**
18396      * Array of pixel locations the element will snap to if we specified a
18397      * vertical graduation/interval.  This array is generated automatically
18398      * when you define a tick interval.
18399      * @property yTicks
18400      * @type int[]
18401      */
18402     yTicks: null,
18403
18404     /**
18405      * By default the drag and drop instance will only respond to the primary
18406      * button click (left button for a right-handed mouse).  Set to true to
18407      * allow drag and drop to start with any mouse click that is propogated
18408      * by the browser
18409      * @property primaryButtonOnly
18410      * @type boolean
18411      */
18412     primaryButtonOnly: true,
18413
18414     /**
18415      * The availabe property is false until the linked dom element is accessible.
18416      * @property available
18417      * @type boolean
18418      */
18419     available: false,
18420
18421     /**
18422      * By default, drags can only be initiated if the mousedown occurs in the
18423      * region the linked element is.  This is done in part to work around a
18424      * bug in some browsers that mis-report the mousedown if the previous
18425      * mouseup happened outside of the window.  This property is set to true
18426      * if outer handles are defined.
18427      *
18428      * @property hasOuterHandles
18429      * @type boolean
18430      * @default false
18431      */
18432     hasOuterHandles: false,
18433
18434     /**
18435      * Code that executes immediately before the startDrag event
18436      * @method b4StartDrag
18437      * @private
18438      */
18439     b4StartDrag: function(x, y) { },
18440
18441     /**
18442      * Abstract method called after a drag/drop object is clicked
18443      * and the drag or mousedown time thresholds have beeen met.
18444      * @method startDrag
18445      * @param {int} X click location
18446      * @param {int} Y click location
18447      */
18448     startDrag: function(x, y) { /* override this */ },
18449
18450     /**
18451      * Code that executes immediately before the onDrag event
18452      * @method b4Drag
18453      * @private
18454      */
18455     b4Drag: function(e) { },
18456
18457     /**
18458      * Abstract method called during the onMouseMove event while dragging an
18459      * object.
18460      * @method onDrag
18461      * @param {Event} e the mousemove event
18462      */
18463     onDrag: function(e) { /* override this */ },
18464
18465     /**
18466      * Abstract method called when this element fist begins hovering over
18467      * another DragDrop obj
18468      * @method onDragEnter
18469      * @param {Event} e the mousemove event
18470      * @param {String|DragDrop[]} id In POINT mode, the element
18471      * id this is hovering over.  In INTERSECT mode, an array of one or more
18472      * dragdrop items being hovered over.
18473      */
18474     onDragEnter: function(e, id) { /* override this */ },
18475
18476     /**
18477      * Code that executes immediately before the onDragOver event
18478      * @method b4DragOver
18479      * @private
18480      */
18481     b4DragOver: function(e) { },
18482
18483     /**
18484      * Abstract method called when this element is hovering over another
18485      * DragDrop obj
18486      * @method onDragOver
18487      * @param {Event} e the mousemove event
18488      * @param {String|DragDrop[]} id In POINT mode, the element
18489      * id this is hovering over.  In INTERSECT mode, an array of dd items
18490      * being hovered over.
18491      */
18492     onDragOver: function(e, id) { /* override this */ },
18493
18494     /**
18495      * Code that executes immediately before the onDragOut event
18496      * @method b4DragOut
18497      * @private
18498      */
18499     b4DragOut: function(e) { },
18500
18501     /**
18502      * Abstract method called when we are no longer hovering over an element
18503      * @method onDragOut
18504      * @param {Event} e the mousemove event
18505      * @param {String|DragDrop[]} id In POINT mode, the element
18506      * id this was hovering over.  In INTERSECT mode, an array of dd items
18507      * that the mouse is no longer over.
18508      */
18509     onDragOut: function(e, id) { /* override this */ },
18510
18511     /**
18512      * Code that executes immediately before the onDragDrop event
18513      * @method b4DragDrop
18514      * @private
18515      */
18516     b4DragDrop: function(e) { },
18517
18518     /**
18519      * Abstract method called when this item is dropped on another DragDrop
18520      * obj
18521      * @method onDragDrop
18522      * @param {Event} e the mouseup event
18523      * @param {String|DragDrop[]} id In POINT mode, the element
18524      * id this was dropped on.  In INTERSECT mode, an array of dd items this
18525      * was dropped on.
18526      */
18527     onDragDrop: function(e, id) { /* override this */ },
18528
18529     /**
18530      * Abstract method called when this item is dropped on an area with no
18531      * drop target
18532      * @method onInvalidDrop
18533      * @param {Event} e the mouseup event
18534      */
18535     onInvalidDrop: function(e) { /* override this */ },
18536
18537     /**
18538      * Code that executes immediately before the endDrag event
18539      * @method b4EndDrag
18540      * @private
18541      */
18542     b4EndDrag: function(e) { },
18543
18544     /**
18545      * Fired when we are done dragging the object
18546      * @method endDrag
18547      * @param {Event} e the mouseup event
18548      */
18549     endDrag: function(e) { /* override this */ },
18550
18551     /**
18552      * Code executed immediately before the onMouseDown event
18553      * @method b4MouseDown
18554      * @param {Event} e the mousedown event
18555      * @private
18556      */
18557     b4MouseDown: function(e) {  },
18558
18559     /**
18560      * Event handler that fires when a drag/drop obj gets a mousedown
18561      * @method onMouseDown
18562      * @param {Event} e the mousedown event
18563      */
18564     onMouseDown: function(e) { /* override this */ },
18565
18566     /**
18567      * Event handler that fires when a drag/drop obj gets a mouseup
18568      * @method onMouseUp
18569      * @param {Event} e the mouseup event
18570      */
18571     onMouseUp: function(e) { /* override this */ },
18572
18573     /**
18574      * Override the onAvailable method to do what is needed after the initial
18575      * position was determined.
18576      * @method onAvailable
18577      */
18578     onAvailable: function () {
18579     },
18580
18581     /*
18582      * Provides default constraint padding to "constrainTo" elements (defaults to {left: 0, right:0, top:0, bottom:0}).
18583      * @type Object
18584      */
18585     defaultPadding : {left:0, right:0, top:0, bottom:0},
18586
18587     /*
18588      * Initializes the drag drop object's constraints to restrict movement to a certain element.
18589  *
18590  * Usage:
18591  <pre><code>
18592  var dd = new Roo.dd.DDProxy("dragDiv1", "proxytest",
18593                 { dragElId: "existingProxyDiv" });
18594  dd.startDrag = function(){
18595      this.constrainTo("parent-id");
18596  };
18597  </code></pre>
18598  * Or you can initalize it using the {@link Roo.Element} object:
18599  <pre><code>
18600  Roo.get("dragDiv1").initDDProxy("proxytest", {dragElId: "existingProxyDiv"}, {
18601      startDrag : function(){
18602          this.constrainTo("parent-id");
18603      }
18604  });
18605  </code></pre>
18606      * @param {String/HTMLElement/Element} constrainTo The element to constrain to.
18607      * @param {Object/Number} pad (optional) Pad provides a way to specify "padding" of the constraints,
18608      * and can be either a number for symmetrical padding (4 would be equal to {left:4, right:4, top:4, bottom:4}) or
18609      * an object containing the sides to pad. For example: {right:10, bottom:10}
18610      * @param {Boolean} inContent (optional) Constrain the draggable in the content box of the element (inside padding and borders)
18611      */
18612     constrainTo : function(constrainTo, pad, inContent){
18613         if(typeof pad == "number"){
18614             pad = {left: pad, right:pad, top:pad, bottom:pad};
18615         }
18616         pad = pad || this.defaultPadding;
18617         var b = Roo.get(this.getEl()).getBox();
18618         var ce = Roo.get(constrainTo);
18619         var s = ce.getScroll();
18620         var c, cd = ce.dom;
18621         if(cd == document.body){
18622             c = { x: s.left, y: s.top, width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
18623         }else{
18624             xy = ce.getXY();
18625             c = {x : xy[0]+s.left, y: xy[1]+s.top, width: cd.clientWidth, height: cd.clientHeight};
18626         }
18627
18628
18629         var topSpace = b.y - c.y;
18630         var leftSpace = b.x - c.x;
18631
18632         this.resetConstraints();
18633         this.setXConstraint(leftSpace - (pad.left||0), // left
18634                 c.width - leftSpace - b.width - (pad.right||0) //right
18635         );
18636         this.setYConstraint(topSpace - (pad.top||0), //top
18637                 c.height - topSpace - b.height - (pad.bottom||0) //bottom
18638         );
18639     },
18640
18641     /**
18642      * Returns a reference to the linked element
18643      * @method getEl
18644      * @return {HTMLElement} the html element
18645      */
18646     getEl: function() {
18647         if (!this._domRef) {
18648             this._domRef = Roo.getDom(this.id);
18649         }
18650
18651         return this._domRef;
18652     },
18653
18654     /**
18655      * Returns a reference to the actual element to drag.  By default this is
18656      * the same as the html element, but it can be assigned to another
18657      * element. An example of this can be found in Roo.dd.DDProxy
18658      * @method getDragEl
18659      * @return {HTMLElement} the html element
18660      */
18661     getDragEl: function() {
18662         return Roo.getDom(this.dragElId);
18663     },
18664
18665     /**
18666      * Sets up the DragDrop object.  Must be called in the constructor of any
18667      * Roo.dd.DragDrop subclass
18668      * @method init
18669      * @param id the id of the linked element
18670      * @param {String} sGroup the group of related items
18671      * @param {object} config configuration attributes
18672      */
18673     init: function(id, sGroup, config) {
18674         this.initTarget(id, sGroup, config);
18675         if (!Roo.isTouch) {
18676             Event.on(this.id, "mousedown", this.handleMouseDown, this);
18677         }
18678         Event.on(this.id, "touchstart", this.handleMouseDown, this);
18679         // Event.on(this.id, "selectstart", Event.preventDefault);
18680     },
18681
18682     /**
18683      * Initializes Targeting functionality only... the object does not
18684      * get a mousedown handler.
18685      * @method initTarget
18686      * @param id the id of the linked element
18687      * @param {String} sGroup the group of related items
18688      * @param {object} config configuration attributes
18689      */
18690     initTarget: function(id, sGroup, config) {
18691
18692         // configuration attributes
18693         this.config = config || {};
18694
18695         // create a local reference to the drag and drop manager
18696         this.DDM = Roo.dd.DDM;
18697         // initialize the groups array
18698         this.groups = {};
18699
18700         // assume that we have an element reference instead of an id if the
18701         // parameter is not a string
18702         if (typeof id !== "string") {
18703             id = Roo.id(id);
18704         }
18705
18706         // set the id
18707         this.id = id;
18708
18709         // add to an interaction group
18710         this.addToGroup((sGroup) ? sGroup : "default");
18711
18712         // We don't want to register this as the handle with the manager
18713         // so we just set the id rather than calling the setter.
18714         this.handleElId = id;
18715
18716         // the linked element is the element that gets dragged by default
18717         this.setDragElId(id);
18718
18719         // by default, clicked anchors will not start drag operations.
18720         this.invalidHandleTypes = { A: "A" };
18721         this.invalidHandleIds = {};
18722         this.invalidHandleClasses = [];
18723
18724         this.applyConfig();
18725
18726         this.handleOnAvailable();
18727     },
18728
18729     /**
18730      * Applies the configuration parameters that were passed into the constructor.
18731      * This is supposed to happen at each level through the inheritance chain.  So
18732      * a DDProxy implentation will execute apply config on DDProxy, DD, and
18733      * DragDrop in order to get all of the parameters that are available in
18734      * each object.
18735      * @method applyConfig
18736      */
18737     applyConfig: function() {
18738
18739         // configurable properties:
18740         //    padding, isTarget, maintainOffset, primaryButtonOnly
18741         this.padding           = this.config.padding || [0, 0, 0, 0];
18742         this.isTarget          = (this.config.isTarget !== false);
18743         this.maintainOffset    = (this.config.maintainOffset);
18744         this.primaryButtonOnly = (this.config.primaryButtonOnly !== false);
18745
18746     },
18747
18748     /**
18749      * Executed when the linked element is available
18750      * @method handleOnAvailable
18751      * @private
18752      */
18753     handleOnAvailable: function() {
18754         this.available = true;
18755         this.resetConstraints();
18756         this.onAvailable();
18757     },
18758
18759      /**
18760      * Configures the padding for the target zone in px.  Effectively expands
18761      * (or reduces) the virtual object size for targeting calculations.
18762      * Supports css-style shorthand; if only one parameter is passed, all sides
18763      * will have that padding, and if only two are passed, the top and bottom
18764      * will have the first param, the left and right the second.
18765      * @method setPadding
18766      * @param {int} iTop    Top pad
18767      * @param {int} iRight  Right pad
18768      * @param {int} iBot    Bot pad
18769      * @param {int} iLeft   Left pad
18770      */
18771     setPadding: function(iTop, iRight, iBot, iLeft) {
18772         // this.padding = [iLeft, iRight, iTop, iBot];
18773         if (!iRight && 0 !== iRight) {
18774             this.padding = [iTop, iTop, iTop, iTop];
18775         } else if (!iBot && 0 !== iBot) {
18776             this.padding = [iTop, iRight, iTop, iRight];
18777         } else {
18778             this.padding = [iTop, iRight, iBot, iLeft];
18779         }
18780     },
18781
18782     /**
18783      * Stores the initial placement of the linked element.
18784      * @method setInitialPosition
18785      * @param {int} diffX   the X offset, default 0
18786      * @param {int} diffY   the Y offset, default 0
18787      */
18788     setInitPosition: function(diffX, diffY) {
18789         var el = this.getEl();
18790
18791         if (!this.DDM.verifyEl(el)) {
18792             return;
18793         }
18794
18795         var dx = diffX || 0;
18796         var dy = diffY || 0;
18797
18798         var p = Dom.getXY( el );
18799
18800         this.initPageX = p[0] - dx;
18801         this.initPageY = p[1] - dy;
18802
18803         this.lastPageX = p[0];
18804         this.lastPageY = p[1];
18805
18806
18807         this.setStartPosition(p);
18808     },
18809
18810     /**
18811      * Sets the start position of the element.  This is set when the obj
18812      * is initialized, the reset when a drag is started.
18813      * @method setStartPosition
18814      * @param pos current position (from previous lookup)
18815      * @private
18816      */
18817     setStartPosition: function(pos) {
18818         var p = pos || Dom.getXY( this.getEl() );
18819         this.deltaSetXY = null;
18820
18821         this.startPageX = p[0];
18822         this.startPageY = p[1];
18823     },
18824
18825     /**
18826      * Add this instance to a group of related drag/drop objects.  All
18827      * instances belong to at least one group, and can belong to as many
18828      * groups as needed.
18829      * @method addToGroup
18830      * @param sGroup {string} the name of the group
18831      */
18832     addToGroup: function(sGroup) {
18833         this.groups[sGroup] = true;
18834         this.DDM.regDragDrop(this, sGroup);
18835     },
18836
18837     /**
18838      * Remove's this instance from the supplied interaction group
18839      * @method removeFromGroup
18840      * @param {string}  sGroup  The group to drop
18841      */
18842     removeFromGroup: function(sGroup) {
18843         if (this.groups[sGroup]) {
18844             delete this.groups[sGroup];
18845         }
18846
18847         this.DDM.removeDDFromGroup(this, sGroup);
18848     },
18849
18850     /**
18851      * Allows you to specify that an element other than the linked element
18852      * will be moved with the cursor during a drag
18853      * @method setDragElId
18854      * @param id {string} the id of the element that will be used to initiate the drag
18855      */
18856     setDragElId: function(id) {
18857         this.dragElId = id;
18858     },
18859
18860     /**
18861      * Allows you to specify a child of the linked element that should be
18862      * used to initiate the drag operation.  An example of this would be if
18863      * you have a content div with text and links.  Clicking anywhere in the
18864      * content area would normally start the drag operation.  Use this method
18865      * to specify that an element inside of the content div is the element
18866      * that starts the drag operation.
18867      * @method setHandleElId
18868      * @param id {string} the id of the element that will be used to
18869      * initiate the drag.
18870      */
18871     setHandleElId: function(id) {
18872         if (typeof id !== "string") {
18873             id = Roo.id(id);
18874         }
18875         this.handleElId = id;
18876         this.DDM.regHandle(this.id, id);
18877     },
18878
18879     /**
18880      * Allows you to set an element outside of the linked element as a drag
18881      * handle
18882      * @method setOuterHandleElId
18883      * @param id the id of the element that will be used to initiate the drag
18884      */
18885     setOuterHandleElId: function(id) {
18886         if (typeof id !== "string") {
18887             id = Roo.id(id);
18888         }
18889         Event.on(id, "mousedown",
18890                 this.handleMouseDown, this);
18891         this.setHandleElId(id);
18892
18893         this.hasOuterHandles = true;
18894     },
18895
18896     /**
18897      * Remove all drag and drop hooks for this element
18898      * @method unreg
18899      */
18900     unreg: function() {
18901         Event.un(this.id, "mousedown",
18902                 this.handleMouseDown);
18903         Event.un(this.id, "touchstart",
18904                 this.handleMouseDown);
18905         this._domRef = null;
18906         this.DDM._remove(this);
18907     },
18908
18909     destroy : function(){
18910         this.unreg();
18911     },
18912
18913     /**
18914      * Returns true if this instance is locked, or the drag drop mgr is locked
18915      * (meaning that all drag/drop is disabled on the page.)
18916      * @method isLocked
18917      * @return {boolean} true if this obj or all drag/drop is locked, else
18918      * false
18919      */
18920     isLocked: function() {
18921         return (this.DDM.isLocked() || this.locked);
18922     },
18923
18924     /**
18925      * Fired when this object is clicked
18926      * @method handleMouseDown
18927      * @param {Event} e
18928      * @param {Roo.dd.DragDrop} oDD the clicked dd object (this dd obj)
18929      * @private
18930      */
18931     handleMouseDown: function(e, oDD){
18932      
18933         if (!Roo.isTouch && this.primaryButtonOnly && e.button != 0) {
18934             //Roo.log('not touch/ button !=0');
18935             return;
18936         }
18937         if (e.browserEvent.touches && e.browserEvent.touches.length != 1) {
18938             return; // double touch..
18939         }
18940         
18941
18942         if (this.isLocked()) {
18943             //Roo.log('locked');
18944             return;
18945         }
18946
18947         this.DDM.refreshCache(this.groups);
18948 //        Roo.log([Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e)]);
18949         var pt = new Roo.lib.Point(Roo.lib.Event.getPageX(e), Roo.lib.Event.getPageY(e));
18950         if (!this.hasOuterHandles && !this.DDM.isOverTarget(pt, this) )  {
18951             //Roo.log('no outer handes or not over target');
18952                 // do nothing.
18953         } else {
18954 //            Roo.log('check validator');
18955             if (this.clickValidator(e)) {
18956 //                Roo.log('validate success');
18957                 // set the initial element position
18958                 this.setStartPosition();
18959
18960
18961                 this.b4MouseDown(e);
18962                 this.onMouseDown(e);
18963
18964                 this.DDM.handleMouseDown(e, this);
18965
18966                 this.DDM.stopEvent(e);
18967             } else {
18968
18969
18970             }
18971         }
18972     },
18973
18974     clickValidator: function(e) {
18975         var target = e.getTarget();
18976         return ( this.isValidHandleChild(target) &&
18977                     (this.id == this.handleElId ||
18978                         this.DDM.handleWasClicked(target, this.id)) );
18979     },
18980
18981     /**
18982      * Allows you to specify a tag name that should not start a drag operation
18983      * when clicked.  This is designed to facilitate embedding links within a
18984      * drag handle that do something other than start the drag.
18985      * @method addInvalidHandleType
18986      * @param {string} tagName the type of element to exclude
18987      */
18988     addInvalidHandleType: function(tagName) {
18989         var type = tagName.toUpperCase();
18990         this.invalidHandleTypes[type] = type;
18991     },
18992
18993     /**
18994      * Lets you to specify an element id for a child of a drag handle
18995      * that should not initiate a drag
18996      * @method addInvalidHandleId
18997      * @param {string} id the element id of the element you wish to ignore
18998      */
18999     addInvalidHandleId: function(id) {
19000         if (typeof id !== "string") {
19001             id = Roo.id(id);
19002         }
19003         this.invalidHandleIds[id] = id;
19004     },
19005
19006     /**
19007      * Lets you specify a css class of elements that will not initiate a drag
19008      * @method addInvalidHandleClass
19009      * @param {string} cssClass the class of the elements you wish to ignore
19010      */
19011     addInvalidHandleClass: function(cssClass) {
19012         this.invalidHandleClasses.push(cssClass);
19013     },
19014
19015     /**
19016      * Unsets an excluded tag name set by addInvalidHandleType
19017      * @method removeInvalidHandleType
19018      * @param {string} tagName the type of element to unexclude
19019      */
19020     removeInvalidHandleType: function(tagName) {
19021         var type = tagName.toUpperCase();
19022         // this.invalidHandleTypes[type] = null;
19023         delete this.invalidHandleTypes[type];
19024     },
19025
19026     /**
19027      * Unsets an invalid handle id
19028      * @method removeInvalidHandleId
19029      * @param {string} id the id of the element to re-enable
19030      */
19031     removeInvalidHandleId: function(id) {
19032         if (typeof id !== "string") {
19033             id = Roo.id(id);
19034         }
19035         delete this.invalidHandleIds[id];
19036     },
19037
19038     /**
19039      * Unsets an invalid css class
19040      * @method removeInvalidHandleClass
19041      * @param {string} cssClass the class of the element(s) you wish to
19042      * re-enable
19043      */
19044     removeInvalidHandleClass: function(cssClass) {
19045         for (var i=0, len=this.invalidHandleClasses.length; i<len; ++i) {
19046             if (this.invalidHandleClasses[i] == cssClass) {
19047                 delete this.invalidHandleClasses[i];
19048             }
19049         }
19050     },
19051
19052     /**
19053      * Checks the tag exclusion list to see if this click should be ignored
19054      * @method isValidHandleChild
19055      * @param {HTMLElement} node the HTMLElement to evaluate
19056      * @return {boolean} true if this is a valid tag type, false if not
19057      */
19058     isValidHandleChild: function(node) {
19059
19060         var valid = true;
19061         // var n = (node.nodeName == "#text") ? node.parentNode : node;
19062         var nodeName;
19063         try {
19064             nodeName = node.nodeName.toUpperCase();
19065         } catch(e) {
19066             nodeName = node.nodeName;
19067         }
19068         valid = valid && !this.invalidHandleTypes[nodeName];
19069         valid = valid && !this.invalidHandleIds[node.id];
19070
19071         for (var i=0, len=this.invalidHandleClasses.length; valid && i<len; ++i) {
19072             valid = !Dom.hasClass(node, this.invalidHandleClasses[i]);
19073         }
19074
19075
19076         return valid;
19077
19078     },
19079
19080     /**
19081      * Create the array of horizontal tick marks if an interval was specified
19082      * in setXConstraint().
19083      * @method setXTicks
19084      * @private
19085      */
19086     setXTicks: function(iStartX, iTickSize) {
19087         this.xTicks = [];
19088         this.xTickSize = iTickSize;
19089
19090         var tickMap = {};
19091
19092         for (var i = this.initPageX; i >= this.minX; i = i - iTickSize) {
19093             if (!tickMap[i]) {
19094                 this.xTicks[this.xTicks.length] = i;
19095                 tickMap[i] = true;
19096             }
19097         }
19098
19099         for (i = this.initPageX; i <= this.maxX; i = i + iTickSize) {
19100             if (!tickMap[i]) {
19101                 this.xTicks[this.xTicks.length] = i;
19102                 tickMap[i] = true;
19103             }
19104         }
19105
19106         this.xTicks.sort(this.DDM.numericSort) ;
19107     },
19108
19109     /**
19110      * Create the array of vertical tick marks if an interval was specified in
19111      * setYConstraint().
19112      * @method setYTicks
19113      * @private
19114      */
19115     setYTicks: function(iStartY, iTickSize) {
19116         this.yTicks = [];
19117         this.yTickSize = iTickSize;
19118
19119         var tickMap = {};
19120
19121         for (var i = this.initPageY; i >= this.minY; i = i - iTickSize) {
19122             if (!tickMap[i]) {
19123                 this.yTicks[this.yTicks.length] = i;
19124                 tickMap[i] = true;
19125             }
19126         }
19127
19128         for (i = this.initPageY; i <= this.maxY; i = i + iTickSize) {
19129             if (!tickMap[i]) {
19130                 this.yTicks[this.yTicks.length] = i;
19131                 tickMap[i] = true;
19132             }
19133         }
19134
19135         this.yTicks.sort(this.DDM.numericSort) ;
19136     },
19137
19138     /**
19139      * By default, the element can be dragged any place on the screen.  Use
19140      * this method to limit the horizontal travel of the element.  Pass in
19141      * 0,0 for the parameters if you want to lock the drag to the y axis.
19142      * @method setXConstraint
19143      * @param {int} iLeft the number of pixels the element can move to the left
19144      * @param {int} iRight the number of pixels the element can move to the
19145      * right
19146      * @param {int} iTickSize optional parameter for specifying that the
19147      * element
19148      * should move iTickSize pixels at a time.
19149      */
19150     setXConstraint: function(iLeft, iRight, iTickSize) {
19151         this.leftConstraint = iLeft;
19152         this.rightConstraint = iRight;
19153
19154         this.minX = this.initPageX - iLeft;
19155         this.maxX = this.initPageX + iRight;
19156         if (iTickSize) { this.setXTicks(this.initPageX, iTickSize); }
19157
19158         this.constrainX = true;
19159     },
19160
19161     /**
19162      * Clears any constraints applied to this instance.  Also clears ticks
19163      * since they can't exist independent of a constraint at this time.
19164      * @method clearConstraints
19165      */
19166     clearConstraints: function() {
19167         this.constrainX = false;
19168         this.constrainY = false;
19169         this.clearTicks();
19170     },
19171
19172     /**
19173      * Clears any tick interval defined for this instance
19174      * @method clearTicks
19175      */
19176     clearTicks: function() {
19177         this.xTicks = null;
19178         this.yTicks = null;
19179         this.xTickSize = 0;
19180         this.yTickSize = 0;
19181     },
19182
19183     /**
19184      * By default, the element can be dragged any place on the screen.  Set
19185      * this to limit the vertical travel of the element.  Pass in 0,0 for the
19186      * parameters if you want to lock the drag to the x axis.
19187      * @method setYConstraint
19188      * @param {int} iUp the number of pixels the element can move up
19189      * @param {int} iDown the number of pixels the element can move down
19190      * @param {int} iTickSize optional parameter for specifying that the
19191      * element should move iTickSize pixels at a time.
19192      */
19193     setYConstraint: function(iUp, iDown, iTickSize) {
19194         this.topConstraint = iUp;
19195         this.bottomConstraint = iDown;
19196
19197         this.minY = this.initPageY - iUp;
19198         this.maxY = this.initPageY + iDown;
19199         if (iTickSize) { this.setYTicks(this.initPageY, iTickSize); }
19200
19201         this.constrainY = true;
19202
19203     },
19204
19205     /**
19206      * resetConstraints must be called if you manually reposition a dd element.
19207      * @method resetConstraints
19208      * @param {boolean} maintainOffset
19209      */
19210     resetConstraints: function() {
19211
19212
19213         // Maintain offsets if necessary
19214         if (this.initPageX || this.initPageX === 0) {
19215             // figure out how much this thing has moved
19216             var dx = (this.maintainOffset) ? this.lastPageX - this.initPageX : 0;
19217             var dy = (this.maintainOffset) ? this.lastPageY - this.initPageY : 0;
19218
19219             this.setInitPosition(dx, dy);
19220
19221         // This is the first time we have detected the element's position
19222         } else {
19223             this.setInitPosition();
19224         }
19225
19226         if (this.constrainX) {
19227             this.setXConstraint( this.leftConstraint,
19228                                  this.rightConstraint,
19229                                  this.xTickSize        );
19230         }
19231
19232         if (this.constrainY) {
19233             this.setYConstraint( this.topConstraint,
19234                                  this.bottomConstraint,
19235                                  this.yTickSize         );
19236         }
19237     },
19238
19239     /**
19240      * Normally the drag element is moved pixel by pixel, but we can specify
19241      * that it move a number of pixels at a time.  This method resolves the
19242      * location when we have it set up like this.
19243      * @method getTick
19244      * @param {int} val where we want to place the object
19245      * @param {int[]} tickArray sorted array of valid points
19246      * @return {int} the closest tick
19247      * @private
19248      */
19249     getTick: function(val, tickArray) {
19250
19251         if (!tickArray) {
19252             // If tick interval is not defined, it is effectively 1 pixel,
19253             // so we return the value passed to us.
19254             return val;
19255         } else if (tickArray[0] >= val) {
19256             // The value is lower than the first tick, so we return the first
19257             // tick.
19258             return tickArray[0];
19259         } else {
19260             for (var i=0, len=tickArray.length; i<len; ++i) {
19261                 var next = i + 1;
19262                 if (tickArray[next] && tickArray[next] >= val) {
19263                     var diff1 = val - tickArray[i];
19264                     var diff2 = tickArray[next] - val;
19265                     return (diff2 > diff1) ? tickArray[i] : tickArray[next];
19266                 }
19267             }
19268
19269             // The value is larger than the last tick, so we return the last
19270             // tick.
19271             return tickArray[tickArray.length - 1];
19272         }
19273     },
19274
19275     /**
19276      * toString method
19277      * @method toString
19278      * @return {string} string representation of the dd obj
19279      */
19280     toString: function() {
19281         return ("DragDrop " + this.id);
19282     }
19283
19284 });
19285
19286 })();
19287 /*
19288  * Based on:
19289  * Ext JS Library 1.1.1
19290  * Copyright(c) 2006-2007, Ext JS, LLC.
19291  *
19292  * Originally Released Under LGPL - original licence link has changed is not relivant.
19293  *
19294  * Fork - LGPL
19295  * <script type="text/javascript">
19296  */
19297
19298
19299 /**
19300  * The drag and drop utility provides a framework for building drag and drop
19301  * applications.  In addition to enabling drag and drop for specific elements,
19302  * the drag and drop elements are tracked by the manager class, and the
19303  * interactions between the various elements are tracked during the drag and
19304  * the implementing code is notified about these important moments.
19305  */
19306
19307 // Only load the library once.  Rewriting the manager class would orphan
19308 // existing drag and drop instances.
19309 if (!Roo.dd.DragDropMgr) {
19310
19311 /**
19312  * @class Roo.dd.DragDropMgr
19313  * DragDropMgr is a singleton that tracks the element interaction for
19314  * all DragDrop items in the window.  Generally, you will not call
19315  * this class directly, but it does have helper methods that could
19316  * be useful in your DragDrop implementations.
19317  * @singleton
19318  */
19319 Roo.dd.DragDropMgr = function() {
19320
19321     var Event = Roo.EventManager;
19322
19323     return {
19324
19325         /**
19326          * Two dimensional Array of registered DragDrop objects.  The first
19327          * dimension is the DragDrop item group, the second the DragDrop
19328          * object.
19329          * @property ids
19330          * @type {string: string}
19331          * @private
19332          * @static
19333          */
19334         ids: {},
19335
19336         /**
19337          * Array of element ids defined as drag handles.  Used to determine
19338          * if the element that generated the mousedown event is actually the
19339          * handle and not the html element itself.
19340          * @property handleIds
19341          * @type {string: string}
19342          * @private
19343          * @static
19344          */
19345         handleIds: {},
19346
19347         /**
19348          * the DragDrop object that is currently being dragged
19349          * @property dragCurrent
19350          * @type DragDrop
19351          * @private
19352          * @static
19353          **/
19354         dragCurrent: null,
19355
19356         /**
19357          * the DragDrop object(s) that are being hovered over
19358          * @property dragOvers
19359          * @type Array
19360          * @private
19361          * @static
19362          */
19363         dragOvers: {},
19364
19365         /**
19366          * the X distance between the cursor and the object being dragged
19367          * @property deltaX
19368          * @type int
19369          * @private
19370          * @static
19371          */
19372         deltaX: 0,
19373
19374         /**
19375          * the Y distance between the cursor and the object being dragged
19376          * @property deltaY
19377          * @type int
19378          * @private
19379          * @static
19380          */
19381         deltaY: 0,
19382
19383         /**
19384          * Flag to determine if we should prevent the default behavior of the
19385          * events we define. By default this is true, but this can be set to
19386          * false if you need the default behavior (not recommended)
19387          * @property preventDefault
19388          * @type boolean
19389          * @static
19390          */
19391         preventDefault: true,
19392
19393         /**
19394          * Flag to determine if we should stop the propagation of the events
19395          * we generate. This is true by default but you may want to set it to
19396          * false if the html element contains other features that require the
19397          * mouse click.
19398          * @property stopPropagation
19399          * @type boolean
19400          * @static
19401          */
19402         stopPropagation: true,
19403
19404         /**
19405          * Internal flag that is set to true when drag and drop has been
19406          * intialized
19407          * @property initialized
19408          * @private
19409          * @static
19410          */
19411         initalized: false,
19412
19413         /**
19414          * All drag and drop can be disabled.
19415          * @property locked
19416          * @private
19417          * @static
19418          */
19419         locked: false,
19420
19421         /**
19422          * Called the first time an element is registered.
19423          * @method init
19424          * @private
19425          * @static
19426          */
19427         init: function() {
19428             this.initialized = true;
19429         },
19430
19431         /**
19432          * In point mode, drag and drop interaction is defined by the
19433          * location of the cursor during the drag/drop
19434          * @property POINT
19435          * @type int
19436          * @static
19437          */
19438         POINT: 0,
19439
19440         /**
19441          * In intersect mode, drag and drop interactio nis defined by the
19442          * overlap of two or more drag and drop objects.
19443          * @property INTERSECT
19444          * @type int
19445          * @static
19446          */
19447         INTERSECT: 1,
19448
19449         /**
19450          * The current drag and drop mode.  Default: POINT
19451          * @property mode
19452          * @type int
19453          * @static
19454          */
19455         mode: 0,
19456
19457         /**
19458          * Runs method on all drag and drop objects
19459          * @method _execOnAll
19460          * @private
19461          * @static
19462          */
19463         _execOnAll: function(sMethod, args) {
19464             for (var i in this.ids) {
19465                 for (var j in this.ids[i]) {
19466                     var oDD = this.ids[i][j];
19467                     if (! this.isTypeOfDD(oDD)) {
19468                         continue;
19469                     }
19470                     oDD[sMethod].apply(oDD, args);
19471                 }
19472             }
19473         },
19474
19475         /**
19476          * Drag and drop initialization.  Sets up the global event handlers
19477          * @method _onLoad
19478          * @private
19479          * @static
19480          */
19481         _onLoad: function() {
19482
19483             this.init();
19484
19485             if (!Roo.isTouch) {
19486                 Event.on(document, "mouseup",   this.handleMouseUp, this, true);
19487                 Event.on(document, "mousemove", this.handleMouseMove, this, true);
19488             }
19489             Event.on(document, "touchend",   this.handleMouseUp, this, true);
19490             Event.on(document, "touchmove", this.handleMouseMove, this, true);
19491             
19492             Event.on(window,   "unload",    this._onUnload, this, true);
19493             Event.on(window,   "resize",    this._onResize, this, true);
19494             // Event.on(window,   "mouseout",    this._test);
19495
19496         },
19497
19498         /**
19499          * Reset constraints on all drag and drop objs
19500          * @method _onResize
19501          * @private
19502          * @static
19503          */
19504         _onResize: function(e) {
19505             this._execOnAll("resetConstraints", []);
19506         },
19507
19508         /**
19509          * Lock all drag and drop functionality
19510          * @method lock
19511          * @static
19512          */
19513         lock: function() { this.locked = true; },
19514
19515         /**
19516          * Unlock all drag and drop functionality
19517          * @method unlock
19518          * @static
19519          */
19520         unlock: function() { this.locked = false; },
19521
19522         /**
19523          * Is drag and drop locked?
19524          * @method isLocked
19525          * @return {boolean} True if drag and drop is locked, false otherwise.
19526          * @static
19527          */
19528         isLocked: function() { return this.locked; },
19529
19530         /**
19531          * Location cache that is set for all drag drop objects when a drag is
19532          * initiated, cleared when the drag is finished.
19533          * @property locationCache
19534          * @private
19535          * @static
19536          */
19537         locationCache: {},
19538
19539         /**
19540          * Set useCache to false if you want to force object the lookup of each
19541          * drag and drop linked element constantly during a drag.
19542          * @property useCache
19543          * @type boolean
19544          * @static
19545          */
19546         useCache: true,
19547
19548         /**
19549          * The number of pixels that the mouse needs to move after the
19550          * mousedown before the drag is initiated.  Default=3;
19551          * @property clickPixelThresh
19552          * @type int
19553          * @static
19554          */
19555         clickPixelThresh: 3,
19556
19557         /**
19558          * The number of milliseconds after the mousedown event to initiate the
19559          * drag if we don't get a mouseup event. Default=1000
19560          * @property clickTimeThresh
19561          * @type int
19562          * @static
19563          */
19564         clickTimeThresh: 350,
19565
19566         /**
19567          * Flag that indicates that either the drag pixel threshold or the
19568          * mousdown time threshold has been met
19569          * @property dragThreshMet
19570          * @type boolean
19571          * @private
19572          * @static
19573          */
19574         dragThreshMet: false,
19575
19576         /**
19577          * Timeout used for the click time threshold
19578          * @property clickTimeout
19579          * @type Object
19580          * @private
19581          * @static
19582          */
19583         clickTimeout: null,
19584
19585         /**
19586          * The X position of the mousedown event stored for later use when a
19587          * drag threshold is met.
19588          * @property startX
19589          * @type int
19590          * @private
19591          * @static
19592          */
19593         startX: 0,
19594
19595         /**
19596          * The Y position of the mousedown event stored for later use when a
19597          * drag threshold is met.
19598          * @property startY
19599          * @type int
19600          * @private
19601          * @static
19602          */
19603         startY: 0,
19604
19605         /**
19606          * Each DragDrop instance must be registered with the DragDropMgr.
19607          * This is executed in DragDrop.init()
19608          * @method regDragDrop
19609          * @param {DragDrop} oDD the DragDrop object to register
19610          * @param {String} sGroup the name of the group this element belongs to
19611          * @static
19612          */
19613         regDragDrop: function(oDD, sGroup) {
19614             if (!this.initialized) { this.init(); }
19615
19616             if (!this.ids[sGroup]) {
19617                 this.ids[sGroup] = {};
19618             }
19619             this.ids[sGroup][oDD.id] = oDD;
19620         },
19621
19622         /**
19623          * Removes the supplied dd instance from the supplied group. Executed
19624          * by DragDrop.removeFromGroup, so don't call this function directly.
19625          * @method removeDDFromGroup
19626          * @private
19627          * @static
19628          */
19629         removeDDFromGroup: function(oDD, sGroup) {
19630             if (!this.ids[sGroup]) {
19631                 this.ids[sGroup] = {};
19632             }
19633
19634             var obj = this.ids[sGroup];
19635             if (obj && obj[oDD.id]) {
19636                 delete obj[oDD.id];
19637             }
19638         },
19639
19640         /**
19641          * Unregisters a drag and drop item.  This is executed in
19642          * DragDrop.unreg, use that method instead of calling this directly.
19643          * @method _remove
19644          * @private
19645          * @static
19646          */
19647         _remove: function(oDD) {
19648             for (var g in oDD.groups) {
19649                 if (g && this.ids[g][oDD.id]) {
19650                     delete this.ids[g][oDD.id];
19651                 }
19652             }
19653             delete this.handleIds[oDD.id];
19654         },
19655
19656         /**
19657          * Each DragDrop handle element must be registered.  This is done
19658          * automatically when executing DragDrop.setHandleElId()
19659          * @method regHandle
19660          * @param {String} sDDId the DragDrop id this element is a handle for
19661          * @param {String} sHandleId the id of the element that is the drag
19662          * handle
19663          * @static
19664          */
19665         regHandle: function(sDDId, sHandleId) {
19666             if (!this.handleIds[sDDId]) {
19667                 this.handleIds[sDDId] = {};
19668             }
19669             this.handleIds[sDDId][sHandleId] = sHandleId;
19670         },
19671
19672         /**
19673          * Utility function to determine if a given element has been
19674          * registered as a drag drop item.
19675          * @method isDragDrop
19676          * @param {String} id the element id to check
19677          * @return {boolean} true if this element is a DragDrop item,
19678          * false otherwise
19679          * @static
19680          */
19681         isDragDrop: function(id) {
19682             return ( this.getDDById(id) ) ? true : false;
19683         },
19684
19685         /**
19686          * Returns the drag and drop instances that are in all groups the
19687          * passed in instance belongs to.
19688          * @method getRelated
19689          * @param {DragDrop} p_oDD the obj to get related data for
19690          * @param {boolean} bTargetsOnly if true, only return targetable objs
19691          * @return {DragDrop[]} the related instances
19692          * @static
19693          */
19694         getRelated: function(p_oDD, bTargetsOnly) {
19695             var oDDs = [];
19696             for (var i in p_oDD.groups) {
19697                 for (j in this.ids[i]) {
19698                     var dd = this.ids[i][j];
19699                     if (! this.isTypeOfDD(dd)) {
19700                         continue;
19701                     }
19702                     if (!bTargetsOnly || dd.isTarget) {
19703                         oDDs[oDDs.length] = dd;
19704                     }
19705                 }
19706             }
19707
19708             return oDDs;
19709         },
19710
19711         /**
19712          * Returns true if the specified dd target is a legal target for
19713          * the specifice drag obj
19714          * @method isLegalTarget
19715          * @param {DragDrop} the drag obj
19716          * @param {DragDrop} the target
19717          * @return {boolean} true if the target is a legal target for the
19718          * dd obj
19719          * @static
19720          */
19721         isLegalTarget: function (oDD, oTargetDD) {
19722             var targets = this.getRelated(oDD, true);
19723             for (var i=0, len=targets.length;i<len;++i) {
19724                 if (targets[i].id == oTargetDD.id) {
19725                     return true;
19726                 }
19727             }
19728
19729             return false;
19730         },
19731
19732         /**
19733          * My goal is to be able to transparently determine if an object is
19734          * typeof DragDrop, and the exact subclass of DragDrop.  typeof
19735          * returns "object", oDD.constructor.toString() always returns
19736          * "DragDrop" and not the name of the subclass.  So for now it just
19737          * evaluates a well-known variable in DragDrop.
19738          * @method isTypeOfDD
19739          * @param {Object} the object to evaluate
19740          * @return {boolean} true if typeof oDD = DragDrop
19741          * @static
19742          */
19743         isTypeOfDD: function (oDD) {
19744             return (oDD && oDD.__ygDragDrop);
19745         },
19746
19747         /**
19748          * Utility function to determine if a given element has been
19749          * registered as a drag drop handle for the given Drag Drop object.
19750          * @method isHandle
19751          * @param {String} id the element id to check
19752          * @return {boolean} true if this element is a DragDrop handle, false
19753          * otherwise
19754          * @static
19755          */
19756         isHandle: function(sDDId, sHandleId) {
19757             return ( this.handleIds[sDDId] &&
19758                             this.handleIds[sDDId][sHandleId] );
19759         },
19760
19761         /**
19762          * Returns the DragDrop instance for a given id
19763          * @method getDDById
19764          * @param {String} id the id of the DragDrop object
19765          * @return {DragDrop} the drag drop object, null if it is not found
19766          * @static
19767          */
19768         getDDById: function(id) {
19769             for (var i in this.ids) {
19770                 if (this.ids[i][id]) {
19771                     return this.ids[i][id];
19772                 }
19773             }
19774             return null;
19775         },
19776
19777         /**
19778          * Fired after a registered DragDrop object gets the mousedown event.
19779          * Sets up the events required to track the object being dragged
19780          * @method handleMouseDown
19781          * @param {Event} e the event
19782          * @param oDD the DragDrop object being dragged
19783          * @private
19784          * @static
19785          */
19786         handleMouseDown: function(e, oDD) {
19787             if(Roo.QuickTips){
19788                 Roo.QuickTips.disable();
19789             }
19790             this.currentTarget = e.getTarget();
19791
19792             this.dragCurrent = oDD;
19793
19794             var el = oDD.getEl();
19795
19796             // track start position
19797             this.startX = e.getPageX();
19798             this.startY = e.getPageY();
19799
19800             this.deltaX = this.startX - el.offsetLeft;
19801             this.deltaY = this.startY - el.offsetTop;
19802
19803             this.dragThreshMet = false;
19804
19805             this.clickTimeout = setTimeout(
19806                     function() {
19807                         var DDM = Roo.dd.DDM;
19808                         DDM.startDrag(DDM.startX, DDM.startY);
19809                     },
19810                     this.clickTimeThresh );
19811         },
19812
19813         /**
19814          * Fired when either the drag pixel threshol or the mousedown hold
19815          * time threshold has been met.
19816          * @method startDrag
19817          * @param x {int} the X position of the original mousedown
19818          * @param y {int} the Y position of the original mousedown
19819          * @static
19820          */
19821         startDrag: function(x, y) {
19822             clearTimeout(this.clickTimeout);
19823             if (this.dragCurrent) {
19824                 this.dragCurrent.b4StartDrag(x, y);
19825                 this.dragCurrent.startDrag(x, y);
19826             }
19827             this.dragThreshMet = true;
19828         },
19829
19830         /**
19831          * Internal function to handle the mouseup event.  Will be invoked
19832          * from the context of the document.
19833          * @method handleMouseUp
19834          * @param {Event} e the event
19835          * @private
19836          * @static
19837          */
19838         handleMouseUp: function(e) {
19839
19840             if(Roo.QuickTips){
19841                 Roo.QuickTips.enable();
19842             }
19843             if (! this.dragCurrent) {
19844                 return;
19845             }
19846
19847             clearTimeout(this.clickTimeout);
19848
19849             if (this.dragThreshMet) {
19850                 this.fireEvents(e, true);
19851             } else {
19852             }
19853
19854             this.stopDrag(e);
19855
19856             this.stopEvent(e);
19857         },
19858
19859         /**
19860          * Utility to stop event propagation and event default, if these
19861          * features are turned on.
19862          * @method stopEvent
19863          * @param {Event} e the event as returned by this.getEvent()
19864          * @static
19865          */
19866         stopEvent: function(e){
19867             if(this.stopPropagation) {
19868                 e.stopPropagation();
19869             }
19870
19871             if (this.preventDefault) {
19872                 e.preventDefault();
19873             }
19874         },
19875
19876         /**
19877          * Internal function to clean up event handlers after the drag
19878          * operation is complete
19879          * @method stopDrag
19880          * @param {Event} e the event
19881          * @private
19882          * @static
19883          */
19884         stopDrag: function(e) {
19885             // Fire the drag end event for the item that was dragged
19886             if (this.dragCurrent) {
19887                 if (this.dragThreshMet) {
19888                     this.dragCurrent.b4EndDrag(e);
19889                     this.dragCurrent.endDrag(e);
19890                 }
19891
19892                 this.dragCurrent.onMouseUp(e);
19893             }
19894
19895             this.dragCurrent = null;
19896             this.dragOvers = {};
19897         },
19898
19899         /**
19900          * Internal function to handle the mousemove event.  Will be invoked
19901          * from the context of the html element.
19902          *
19903          * @TODO figure out what we can do about mouse events lost when the
19904          * user drags objects beyond the window boundary.  Currently we can
19905          * detect this in internet explorer by verifying that the mouse is
19906          * down during the mousemove event.  Firefox doesn't give us the
19907          * button state on the mousemove event.
19908          * @method handleMouseMove
19909          * @param {Event} e the event
19910          * @private
19911          * @static
19912          */
19913         handleMouseMove: function(e) {
19914             if (! this.dragCurrent) {
19915                 return true;
19916             }
19917
19918             // var button = e.which || e.button;
19919
19920             // check for IE mouseup outside of page boundary
19921             if (Roo.isIE && (e.button !== 0 && e.button !== 1 && e.button !== 2)) {
19922                 this.stopEvent(e);
19923                 return this.handleMouseUp(e);
19924             }
19925
19926             if (!this.dragThreshMet) {
19927                 var diffX = Math.abs(this.startX - e.getPageX());
19928                 var diffY = Math.abs(this.startY - e.getPageY());
19929                 if (diffX > this.clickPixelThresh ||
19930                             diffY > this.clickPixelThresh) {
19931                     this.startDrag(this.startX, this.startY);
19932                 }
19933             }
19934
19935             if (this.dragThreshMet) {
19936                 this.dragCurrent.b4Drag(e);
19937                 this.dragCurrent.onDrag(e);
19938                 if(!this.dragCurrent.moveOnly){
19939                     this.fireEvents(e, false);
19940                 }
19941             }
19942
19943             this.stopEvent(e);
19944
19945             return true;
19946         },
19947
19948         /**
19949          * Iterates over all of the DragDrop elements to find ones we are
19950          * hovering over or dropping on
19951          * @method fireEvents
19952          * @param {Event} e the event
19953          * @param {boolean} isDrop is this a drop op or a mouseover op?
19954          * @private
19955          * @static
19956          */
19957         fireEvents: function(e, isDrop) {
19958             var dc = this.dragCurrent;
19959
19960             // If the user did the mouse up outside of the window, we could
19961             // get here even though we have ended the drag.
19962             if (!dc || dc.isLocked()) {
19963                 return;
19964             }
19965
19966             var pt = e.getPoint();
19967
19968             // cache the previous dragOver array
19969             var oldOvers = [];
19970
19971             var outEvts   = [];
19972             var overEvts  = [];
19973             var dropEvts  = [];
19974             var enterEvts = [];
19975
19976             // Check to see if the object(s) we were hovering over is no longer
19977             // being hovered over so we can fire the onDragOut event
19978             for (var i in this.dragOvers) {
19979
19980                 var ddo = this.dragOvers[i];
19981
19982                 if (! this.isTypeOfDD(ddo)) {
19983                     continue;
19984                 }
19985
19986                 if (! this.isOverTarget(pt, ddo, this.mode)) {
19987                     outEvts.push( ddo );
19988                 }
19989
19990                 oldOvers[i] = true;
19991                 delete this.dragOvers[i];
19992             }
19993
19994             for (var sGroup in dc.groups) {
19995
19996                 if ("string" != typeof sGroup) {
19997                     continue;
19998                 }
19999
20000                 for (i in this.ids[sGroup]) {
20001                     var oDD = this.ids[sGroup][i];
20002                     if (! this.isTypeOfDD(oDD)) {
20003                         continue;
20004                     }
20005
20006                     if (oDD.isTarget && !oDD.isLocked() && oDD != dc) {
20007                         if (this.isOverTarget(pt, oDD, this.mode)) {
20008                             // look for drop interactions
20009                             if (isDrop) {
20010                                 dropEvts.push( oDD );
20011                             // look for drag enter and drag over interactions
20012                             } else {
20013
20014                                 // initial drag over: dragEnter fires
20015                                 if (!oldOvers[oDD.id]) {
20016                                     enterEvts.push( oDD );
20017                                 // subsequent drag overs: dragOver fires
20018                                 } else {
20019                                     overEvts.push( oDD );
20020                                 }
20021
20022                                 this.dragOvers[oDD.id] = oDD;
20023                             }
20024                         }
20025                     }
20026                 }
20027             }
20028
20029             if (this.mode) {
20030                 if (outEvts.length) {
20031                     dc.b4DragOut(e, outEvts);
20032                     dc.onDragOut(e, outEvts);
20033                 }
20034
20035                 if (enterEvts.length) {
20036                     dc.onDragEnter(e, enterEvts);
20037                 }
20038
20039                 if (overEvts.length) {
20040                     dc.b4DragOver(e, overEvts);
20041                     dc.onDragOver(e, overEvts);
20042                 }
20043
20044                 if (dropEvts.length) {
20045                     dc.b4DragDrop(e, dropEvts);
20046                     dc.onDragDrop(e, dropEvts);
20047                 }
20048
20049             } else {
20050                 // fire dragout events
20051                 var len = 0;
20052                 for (i=0, len=outEvts.length; i<len; ++i) {
20053                     dc.b4DragOut(e, outEvts[i].id);
20054                     dc.onDragOut(e, outEvts[i].id);
20055                 }
20056
20057                 // fire enter events
20058                 for (i=0,len=enterEvts.length; i<len; ++i) {
20059                     // dc.b4DragEnter(e, oDD.id);
20060                     dc.onDragEnter(e, enterEvts[i].id);
20061                 }
20062
20063                 // fire over events
20064                 for (i=0,len=overEvts.length; i<len; ++i) {
20065                     dc.b4DragOver(e, overEvts[i].id);
20066                     dc.onDragOver(e, overEvts[i].id);
20067                 }
20068
20069                 // fire drop events
20070                 for (i=0, len=dropEvts.length; i<len; ++i) {
20071                     dc.b4DragDrop(e, dropEvts[i].id);
20072                     dc.onDragDrop(e, dropEvts[i].id);
20073                 }
20074
20075             }
20076
20077             // notify about a drop that did not find a target
20078             if (isDrop && !dropEvts.length) {
20079                 dc.onInvalidDrop(e);
20080             }
20081
20082         },
20083
20084         /**
20085          * Helper function for getting the best match from the list of drag
20086          * and drop objects returned by the drag and drop events when we are
20087          * in INTERSECT mode.  It returns either the first object that the
20088          * cursor is over, or the object that has the greatest overlap with
20089          * the dragged element.
20090          * @method getBestMatch
20091          * @param  {DragDrop[]} dds The array of drag and drop objects
20092          * targeted
20093          * @return {DragDrop}       The best single match
20094          * @static
20095          */
20096         getBestMatch: function(dds) {
20097             var winner = null;
20098             // Return null if the input is not what we expect
20099             //if (!dds || !dds.length || dds.length == 0) {
20100                // winner = null;
20101             // If there is only one item, it wins
20102             //} else if (dds.length == 1) {
20103
20104             var len = dds.length;
20105
20106             if (len == 1) {
20107                 winner = dds[0];
20108             } else {
20109                 // Loop through the targeted items
20110                 for (var i=0; i<len; ++i) {
20111                     var dd = dds[i];
20112                     // If the cursor is over the object, it wins.  If the
20113                     // cursor is over multiple matches, the first one we come
20114                     // to wins.
20115                     if (dd.cursorIsOver) {
20116                         winner = dd;
20117                         break;
20118                     // Otherwise the object with the most overlap wins
20119                     } else {
20120                         if (!winner ||
20121                             winner.overlap.getArea() < dd.overlap.getArea()) {
20122                             winner = dd;
20123                         }
20124                     }
20125                 }
20126             }
20127
20128             return winner;
20129         },
20130
20131         /**
20132          * Refreshes the cache of the top-left and bottom-right points of the
20133          * drag and drop objects in the specified group(s).  This is in the
20134          * format that is stored in the drag and drop instance, so typical
20135          * usage is:
20136          * <code>
20137          * Roo.dd.DragDropMgr.refreshCache(ddinstance.groups);
20138          * </code>
20139          * Alternatively:
20140          * <code>
20141          * Roo.dd.DragDropMgr.refreshCache({group1:true, group2:true});
20142          * </code>
20143          * @TODO this really should be an indexed array.  Alternatively this
20144          * method could accept both.
20145          * @method refreshCache
20146          * @param {Object} groups an associative array of groups to refresh
20147          * @static
20148          */
20149         refreshCache: function(groups) {
20150             for (var sGroup in groups) {
20151                 if ("string" != typeof sGroup) {
20152                     continue;
20153                 }
20154                 for (var i in this.ids[sGroup]) {
20155                     var oDD = this.ids[sGroup][i];
20156
20157                     if (this.isTypeOfDD(oDD)) {
20158                     // if (this.isTypeOfDD(oDD) && oDD.isTarget) {
20159                         var loc = this.getLocation(oDD);
20160                         if (loc) {
20161                             this.locationCache[oDD.id] = loc;
20162                         } else {
20163                             delete this.locationCache[oDD.id];
20164                             // this will unregister the drag and drop object if
20165                             // the element is not in a usable state
20166                             // oDD.unreg();
20167                         }
20168                     }
20169                 }
20170             }
20171         },
20172
20173         /**
20174          * This checks to make sure an element exists and is in the DOM.  The
20175          * main purpose is to handle cases where innerHTML is used to remove
20176          * drag and drop objects from the DOM.  IE provides an 'unspecified
20177          * error' when trying to access the offsetParent of such an element
20178          * @method verifyEl
20179          * @param {HTMLElement} el the element to check
20180          * @return {boolean} true if the element looks usable
20181          * @static
20182          */
20183         verifyEl: function(el) {
20184             if (el) {
20185                 var parent;
20186                 if(Roo.isIE){
20187                     try{
20188                         parent = el.offsetParent;
20189                     }catch(e){}
20190                 }else{
20191                     parent = el.offsetParent;
20192                 }
20193                 if (parent) {
20194                     return true;
20195                 }
20196             }
20197
20198             return false;
20199         },
20200
20201         /**
20202          * Returns a Region object containing the drag and drop element's position
20203          * and size, including the padding configured for it
20204          * @method getLocation
20205          * @param {DragDrop} oDD the drag and drop object to get the
20206          *                       location for
20207          * @return {Roo.lib.Region} a Region object representing the total area
20208          *                             the element occupies, including any padding
20209          *                             the instance is configured for.
20210          * @static
20211          */
20212         getLocation: function(oDD) {
20213             if (! this.isTypeOfDD(oDD)) {
20214                 return null;
20215             }
20216
20217             var el = oDD.getEl(), pos, x1, x2, y1, y2, t, r, b, l;
20218
20219             try {
20220                 pos= Roo.lib.Dom.getXY(el);
20221             } catch (e) { }
20222
20223             if (!pos) {
20224                 return null;
20225             }
20226
20227             x1 = pos[0];
20228             x2 = x1 + el.offsetWidth;
20229             y1 = pos[1];
20230             y2 = y1 + el.offsetHeight;
20231
20232             t = y1 - oDD.padding[0];
20233             r = x2 + oDD.padding[1];
20234             b = y2 + oDD.padding[2];
20235             l = x1 - oDD.padding[3];
20236
20237             return new Roo.lib.Region( t, r, b, l );
20238         },
20239
20240         /**
20241          * Checks the cursor location to see if it over the target
20242          * @method isOverTarget
20243          * @param {Roo.lib.Point} pt The point to evaluate
20244          * @param {DragDrop} oTarget the DragDrop object we are inspecting
20245          * @return {boolean} true if the mouse is over the target
20246          * @private
20247          * @static
20248          */
20249         isOverTarget: function(pt, oTarget, intersect) {
20250             // use cache if available
20251             var loc = this.locationCache[oTarget.id];
20252             if (!loc || !this.useCache) {
20253                 loc = this.getLocation(oTarget);
20254                 this.locationCache[oTarget.id] = loc;
20255
20256             }
20257
20258             if (!loc) {
20259                 return false;
20260             }
20261
20262             oTarget.cursorIsOver = loc.contains( pt );
20263
20264             // DragDrop is using this as a sanity check for the initial mousedown
20265             // in this case we are done.  In POINT mode, if the drag obj has no
20266             // contraints, we are also done. Otherwise we need to evaluate the
20267             // location of the target as related to the actual location of the
20268             // dragged element.
20269             var dc = this.dragCurrent;
20270             if (!dc || !dc.getTargetCoord ||
20271                     (!intersect && !dc.constrainX && !dc.constrainY)) {
20272                 return oTarget.cursorIsOver;
20273             }
20274
20275             oTarget.overlap = null;
20276
20277             // Get the current location of the drag element, this is the
20278             // location of the mouse event less the delta that represents
20279             // where the original mousedown happened on the element.  We
20280             // need to consider constraints and ticks as well.
20281             var pos = dc.getTargetCoord(pt.x, pt.y);
20282
20283             var el = dc.getDragEl();
20284             var curRegion = new Roo.lib.Region( pos.y,
20285                                                    pos.x + el.offsetWidth,
20286                                                    pos.y + el.offsetHeight,
20287                                                    pos.x );
20288
20289             var overlap = curRegion.intersect(loc);
20290
20291             if (overlap) {
20292                 oTarget.overlap = overlap;
20293                 return (intersect) ? true : oTarget.cursorIsOver;
20294             } else {
20295                 return false;
20296             }
20297         },
20298
20299         /**
20300          * unload event handler
20301          * @method _onUnload
20302          * @private
20303          * @static
20304          */
20305         _onUnload: function(e, me) {
20306             Roo.dd.DragDropMgr.unregAll();
20307         },
20308
20309         /**
20310          * Cleans up the drag and drop events and objects.
20311          * @method unregAll
20312          * @private
20313          * @static
20314          */
20315         unregAll: function() {
20316
20317             if (this.dragCurrent) {
20318                 this.stopDrag();
20319                 this.dragCurrent = null;
20320             }
20321
20322             this._execOnAll("unreg", []);
20323
20324             for (i in this.elementCache) {
20325                 delete this.elementCache[i];
20326             }
20327
20328             this.elementCache = {};
20329             this.ids = {};
20330         },
20331
20332         /**
20333          * A cache of DOM elements
20334          * @property elementCache
20335          * @private
20336          * @static
20337          */
20338         elementCache: {},
20339
20340         /**
20341          * Get the wrapper for the DOM element specified
20342          * @method getElWrapper
20343          * @param {String} id the id of the element to get
20344          * @return {Roo.dd.DDM.ElementWrapper} the wrapped element
20345          * @private
20346          * @deprecated This wrapper isn't that useful
20347          * @static
20348          */
20349         getElWrapper: function(id) {
20350             var oWrapper = this.elementCache[id];
20351             if (!oWrapper || !oWrapper.el) {
20352                 oWrapper = this.elementCache[id] =
20353                     new this.ElementWrapper(Roo.getDom(id));
20354             }
20355             return oWrapper;
20356         },
20357
20358         /**
20359          * Returns the actual DOM element
20360          * @method getElement
20361          * @param {String} id the id of the elment to get
20362          * @return {Object} The element
20363          * @deprecated use Roo.getDom instead
20364          * @static
20365          */
20366         getElement: function(id) {
20367             return Roo.getDom(id);
20368         },
20369
20370         /**
20371          * Returns the style property for the DOM element (i.e.,
20372          * document.getElById(id).style)
20373          * @method getCss
20374          * @param {String} id the id of the elment to get
20375          * @return {Object} The style property of the element
20376          * @deprecated use Roo.getDom instead
20377          * @static
20378          */
20379         getCss: function(id) {
20380             var el = Roo.getDom(id);
20381             return (el) ? el.style : null;
20382         },
20383
20384         /**
20385          * Inner class for cached elements
20386          * @class DragDropMgr.ElementWrapper
20387          * @for DragDropMgr
20388          * @private
20389          * @deprecated
20390          */
20391         ElementWrapper: function(el) {
20392                 /**
20393                  * The element
20394                  * @property el
20395                  */
20396                 this.el = el || null;
20397                 /**
20398                  * The element id
20399                  * @property id
20400                  */
20401                 this.id = this.el && el.id;
20402                 /**
20403                  * A reference to the style property
20404                  * @property css
20405                  */
20406                 this.css = this.el && el.style;
20407             },
20408
20409         /**
20410          * Returns the X position of an html element
20411          * @method getPosX
20412          * @param el the element for which to get the position
20413          * @return {int} the X coordinate
20414          * @for DragDropMgr
20415          * @deprecated use Roo.lib.Dom.getX instead
20416          * @static
20417          */
20418         getPosX: function(el) {
20419             return Roo.lib.Dom.getX(el);
20420         },
20421
20422         /**
20423          * Returns the Y position of an html element
20424          * @method getPosY
20425          * @param el the element for which to get the position
20426          * @return {int} the Y coordinate
20427          * @deprecated use Roo.lib.Dom.getY instead
20428          * @static
20429          */
20430         getPosY: function(el) {
20431             return Roo.lib.Dom.getY(el);
20432         },
20433
20434         /**
20435          * Swap two nodes.  In IE, we use the native method, for others we
20436          * emulate the IE behavior
20437          * @method swapNode
20438          * @param n1 the first node to swap
20439          * @param n2 the other node to swap
20440          * @static
20441          */
20442         swapNode: function(n1, n2) {
20443             if (n1.swapNode) {
20444                 n1.swapNode(n2);
20445             } else {
20446                 var p = n2.parentNode;
20447                 var s = n2.nextSibling;
20448
20449                 if (s == n1) {
20450                     p.insertBefore(n1, n2);
20451                 } else if (n2 == n1.nextSibling) {
20452                     p.insertBefore(n2, n1);
20453                 } else {
20454                     n1.parentNode.replaceChild(n2, n1);
20455                     p.insertBefore(n1, s);
20456                 }
20457             }
20458         },
20459
20460         /**
20461          * Returns the current scroll position
20462          * @method getScroll
20463          * @private
20464          * @static
20465          */
20466         getScroll: function () {
20467             var t, l, dde=document.documentElement, db=document.body;
20468             if (dde && (dde.scrollTop || dde.scrollLeft)) {
20469                 t = dde.scrollTop;
20470                 l = dde.scrollLeft;
20471             } else if (db) {
20472                 t = db.scrollTop;
20473                 l = db.scrollLeft;
20474             } else {
20475
20476             }
20477             return { top: t, left: l };
20478         },
20479
20480         /**
20481          * Returns the specified element style property
20482          * @method getStyle
20483          * @param {HTMLElement} el          the element
20484          * @param {string}      styleProp   the style property
20485          * @return {string} The value of the style property
20486          * @deprecated use Roo.lib.Dom.getStyle
20487          * @static
20488          */
20489         getStyle: function(el, styleProp) {
20490             return Roo.fly(el).getStyle(styleProp);
20491         },
20492
20493         /**
20494          * Gets the scrollTop
20495          * @method getScrollTop
20496          * @return {int} the document's scrollTop
20497          * @static
20498          */
20499         getScrollTop: function () { return this.getScroll().top; },
20500
20501         /**
20502          * Gets the scrollLeft
20503          * @method getScrollLeft
20504          * @return {int} the document's scrollTop
20505          * @static
20506          */
20507         getScrollLeft: function () { return this.getScroll().left; },
20508
20509         /**
20510          * Sets the x/y position of an element to the location of the
20511          * target element.
20512          * @method moveToEl
20513          * @param {HTMLElement} moveEl      The element to move
20514          * @param {HTMLElement} targetEl    The position reference element
20515          * @static
20516          */
20517         moveToEl: function (moveEl, targetEl) {
20518             var aCoord = Roo.lib.Dom.getXY(targetEl);
20519             Roo.lib.Dom.setXY(moveEl, aCoord);
20520         },
20521
20522         /**
20523          * Numeric array sort function
20524          * @method numericSort
20525          * @static
20526          */
20527         numericSort: function(a, b) { return (a - b); },
20528
20529         /**
20530          * Internal counter
20531          * @property _timeoutCount
20532          * @private
20533          * @static
20534          */
20535         _timeoutCount: 0,
20536
20537         /**
20538          * Trying to make the load order less important.  Without this we get
20539          * an error if this file is loaded before the Event Utility.
20540          * @method _addListeners
20541          * @private
20542          * @static
20543          */
20544         _addListeners: function() {
20545             var DDM = Roo.dd.DDM;
20546             if ( Roo.lib.Event && document ) {
20547                 DDM._onLoad();
20548             } else {
20549                 if (DDM._timeoutCount > 2000) {
20550                 } else {
20551                     setTimeout(DDM._addListeners, 10);
20552                     if (document && document.body) {
20553                         DDM._timeoutCount += 1;
20554                     }
20555                 }
20556             }
20557         },
20558
20559         /**
20560          * Recursively searches the immediate parent and all child nodes for
20561          * the handle element in order to determine wheter or not it was
20562          * clicked.
20563          * @method handleWasClicked
20564          * @param node the html element to inspect
20565          * @static
20566          */
20567         handleWasClicked: function(node, id) {
20568             if (this.isHandle(id, node.id)) {
20569                 return true;
20570             } else {
20571                 // check to see if this is a text node child of the one we want
20572                 var p = node.parentNode;
20573
20574                 while (p) {
20575                     if (this.isHandle(id, p.id)) {
20576                         return true;
20577                     } else {
20578                         p = p.parentNode;
20579                     }
20580                 }
20581             }
20582
20583             return false;
20584         }
20585
20586     };
20587
20588 }();
20589
20590 // shorter alias, save a few bytes
20591 Roo.dd.DDM = Roo.dd.DragDropMgr;
20592 Roo.dd.DDM._addListeners();
20593
20594 }/*
20595  * Based on:
20596  * Ext JS Library 1.1.1
20597  * Copyright(c) 2006-2007, Ext JS, LLC.
20598  *
20599  * Originally Released Under LGPL - original licence link has changed is not relivant.
20600  *
20601  * Fork - LGPL
20602  * <script type="text/javascript">
20603  */
20604
20605 /**
20606  * @class Roo.dd.DD
20607  * A DragDrop implementation where the linked element follows the
20608  * mouse cursor during a drag.
20609  * @extends Roo.dd.DragDrop
20610  * @constructor
20611  * @param {String} id the id of the linked element
20612  * @param {String} sGroup the group of related DragDrop items
20613  * @param {object} config an object containing configurable attributes
20614  *                Valid properties for DD:
20615  *                    scroll
20616  */
20617 Roo.dd.DD = function(id, sGroup, config) {
20618     if (id) {
20619         this.init(id, sGroup, config);
20620     }
20621 };
20622
20623 Roo.extend(Roo.dd.DD, Roo.dd.DragDrop, {
20624
20625     /**
20626      * When set to true, the utility automatically tries to scroll the browser
20627      * window wehn a drag and drop element is dragged near the viewport boundary.
20628      * Defaults to true.
20629      * @property scroll
20630      * @type boolean
20631      */
20632     scroll: true,
20633
20634     /**
20635      * Sets the pointer offset to the distance between the linked element's top
20636      * left corner and the location the element was clicked
20637      * @method autoOffset
20638      * @param {int} iPageX the X coordinate of the click
20639      * @param {int} iPageY the Y coordinate of the click
20640      */
20641     autoOffset: function(iPageX, iPageY) {
20642         var x = iPageX - this.startPageX;
20643         var y = iPageY - this.startPageY;
20644         this.setDelta(x, y);
20645     },
20646
20647     /**
20648      * Sets the pointer offset.  You can call this directly to force the
20649      * offset to be in a particular location (e.g., pass in 0,0 to set it
20650      * to the center of the object)
20651      * @method setDelta
20652      * @param {int} iDeltaX the distance from the left
20653      * @param {int} iDeltaY the distance from the top
20654      */
20655     setDelta: function(iDeltaX, iDeltaY) {
20656         this.deltaX = iDeltaX;
20657         this.deltaY = iDeltaY;
20658     },
20659
20660     /**
20661      * Sets the drag element to the location of the mousedown or click event,
20662      * maintaining the cursor location relative to the location on the element
20663      * that was clicked.  Override this if you want to place the element in a
20664      * location other than where the cursor is.
20665      * @method setDragElPos
20666      * @param {int} iPageX the X coordinate of the mousedown or drag event
20667      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20668      */
20669     setDragElPos: function(iPageX, iPageY) {
20670         // the first time we do this, we are going to check to make sure
20671         // the element has css positioning
20672
20673         var el = this.getDragEl();
20674         this.alignElWithMouse(el, iPageX, iPageY);
20675     },
20676
20677     /**
20678      * Sets the element to the location of the mousedown or click event,
20679      * maintaining the cursor location relative to the location on the element
20680      * that was clicked.  Override this if you want to place the element in a
20681      * location other than where the cursor is.
20682      * @method alignElWithMouse
20683      * @param {HTMLElement} el the element to move
20684      * @param {int} iPageX the X coordinate of the mousedown or drag event
20685      * @param {int} iPageY the Y coordinate of the mousedown or drag event
20686      */
20687     alignElWithMouse: function(el, iPageX, iPageY) {
20688         var oCoord = this.getTargetCoord(iPageX, iPageY);
20689         var fly = el.dom ? el : Roo.fly(el);
20690         if (!this.deltaSetXY) {
20691             var aCoord = [oCoord.x, oCoord.y];
20692             fly.setXY(aCoord);
20693             var newLeft = fly.getLeft(true);
20694             var newTop  = fly.getTop(true);
20695             this.deltaSetXY = [ newLeft - oCoord.x, newTop - oCoord.y ];
20696         } else {
20697             fly.setLeftTop(oCoord.x + this.deltaSetXY[0], oCoord.y + this.deltaSetXY[1]);
20698         }
20699
20700         this.cachePosition(oCoord.x, oCoord.y);
20701         this.autoScroll(oCoord.x, oCoord.y, el.offsetHeight, el.offsetWidth);
20702         return oCoord;
20703     },
20704
20705     /**
20706      * Saves the most recent position so that we can reset the constraints and
20707      * tick marks on-demand.  We need to know this so that we can calculate the
20708      * number of pixels the element is offset from its original position.
20709      * @method cachePosition
20710      * @param iPageX the current x position (optional, this just makes it so we
20711      * don't have to look it up again)
20712      * @param iPageY the current y position (optional, this just makes it so we
20713      * don't have to look it up again)
20714      */
20715     cachePosition: function(iPageX, iPageY) {
20716         if (iPageX) {
20717             this.lastPageX = iPageX;
20718             this.lastPageY = iPageY;
20719         } else {
20720             var aCoord = Roo.lib.Dom.getXY(this.getEl());
20721             this.lastPageX = aCoord[0];
20722             this.lastPageY = aCoord[1];
20723         }
20724     },
20725
20726     /**
20727      * Auto-scroll the window if the dragged object has been moved beyond the
20728      * visible window boundary.
20729      * @method autoScroll
20730      * @param {int} x the drag element's x position
20731      * @param {int} y the drag element's y position
20732      * @param {int} h the height of the drag element
20733      * @param {int} w the width of the drag element
20734      * @private
20735      */
20736     autoScroll: function(x, y, h, w) {
20737
20738         if (this.scroll) {
20739             // The client height
20740             var clientH = Roo.lib.Dom.getViewWidth();
20741
20742             // The client width
20743             var clientW = Roo.lib.Dom.getViewHeight();
20744
20745             // The amt scrolled down
20746             var st = this.DDM.getScrollTop();
20747
20748             // The amt scrolled right
20749             var sl = this.DDM.getScrollLeft();
20750
20751             // Location of the bottom of the element
20752             var bot = h + y;
20753
20754             // Location of the right of the element
20755             var right = w + x;
20756
20757             // The distance from the cursor to the bottom of the visible area,
20758             // adjusted so that we don't scroll if the cursor is beyond the
20759             // element drag constraints
20760             var toBot = (clientH + st - y - this.deltaY);
20761
20762             // The distance from the cursor to the right of the visible area
20763             var toRight = (clientW + sl - x - this.deltaX);
20764
20765
20766             // How close to the edge the cursor must be before we scroll
20767             // var thresh = (document.all) ? 100 : 40;
20768             var thresh = 40;
20769
20770             // How many pixels to scroll per autoscroll op.  This helps to reduce
20771             // clunky scrolling. IE is more sensitive about this ... it needs this
20772             // value to be higher.
20773             var scrAmt = (document.all) ? 80 : 30;
20774
20775             // Scroll down if we are near the bottom of the visible page and the
20776             // obj extends below the crease
20777             if ( bot > clientH && toBot < thresh ) {
20778                 window.scrollTo(sl, st + scrAmt);
20779             }
20780
20781             // Scroll up if the window is scrolled down and the top of the object
20782             // goes above the top border
20783             if ( y < st && st > 0 && y - st < thresh ) {
20784                 window.scrollTo(sl, st - scrAmt);
20785             }
20786
20787             // Scroll right if the obj is beyond the right border and the cursor is
20788             // near the border.
20789             if ( right > clientW && toRight < thresh ) {
20790                 window.scrollTo(sl + scrAmt, st);
20791             }
20792
20793             // Scroll left if the window has been scrolled to the right and the obj
20794             // extends past the left border
20795             if ( x < sl && sl > 0 && x - sl < thresh ) {
20796                 window.scrollTo(sl - scrAmt, st);
20797             }
20798         }
20799     },
20800
20801     /**
20802      * Finds the location the element should be placed if we want to move
20803      * it to where the mouse location less the click offset would place us.
20804      * @method getTargetCoord
20805      * @param {int} iPageX the X coordinate of the click
20806      * @param {int} iPageY the Y coordinate of the click
20807      * @return an object that contains the coordinates (Object.x and Object.y)
20808      * @private
20809      */
20810     getTargetCoord: function(iPageX, iPageY) {
20811
20812
20813         var x = iPageX - this.deltaX;
20814         var y = iPageY - this.deltaY;
20815
20816         if (this.constrainX) {
20817             if (x < this.minX) { x = this.minX; }
20818             if (x > this.maxX) { x = this.maxX; }
20819         }
20820
20821         if (this.constrainY) {
20822             if (y < this.minY) { y = this.minY; }
20823             if (y > this.maxY) { y = this.maxY; }
20824         }
20825
20826         x = this.getTick(x, this.xTicks);
20827         y = this.getTick(y, this.yTicks);
20828
20829
20830         return {x:x, y:y};
20831     },
20832
20833     /*
20834      * Sets up config options specific to this class. Overrides
20835      * Roo.dd.DragDrop, but all versions of this method through the
20836      * inheritance chain are called
20837      */
20838     applyConfig: function() {
20839         Roo.dd.DD.superclass.applyConfig.call(this);
20840         this.scroll = (this.config.scroll !== false);
20841     },
20842
20843     /*
20844      * Event that fires prior to the onMouseDown event.  Overrides
20845      * Roo.dd.DragDrop.
20846      */
20847     b4MouseDown: function(e) {
20848         // this.resetConstraints();
20849         this.autoOffset(e.getPageX(),
20850                             e.getPageY());
20851     },
20852
20853     /*
20854      * Event that fires prior to the onDrag event.  Overrides
20855      * Roo.dd.DragDrop.
20856      */
20857     b4Drag: function(e) {
20858         this.setDragElPos(e.getPageX(),
20859                             e.getPageY());
20860     },
20861
20862     toString: function() {
20863         return ("DD " + this.id);
20864     }
20865
20866     //////////////////////////////////////////////////////////////////////////
20867     // Debugging ygDragDrop events that can be overridden
20868     //////////////////////////////////////////////////////////////////////////
20869     /*
20870     startDrag: function(x, y) {
20871     },
20872
20873     onDrag: function(e) {
20874     },
20875
20876     onDragEnter: function(e, id) {
20877     },
20878
20879     onDragOver: function(e, id) {
20880     },
20881
20882     onDragOut: function(e, id) {
20883     },
20884
20885     onDragDrop: function(e, id) {
20886     },
20887
20888     endDrag: function(e) {
20889     }
20890
20891     */
20892
20893 });/*
20894  * Based on:
20895  * Ext JS Library 1.1.1
20896  * Copyright(c) 2006-2007, Ext JS, LLC.
20897  *
20898  * Originally Released Under LGPL - original licence link has changed is not relivant.
20899  *
20900  * Fork - LGPL
20901  * <script type="text/javascript">
20902  */
20903
20904 /**
20905  * @class Roo.dd.DDProxy
20906  * A DragDrop implementation that inserts an empty, bordered div into
20907  * the document that follows the cursor during drag operations.  At the time of
20908  * the click, the frame div is resized to the dimensions of the linked html
20909  * element, and moved to the exact location of the linked element.
20910  *
20911  * References to the "frame" element refer to the single proxy element that
20912  * was created to be dragged in place of all DDProxy elements on the
20913  * page.
20914  *
20915  * @extends Roo.dd.DD
20916  * @constructor
20917  * @param {String} id the id of the linked html element
20918  * @param {String} sGroup the group of related DragDrop objects
20919  * @param {object} config an object containing configurable attributes
20920  *                Valid properties for DDProxy in addition to those in DragDrop:
20921  *                   resizeFrame, centerFrame, dragElId
20922  */
20923 Roo.dd.DDProxy = function(id, sGroup, config) {
20924     if (id) {
20925         this.init(id, sGroup, config);
20926         this.initFrame();
20927     }
20928 };
20929
20930 /**
20931  * The default drag frame div id
20932  * @property Roo.dd.DDProxy.dragElId
20933  * @type String
20934  * @static
20935  */
20936 Roo.dd.DDProxy.dragElId = "ygddfdiv";
20937
20938 Roo.extend(Roo.dd.DDProxy, Roo.dd.DD, {
20939
20940     /**
20941      * By default we resize the drag frame to be the same size as the element
20942      * we want to drag (this is to get the frame effect).  We can turn it off
20943      * if we want a different behavior.
20944      * @property resizeFrame
20945      * @type boolean
20946      */
20947     resizeFrame: true,
20948
20949     /**
20950      * By default the frame is positioned exactly where the drag element is, so
20951      * we use the cursor offset provided by Roo.dd.DD.  Another option that works only if
20952      * you do not have constraints on the obj is to have the drag frame centered
20953      * around the cursor.  Set centerFrame to true for this effect.
20954      * @property centerFrame
20955      * @type boolean
20956      */
20957     centerFrame: false,
20958
20959     /**
20960      * Creates the proxy element if it does not yet exist
20961      * @method createFrame
20962      */
20963     createFrame: function() {
20964         var self = this;
20965         var body = document.body;
20966
20967         if (!body || !body.firstChild) {
20968             setTimeout( function() { self.createFrame(); }, 50 );
20969             return;
20970         }
20971
20972         var div = this.getDragEl();
20973
20974         if (!div) {
20975             div    = document.createElement("div");
20976             div.id = this.dragElId;
20977             var s  = div.style;
20978
20979             s.position   = "absolute";
20980             s.visibility = "hidden";
20981             s.cursor     = "move";
20982             s.border     = "2px solid #aaa";
20983             s.zIndex     = 999;
20984
20985             // appendChild can blow up IE if invoked prior to the window load event
20986             // while rendering a table.  It is possible there are other scenarios
20987             // that would cause this to happen as well.
20988             body.insertBefore(div, body.firstChild);
20989         }
20990     },
20991
20992     /**
20993      * Initialization for the drag frame element.  Must be called in the
20994      * constructor of all subclasses
20995      * @method initFrame
20996      */
20997     initFrame: function() {
20998         this.createFrame();
20999     },
21000
21001     applyConfig: function() {
21002         Roo.dd.DDProxy.superclass.applyConfig.call(this);
21003
21004         this.resizeFrame = (this.config.resizeFrame !== false);
21005         this.centerFrame = (this.config.centerFrame);
21006         this.setDragElId(this.config.dragElId || Roo.dd.DDProxy.dragElId);
21007     },
21008
21009     /**
21010      * Resizes the drag frame to the dimensions of the clicked object, positions
21011      * it over the object, and finally displays it
21012      * @method showFrame
21013      * @param {int} iPageX X click position
21014      * @param {int} iPageY Y click position
21015      * @private
21016      */
21017     showFrame: function(iPageX, iPageY) {
21018         var el = this.getEl();
21019         var dragEl = this.getDragEl();
21020         var s = dragEl.style;
21021
21022         this._resizeProxy();
21023
21024         if (this.centerFrame) {
21025             this.setDelta( Math.round(parseInt(s.width,  10)/2),
21026                            Math.round(parseInt(s.height, 10)/2) );
21027         }
21028
21029         this.setDragElPos(iPageX, iPageY);
21030
21031         Roo.fly(dragEl).show();
21032     },
21033
21034     /**
21035      * The proxy is automatically resized to the dimensions of the linked
21036      * element when a drag is initiated, unless resizeFrame is set to false
21037      * @method _resizeProxy
21038      * @private
21039      */
21040     _resizeProxy: function() {
21041         if (this.resizeFrame) {
21042             var el = this.getEl();
21043             Roo.fly(this.getDragEl()).setSize(el.offsetWidth, el.offsetHeight);
21044         }
21045     },
21046
21047     // overrides Roo.dd.DragDrop
21048     b4MouseDown: function(e) {
21049         var x = e.getPageX();
21050         var y = e.getPageY();
21051         this.autoOffset(x, y);
21052         this.setDragElPos(x, y);
21053     },
21054
21055     // overrides Roo.dd.DragDrop
21056     b4StartDrag: function(x, y) {
21057         // show the drag frame
21058         this.showFrame(x, y);
21059     },
21060
21061     // overrides Roo.dd.DragDrop
21062     b4EndDrag: function(e) {
21063         Roo.fly(this.getDragEl()).hide();
21064     },
21065
21066     // overrides Roo.dd.DragDrop
21067     // By default we try to move the element to the last location of the frame.
21068     // This is so that the default behavior mirrors that of Roo.dd.DD.
21069     endDrag: function(e) {
21070
21071         var lel = this.getEl();
21072         var del = this.getDragEl();
21073
21074         // Show the drag frame briefly so we can get its position
21075         del.style.visibility = "";
21076
21077         this.beforeMove();
21078         // Hide the linked element before the move to get around a Safari
21079         // rendering bug.
21080         lel.style.visibility = "hidden";
21081         Roo.dd.DDM.moveToEl(lel, del);
21082         del.style.visibility = "hidden";
21083         lel.style.visibility = "";
21084
21085         this.afterDrag();
21086     },
21087
21088     beforeMove : function(){
21089
21090     },
21091
21092     afterDrag : function(){
21093
21094     },
21095
21096     toString: function() {
21097         return ("DDProxy " + this.id);
21098     }
21099
21100 });
21101 /*
21102  * Based on:
21103  * Ext JS Library 1.1.1
21104  * Copyright(c) 2006-2007, Ext JS, LLC.
21105  *
21106  * Originally Released Under LGPL - original licence link has changed is not relivant.
21107  *
21108  * Fork - LGPL
21109  * <script type="text/javascript">
21110  */
21111
21112  /**
21113  * @class Roo.dd.DDTarget
21114  * A DragDrop implementation that does not move, but can be a drop
21115  * target.  You would get the same result by simply omitting implementation
21116  * for the event callbacks, but this way we reduce the processing cost of the
21117  * event listener and the callbacks.
21118  * @extends Roo.dd.DragDrop
21119  * @constructor
21120  * @param {String} id the id of the element that is a drop target
21121  * @param {String} sGroup the group of related DragDrop objects
21122  * @param {object} config an object containing configurable attributes
21123  *                 Valid properties for DDTarget in addition to those in
21124  *                 DragDrop:
21125  *                    none
21126  */
21127 Roo.dd.DDTarget = function(id, sGroup, config) {
21128     if (id) {
21129         this.initTarget(id, sGroup, config);
21130     }
21131     if (config && (config.listeners || config.events)) { 
21132         Roo.dd.DragDrop.superclass.constructor.call(this,  { 
21133             listeners : config.listeners || {}, 
21134             events : config.events || {} 
21135         });    
21136     }
21137 };
21138
21139 // Roo.dd.DDTarget.prototype = new Roo.dd.DragDrop();
21140 Roo.extend(Roo.dd.DDTarget, Roo.dd.DragDrop, {
21141     toString: function() {
21142         return ("DDTarget " + this.id);
21143     }
21144 });
21145 /*
21146  * Based on:
21147  * Ext JS Library 1.1.1
21148  * Copyright(c) 2006-2007, Ext JS, LLC.
21149  *
21150  * Originally Released Under LGPL - original licence link has changed is not relivant.
21151  *
21152  * Fork - LGPL
21153  * <script type="text/javascript">
21154  */
21155  
21156
21157 /**
21158  * @class Roo.dd.ScrollManager
21159  * Provides automatic scrolling of overflow regions in the page during drag operations.<br><br>
21160  * <b>Note: This class uses "Point Mode" and is untested in "Intersect Mode".</b>
21161  * @singleton
21162  */
21163 Roo.dd.ScrollManager = function(){
21164     var ddm = Roo.dd.DragDropMgr;
21165     var els = {};
21166     var dragEl = null;
21167     var proc = {};
21168     
21169     
21170     
21171     var onStop = function(e){
21172         dragEl = null;
21173         clearProc();
21174     };
21175     
21176     var triggerRefresh = function(){
21177         if(ddm.dragCurrent){
21178              ddm.refreshCache(ddm.dragCurrent.groups);
21179         }
21180     };
21181     
21182     var doScroll = function(){
21183         if(ddm.dragCurrent){
21184             var dds = Roo.dd.ScrollManager;
21185             if(!dds.animate){
21186                 if(proc.el.scroll(proc.dir, dds.increment)){
21187                     triggerRefresh();
21188                 }
21189             }else{
21190                 proc.el.scroll(proc.dir, dds.increment, true, dds.animDuration, triggerRefresh);
21191             }
21192         }
21193     };
21194     
21195     var clearProc = function(){
21196         if(proc.id){
21197             clearInterval(proc.id);
21198         }
21199         proc.id = 0;
21200         proc.el = null;
21201         proc.dir = "";
21202     };
21203     
21204     var startProc = function(el, dir){
21205          Roo.log('scroll startproc');
21206         clearProc();
21207         proc.el = el;
21208         proc.dir = dir;
21209         proc.id = setInterval(doScroll, Roo.dd.ScrollManager.frequency);
21210     };
21211     
21212     var onFire = function(e, isDrop){
21213        
21214         if(isDrop || !ddm.dragCurrent){ return; }
21215         var dds = Roo.dd.ScrollManager;
21216         if(!dragEl || dragEl != ddm.dragCurrent){
21217             dragEl = ddm.dragCurrent;
21218             // refresh regions on drag start
21219             dds.refreshCache();
21220         }
21221         
21222         var xy = Roo.lib.Event.getXY(e);
21223         var pt = new Roo.lib.Point(xy[0], xy[1]);
21224         for(var id in els){
21225             var el = els[id], r = el._region;
21226             if(r && r.contains(pt) && el.isScrollable()){
21227                 if(r.bottom - pt.y <= dds.thresh){
21228                     if(proc.el != el){
21229                         startProc(el, "down");
21230                     }
21231                     return;
21232                 }else if(r.right - pt.x <= dds.thresh){
21233                     if(proc.el != el){
21234                         startProc(el, "left");
21235                     }
21236                     return;
21237                 }else if(pt.y - r.top <= dds.thresh){
21238                     if(proc.el != el){
21239                         startProc(el, "up");
21240                     }
21241                     return;
21242                 }else if(pt.x - r.left <= dds.thresh){
21243                     if(proc.el != el){
21244                         startProc(el, "right");
21245                     }
21246                     return;
21247                 }
21248             }
21249         }
21250         clearProc();
21251     };
21252     
21253     ddm.fireEvents = ddm.fireEvents.createSequence(onFire, ddm);
21254     ddm.stopDrag = ddm.stopDrag.createSequence(onStop, ddm);
21255     
21256     return {
21257         /**
21258          * Registers new overflow element(s) to auto scroll
21259          * @param {String/HTMLElement/Element/Array} el The id of or the element to be scrolled or an array of either
21260          */
21261         register : function(el){
21262             if(el instanceof Array){
21263                 for(var i = 0, len = el.length; i < len; i++) {
21264                         this.register(el[i]);
21265                 }
21266             }else{
21267                 el = Roo.get(el);
21268                 els[el.id] = el;
21269             }
21270             Roo.dd.ScrollManager.els = els;
21271         },
21272         
21273         /**
21274          * Unregisters overflow element(s) so they are no longer scrolled
21275          * @param {String/HTMLElement/Element/Array} el The id of or the element to be removed or an array of either
21276          */
21277         unregister : function(el){
21278             if(el instanceof Array){
21279                 for(var i = 0, len = el.length; i < len; i++) {
21280                         this.unregister(el[i]);
21281                 }
21282             }else{
21283                 el = Roo.get(el);
21284                 delete els[el.id];
21285             }
21286         },
21287         
21288         /**
21289          * The number of pixels from the edge of a container the pointer needs to be to 
21290          * trigger scrolling (defaults to 25)
21291          * @type Number
21292          */
21293         thresh : 25,
21294         
21295         /**
21296          * The number of pixels to scroll in each scroll increment (defaults to 50)
21297          * @type Number
21298          */
21299         increment : 100,
21300         
21301         /**
21302          * The frequency of scrolls in milliseconds (defaults to 500)
21303          * @type Number
21304          */
21305         frequency : 500,
21306         
21307         /**
21308          * True to animate the scroll (defaults to true)
21309          * @type Boolean
21310          */
21311         animate: true,
21312         
21313         /**
21314          * The animation duration in seconds - 
21315          * MUST BE less than Roo.dd.ScrollManager.frequency! (defaults to .4)
21316          * @type Number
21317          */
21318         animDuration: .4,
21319         
21320         /**
21321          * Manually trigger a cache refresh.
21322          */
21323         refreshCache : function(){
21324             for(var id in els){
21325                 if(typeof els[id] == 'object'){ // for people extending the object prototype
21326                     els[id]._region = els[id].getRegion();
21327                 }
21328             }
21329         }
21330     };
21331 }();/*
21332  * Based on:
21333  * Ext JS Library 1.1.1
21334  * Copyright(c) 2006-2007, Ext JS, LLC.
21335  *
21336  * Originally Released Under LGPL - original licence link has changed is not relivant.
21337  *
21338  * Fork - LGPL
21339  * <script type="text/javascript">
21340  */
21341  
21342
21343 /**
21344  * @class Roo.dd.Registry
21345  * Provides easy access to all drag drop components that are registered on a page.  Items can be retrieved either
21346  * directly by DOM node id, or by passing in the drag drop event that occurred and looking up the event target.
21347  * @singleton
21348  */
21349 Roo.dd.Registry = function(){
21350     var elements = {}; 
21351     var handles = {}; 
21352     var autoIdSeed = 0;
21353
21354     var getId = function(el, autogen){
21355         if(typeof el == "string"){
21356             return el;
21357         }
21358         var id = el.id;
21359         if(!id && autogen !== false){
21360             id = "roodd-" + (++autoIdSeed);
21361             el.id = id;
21362         }
21363         return id;
21364     };
21365     
21366     return {
21367     /**
21368      * Register a drag drop element
21369      * @param {String|HTMLElement} element The id or DOM node to register
21370      * @param {Object} data (optional) A custom data object that will be passed between the elements that are involved
21371      * in drag drop operations.  You can populate this object with any arbitrary properties that your own code
21372      * knows how to interpret, plus there are some specific properties known to the Registry that should be
21373      * populated in the data object (if applicable):
21374      * <pre>
21375 Value      Description<br />
21376 ---------  ------------------------------------------<br />
21377 handles    Array of DOM nodes that trigger dragging<br />
21378            for the element being registered<br />
21379 isHandle   True if the element passed in triggers<br />
21380            dragging itself, else false
21381 </pre>
21382      */
21383         register : function(el, data){
21384             data = data || {};
21385             if(typeof el == "string"){
21386                 el = document.getElementById(el);
21387             }
21388             data.ddel = el;
21389             elements[getId(el)] = data;
21390             if(data.isHandle !== false){
21391                 handles[data.ddel.id] = data;
21392             }
21393             if(data.handles){
21394                 var hs = data.handles;
21395                 for(var i = 0, len = hs.length; i < len; i++){
21396                         handles[getId(hs[i])] = data;
21397                 }
21398             }
21399         },
21400
21401     /**
21402      * Unregister a drag drop element
21403      * @param {String|HTMLElement}  element The id or DOM node to unregister
21404      */
21405         unregister : function(el){
21406             var id = getId(el, false);
21407             var data = elements[id];
21408             if(data){
21409                 delete elements[id];
21410                 if(data.handles){
21411                     var hs = data.handles;
21412                     for(var i = 0, len = hs.length; i < len; i++){
21413                         delete handles[getId(hs[i], false)];
21414                     }
21415                 }
21416             }
21417         },
21418
21419     /**
21420      * Returns the handle registered for a DOM Node by id
21421      * @param {String|HTMLElement} id The DOM node or id to look up
21422      * @return {Object} handle The custom handle data
21423      */
21424         getHandle : function(id){
21425             if(typeof id != "string"){ // must be element?
21426                 id = id.id;
21427             }
21428             return handles[id];
21429         },
21430
21431     /**
21432      * Returns the handle that is registered for the DOM node that is the target of the event
21433      * @param {Event} e The event
21434      * @return {Object} handle The custom handle data
21435      */
21436         getHandleFromEvent : function(e){
21437             var t = Roo.lib.Event.getTarget(e);
21438             return t ? handles[t.id] : null;
21439         },
21440
21441     /**
21442      * Returns a custom data object that is registered for a DOM node by id
21443      * @param {String|HTMLElement} id The DOM node or id to look up
21444      * @return {Object} data The custom data
21445      */
21446         getTarget : function(id){
21447             if(typeof id != "string"){ // must be element?
21448                 id = id.id;
21449             }
21450             return elements[id];
21451         },
21452
21453     /**
21454      * Returns a custom data object that is registered for the DOM node that is the target of the event
21455      * @param {Event} e The event
21456      * @return {Object} data The custom data
21457      */
21458         getTargetFromEvent : function(e){
21459             var t = Roo.lib.Event.getTarget(e);
21460             return t ? elements[t.id] || handles[t.id] : null;
21461         }
21462     };
21463 }();/*
21464  * Based on:
21465  * Ext JS Library 1.1.1
21466  * Copyright(c) 2006-2007, Ext JS, LLC.
21467  *
21468  * Originally Released Under LGPL - original licence link has changed is not relivant.
21469  *
21470  * Fork - LGPL
21471  * <script type="text/javascript">
21472  */
21473  
21474
21475 /**
21476  * @class Roo.dd.StatusProxy
21477  * A specialized drag proxy that supports a drop status icon, {@link Roo.Layer} styles and auto-repair.  This is the
21478  * default drag proxy used by all Roo.dd components.
21479  * @constructor
21480  * @param {Object} config
21481  */
21482 Roo.dd.StatusProxy = function(config){
21483     Roo.apply(this, config);
21484     this.id = this.id || Roo.id();
21485     this.el = new Roo.Layer({
21486         dh: {
21487             id: this.id, tag: "div", cls: "x-dd-drag-proxy "+this.dropNotAllowed, children: [
21488                 {tag: "div", cls: "x-dd-drop-icon"},
21489                 {tag: "div", cls: "x-dd-drag-ghost"}
21490             ]
21491         }, 
21492         shadow: !config || config.shadow !== false
21493     });
21494     this.ghost = Roo.get(this.el.dom.childNodes[1]);
21495     this.dropStatus = this.dropNotAllowed;
21496 };
21497
21498 Roo.dd.StatusProxy.prototype = {
21499     /**
21500      * @cfg {String} dropAllowed
21501      * The CSS class to apply to the status element when drop is allowed (defaults to "x-dd-drop-ok").
21502      */
21503     dropAllowed : "x-dd-drop-ok",
21504     /**
21505      * @cfg {String} dropNotAllowed
21506      * The CSS class to apply to the status element when drop is not allowed (defaults to "x-dd-drop-nodrop").
21507      */
21508     dropNotAllowed : "x-dd-drop-nodrop",
21509
21510     /**
21511      * Updates the proxy's visual element to indicate the status of whether or not drop is allowed
21512      * over the current target element.
21513      * @param {String} cssClass The css class for the new drop status indicator image
21514      */
21515     setStatus : function(cssClass){
21516         cssClass = cssClass || this.dropNotAllowed;
21517         if(this.dropStatus != cssClass){
21518             this.el.replaceClass(this.dropStatus, cssClass);
21519             this.dropStatus = cssClass;
21520         }
21521     },
21522
21523     /**
21524      * Resets the status indicator to the default dropNotAllowed value
21525      * @param {Boolean} clearGhost True to also remove all content from the ghost, false to preserve it
21526      */
21527     reset : function(clearGhost){
21528         this.el.dom.className = "x-dd-drag-proxy " + this.dropNotAllowed;
21529         this.dropStatus = this.dropNotAllowed;
21530         if(clearGhost){
21531             this.ghost.update("");
21532         }
21533     },
21534
21535     /**
21536      * Updates the contents of the ghost element
21537      * @param {String} html The html that will replace the current innerHTML of the ghost element
21538      */
21539     update : function(html){
21540         if(typeof html == "string"){
21541             this.ghost.update(html);
21542         }else{
21543             this.ghost.update("");
21544             html.style.margin = "0";
21545             this.ghost.dom.appendChild(html);
21546         }
21547         // ensure float = none set?? cant remember why though.
21548         var el = this.ghost.dom.firstChild;
21549                 if(el){
21550                         Roo.fly(el).setStyle('float', 'none');
21551                 }
21552     },
21553     
21554     /**
21555      * Returns the underlying proxy {@link Roo.Layer}
21556      * @return {Roo.Layer} el
21557     */
21558     getEl : function(){
21559         return this.el;
21560     },
21561
21562     /**
21563      * Returns the ghost element
21564      * @return {Roo.Element} el
21565      */
21566     getGhost : function(){
21567         return this.ghost;
21568     },
21569
21570     /**
21571      * Hides the proxy
21572      * @param {Boolean} clear True to reset the status and clear the ghost contents, false to preserve them
21573      */
21574     hide : function(clear){
21575         this.el.hide();
21576         if(clear){
21577             this.reset(true);
21578         }
21579     },
21580
21581     /**
21582      * Stops the repair animation if it's currently running
21583      */
21584     stop : function(){
21585         if(this.anim && this.anim.isAnimated && this.anim.isAnimated()){
21586             this.anim.stop();
21587         }
21588     },
21589
21590     /**
21591      * Displays this proxy
21592      */
21593     show : function(){
21594         this.el.show();
21595     },
21596
21597     /**
21598      * Force the Layer to sync its shadow and shim positions to the element
21599      */
21600     sync : function(){
21601         this.el.sync();
21602     },
21603
21604     /**
21605      * Causes the proxy to return to its position of origin via an animation.  Should be called after an
21606      * invalid drop operation by the item being dragged.
21607      * @param {Array} xy The XY position of the element ([x, y])
21608      * @param {Function} callback The function to call after the repair is complete
21609      * @param {Object} scope The scope in which to execute the callback
21610      */
21611     repair : function(xy, callback, scope){
21612         this.callback = callback;
21613         this.scope = scope;
21614         if(xy && this.animRepair !== false){
21615             this.el.addClass("x-dd-drag-repair");
21616             this.el.hideUnders(true);
21617             this.anim = this.el.shift({
21618                 duration: this.repairDuration || .5,
21619                 easing: 'easeOut',
21620                 xy: xy,
21621                 stopFx: true,
21622                 callback: this.afterRepair,
21623                 scope: this
21624             });
21625         }else{
21626             this.afterRepair();
21627         }
21628     },
21629
21630     // private
21631     afterRepair : function(){
21632         this.hide(true);
21633         if(typeof this.callback == "function"){
21634             this.callback.call(this.scope || this);
21635         }
21636         this.callback = null;
21637         this.scope = null;
21638     }
21639 };/*
21640  * Based on:
21641  * Ext JS Library 1.1.1
21642  * Copyright(c) 2006-2007, Ext JS, LLC.
21643  *
21644  * Originally Released Under LGPL - original licence link has changed is not relivant.
21645  *
21646  * Fork - LGPL
21647  * <script type="text/javascript">
21648  */
21649
21650 /**
21651  * @class Roo.dd.DragSource
21652  * @extends Roo.dd.DDProxy
21653  * A simple class that provides the basic implementation needed to make any element draggable.
21654  * @constructor
21655  * @param {String/HTMLElement/Element} el The container element
21656  * @param {Object} config
21657  */
21658 Roo.dd.DragSource = function(el, config){
21659     this.el = Roo.get(el);
21660     this.dragData = {};
21661     
21662     Roo.apply(this, config);
21663     
21664     if(!this.proxy){
21665         this.proxy = new Roo.dd.StatusProxy();
21666     }
21667
21668     Roo.dd.DragSource.superclass.constructor.call(this, this.el.dom, this.ddGroup || this.group,
21669           {dragElId : this.proxy.id, resizeFrame: false, isTarget: false, scroll: this.scroll === true});
21670     
21671     this.dragging = false;
21672 };
21673
21674 Roo.extend(Roo.dd.DragSource, Roo.dd.DDProxy, {
21675     /**
21676      * @cfg {String} dropAllowed
21677      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
21678      */
21679     dropAllowed : "x-dd-drop-ok",
21680     /**
21681      * @cfg {String} dropNotAllowed
21682      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
21683      */
21684     dropNotAllowed : "x-dd-drop-nodrop",
21685
21686     /**
21687      * Returns the data object associated with this drag source
21688      * @return {Object} data An object containing arbitrary data
21689      */
21690     getDragData : function(e){
21691         return this.dragData;
21692     },
21693
21694     // private
21695     onDragEnter : function(e, id){
21696         var target = Roo.dd.DragDropMgr.getDDById(id);
21697         this.cachedTarget = target;
21698         if(this.beforeDragEnter(target, e, id) !== false){
21699             if(target.isNotifyTarget){
21700                 var status = target.notifyEnter(this, e, this.dragData);
21701                 this.proxy.setStatus(status);
21702             }else{
21703                 this.proxy.setStatus(this.dropAllowed);
21704             }
21705             
21706             if(this.afterDragEnter){
21707                 /**
21708                  * An empty function by default, but provided so that you can perform a custom action
21709                  * when the dragged item enters the drop target by providing an implementation.
21710                  * @param {Roo.dd.DragDrop} target The drop target
21711                  * @param {Event} e The event object
21712                  * @param {String} id The id of the dragged element
21713                  * @method afterDragEnter
21714                  */
21715                 this.afterDragEnter(target, e, id);
21716             }
21717         }
21718     },
21719
21720     /**
21721      * An empty function by default, but provided so that you can perform a custom action
21722      * before the dragged item enters the drop target and optionally cancel the onDragEnter.
21723      * @param {Roo.dd.DragDrop} target The drop target
21724      * @param {Event} e The event object
21725      * @param {String} id The id of the dragged element
21726      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21727      */
21728     beforeDragEnter : function(target, e, id){
21729         return true;
21730     },
21731
21732     // private
21733     alignElWithMouse: function() {
21734         Roo.dd.DragSource.superclass.alignElWithMouse.apply(this, arguments);
21735         this.proxy.sync();
21736     },
21737
21738     // private
21739     onDragOver : function(e, id){
21740         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21741         if(this.beforeDragOver(target, e, id) !== false){
21742             if(target.isNotifyTarget){
21743                 var status = target.notifyOver(this, e, this.dragData);
21744                 this.proxy.setStatus(status);
21745             }
21746
21747             if(this.afterDragOver){
21748                 /**
21749                  * An empty function by default, but provided so that you can perform a custom action
21750                  * while the dragged item is over the drop target by providing an implementation.
21751                  * @param {Roo.dd.DragDrop} target The drop target
21752                  * @param {Event} e The event object
21753                  * @param {String} id The id of the dragged element
21754                  * @method afterDragOver
21755                  */
21756                 this.afterDragOver(target, e, id);
21757             }
21758         }
21759     },
21760
21761     /**
21762      * An empty function by default, but provided so that you can perform a custom action
21763      * while the dragged item is over the drop target and optionally cancel the onDragOver.
21764      * @param {Roo.dd.DragDrop} target The drop target
21765      * @param {Event} e The event object
21766      * @param {String} id The id of the dragged element
21767      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21768      */
21769     beforeDragOver : function(target, e, id){
21770         return true;
21771     },
21772
21773     // private
21774     onDragOut : function(e, id){
21775         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21776         if(this.beforeDragOut(target, e, id) !== false){
21777             if(target.isNotifyTarget){
21778                 target.notifyOut(this, e, this.dragData);
21779             }
21780             this.proxy.reset();
21781             if(this.afterDragOut){
21782                 /**
21783                  * An empty function by default, but provided so that you can perform a custom action
21784                  * after the dragged item is dragged out of the target without dropping.
21785                  * @param {Roo.dd.DragDrop} target The drop target
21786                  * @param {Event} e The event object
21787                  * @param {String} id The id of the dragged element
21788                  * @method afterDragOut
21789                  */
21790                 this.afterDragOut(target, e, id);
21791             }
21792         }
21793         this.cachedTarget = null;
21794     },
21795
21796     /**
21797      * An empty function by default, but provided so that you can perform a custom action before the dragged
21798      * item is dragged out of the target without dropping, and optionally cancel the onDragOut.
21799      * @param {Roo.dd.DragDrop} target The drop target
21800      * @param {Event} e The event object
21801      * @param {String} id The id of the dragged element
21802      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21803      */
21804     beforeDragOut : function(target, e, id){
21805         return true;
21806     },
21807     
21808     // private
21809     onDragDrop : function(e, id){
21810         var target = this.cachedTarget || Roo.dd.DragDropMgr.getDDById(id);
21811         if(this.beforeDragDrop(target, e, id) !== false){
21812             if(target.isNotifyTarget){
21813                 if(target.notifyDrop(this, e, this.dragData)){ // valid drop?
21814                     this.onValidDrop(target, e, id);
21815                 }else{
21816                     this.onInvalidDrop(target, e, id);
21817                 }
21818             }else{
21819                 this.onValidDrop(target, e, id);
21820             }
21821             
21822             if(this.afterDragDrop){
21823                 /**
21824                  * An empty function by default, but provided so that you can perform a custom action
21825                  * after a valid drag drop has occurred by providing an implementation.
21826                  * @param {Roo.dd.DragDrop} target The drop target
21827                  * @param {Event} e The event object
21828                  * @param {String} id The id of the dropped element
21829                  * @method afterDragDrop
21830                  */
21831                 this.afterDragDrop(target, e, id);
21832             }
21833         }
21834         delete this.cachedTarget;
21835     },
21836
21837     /**
21838      * An empty function by default, but provided so that you can perform a custom action before the dragged
21839      * item is dropped onto the target and optionally cancel the onDragDrop.
21840      * @param {Roo.dd.DragDrop} target The drop target
21841      * @param {Event} e The event object
21842      * @param {String} id The id of the dragged element
21843      * @return {Boolean} isValid True if the drag drop event is valid, else false to cancel
21844      */
21845     beforeDragDrop : function(target, e, id){
21846         return true;
21847     },
21848
21849     // private
21850     onValidDrop : function(target, e, id){
21851         this.hideProxy();
21852         if(this.afterValidDrop){
21853             /**
21854              * An empty function by default, but provided so that you can perform a custom action
21855              * after a valid drop has occurred by providing an implementation.
21856              * @param {Object} target The target DD 
21857              * @param {Event} e The event object
21858              * @param {String} id The id of the dropped element
21859              * @method afterInvalidDrop
21860              */
21861             this.afterValidDrop(target, e, id);
21862         }
21863     },
21864
21865     // private
21866     getRepairXY : function(e, data){
21867         return this.el.getXY();  
21868     },
21869
21870     // private
21871     onInvalidDrop : function(target, e, id){
21872         this.beforeInvalidDrop(target, e, id);
21873         if(this.cachedTarget){
21874             if(this.cachedTarget.isNotifyTarget){
21875                 this.cachedTarget.notifyOut(this, e, this.dragData);
21876             }
21877             this.cacheTarget = null;
21878         }
21879         this.proxy.repair(this.getRepairXY(e, this.dragData), this.afterRepair, this);
21880
21881         if(this.afterInvalidDrop){
21882             /**
21883              * An empty function by default, but provided so that you can perform a custom action
21884              * after an invalid drop has occurred by providing an implementation.
21885              * @param {Event} e The event object
21886              * @param {String} id The id of the dropped element
21887              * @method afterInvalidDrop
21888              */
21889             this.afterInvalidDrop(e, id);
21890         }
21891     },
21892
21893     // private
21894     afterRepair : function(){
21895         if(Roo.enableFx){
21896             this.el.highlight(this.hlColor || "c3daf9");
21897         }
21898         this.dragging = false;
21899     },
21900
21901     /**
21902      * An empty function by default, but provided so that you can perform a custom action after an invalid
21903      * drop has occurred.
21904      * @param {Roo.dd.DragDrop} target The drop target
21905      * @param {Event} e The event object
21906      * @param {String} id The id of the dragged element
21907      * @return {Boolean} isValid True if the invalid drop should proceed, else false to cancel
21908      */
21909     beforeInvalidDrop : function(target, e, id){
21910         return true;
21911     },
21912
21913     // private
21914     handleMouseDown : function(e){
21915         if(this.dragging) {
21916             return;
21917         }
21918         var data = this.getDragData(e);
21919         if(data && this.onBeforeDrag(data, e) !== false){
21920             this.dragData = data;
21921             this.proxy.stop();
21922             Roo.dd.DragSource.superclass.handleMouseDown.apply(this, arguments);
21923         } 
21924     },
21925
21926     /**
21927      * An empty function by default, but provided so that you can perform a custom action before the initial
21928      * drag event begins and optionally cancel it.
21929      * @param {Object} data An object containing arbitrary data to be shared with drop targets
21930      * @param {Event} e The event object
21931      * @return {Boolean} isValid True if the drag event is valid, else false to cancel
21932      */
21933     onBeforeDrag : function(data, e){
21934         return true;
21935     },
21936
21937     /**
21938      * An empty function by default, but provided so that you can perform a custom action once the initial
21939      * drag event has begun.  The drag cannot be canceled from this function.
21940      * @param {Number} x The x position of the click on the dragged object
21941      * @param {Number} y The y position of the click on the dragged object
21942      */
21943     onStartDrag : Roo.emptyFn,
21944
21945     // private - YUI override
21946     startDrag : function(x, y){
21947         this.proxy.reset();
21948         this.dragging = true;
21949         this.proxy.update("");
21950         this.onInitDrag(x, y);
21951         this.proxy.show();
21952     },
21953
21954     // private
21955     onInitDrag : function(x, y){
21956         var clone = this.el.dom.cloneNode(true);
21957         clone.id = Roo.id(); // prevent duplicate ids
21958         this.proxy.update(clone);
21959         this.onStartDrag(x, y);
21960         return true;
21961     },
21962
21963     /**
21964      * Returns the drag source's underlying {@link Roo.dd.StatusProxy}
21965      * @return {Roo.dd.StatusProxy} proxy The StatusProxy
21966      */
21967     getProxy : function(){
21968         return this.proxy;  
21969     },
21970
21971     /**
21972      * Hides the drag source's {@link Roo.dd.StatusProxy}
21973      */
21974     hideProxy : function(){
21975         this.proxy.hide();  
21976         this.proxy.reset(true);
21977         this.dragging = false;
21978     },
21979
21980     // private
21981     triggerCacheRefresh : function(){
21982         Roo.dd.DDM.refreshCache(this.groups);
21983     },
21984
21985     // private - override to prevent hiding
21986     b4EndDrag: function(e) {
21987     },
21988
21989     // private - override to prevent moving
21990     endDrag : function(e){
21991         this.onEndDrag(this.dragData, e);
21992     },
21993
21994     // private
21995     onEndDrag : function(data, e){
21996     },
21997     
21998     // private - pin to cursor
21999     autoOffset : function(x, y) {
22000         this.setDelta(-12, -20);
22001     }    
22002 });/*
22003  * Based on:
22004  * Ext JS Library 1.1.1
22005  * Copyright(c) 2006-2007, Ext JS, LLC.
22006  *
22007  * Originally Released Under LGPL - original licence link has changed is not relivant.
22008  *
22009  * Fork - LGPL
22010  * <script type="text/javascript">
22011  */
22012
22013
22014 /**
22015  * @class Roo.dd.DropTarget
22016  * @extends Roo.dd.DDTarget
22017  * A simple class that provides the basic implementation needed to make any element a drop target that can have
22018  * draggable items dropped onto it.  The drop has no effect until an implementation of notifyDrop is provided.
22019  * @constructor
22020  * @param {String/HTMLElement/Element} el The container element
22021  * @param {Object} config
22022  */
22023 Roo.dd.DropTarget = function(el, config){
22024     this.el = Roo.get(el);
22025     
22026     var listeners = false; ;
22027     if (config && config.listeners) {
22028         listeners= config.listeners;
22029         delete config.listeners;
22030     }
22031     Roo.apply(this, config);
22032     
22033     if(this.containerScroll){
22034         Roo.dd.ScrollManager.register(this.el);
22035     }
22036     this.addEvents( {
22037          /**
22038          * @scope Roo.dd.DropTarget
22039          */
22040          
22041          /**
22042          * @event enter
22043          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source is now over the
22044          * target.  This default implementation adds the CSS class specified by overClass (if any) to the drop element
22045          * and returns the dropAllowed config value.  This method should be overridden if drop validation is required.
22046          * 
22047          * IMPORTANT : it should set this.overClass and this.dropAllowed
22048          * 
22049          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22050          * @param {Event} e The event
22051          * @param {Object} data An object containing arbitrary data supplied by the drag source
22052          */
22053         "enter" : true,
22054         
22055          /**
22056          * @event over
22057          * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the target.
22058          * This method will be called on every mouse movement while the drag source is over the drop target.
22059          * This default implementation simply returns the dropAllowed config value.
22060          * 
22061          * IMPORTANT : it should set this.dropAllowed
22062          * 
22063          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22064          * @param {Event} e The event
22065          * @param {Object} data An object containing arbitrary data supplied by the drag source
22066          
22067          */
22068         "over" : true,
22069         /**
22070          * @event out
22071          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the source has been dragged
22072          * out of the target without dropping.  This default implementation simply removes the CSS class specified by
22073          * overClass (if any) from the drop element.
22074          * 
22075          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22076          * @param {Event} e The event
22077          * @param {Object} data An object containing arbitrary data supplied by the drag source
22078          */
22079          "out" : true,
22080          
22081         /**
22082          * @event drop
22083          * The function a {@link Roo.dd.DragSource} calls once to notify this drop target that the dragged item has
22084          * been dropped on it.  This method has no default implementation and returns false, so you must provide an
22085          * implementation that does something to process the drop event and returns true so that the drag source's
22086          * repair action does not run.
22087          * 
22088          * IMPORTANT : it should set this.success
22089          * 
22090          * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22091          * @param {Event} e The event
22092          * @param {Object} data An object containing arbitrary data supplied by the drag source
22093         */
22094          "drop" : true
22095     });
22096             
22097      
22098     Roo.dd.DropTarget.superclass.constructor.call(  this, 
22099         this.el.dom, 
22100         this.ddGroup || this.group,
22101         {
22102             isTarget: true,
22103             listeners : listeners || {} 
22104            
22105         
22106         }
22107     );
22108
22109 };
22110
22111 Roo.extend(Roo.dd.DropTarget, Roo.dd.DDTarget, {
22112     /**
22113      * @cfg {String} overClass
22114      * The CSS class applied to the drop target element while the drag source is over it (defaults to "").
22115      */
22116      /**
22117      * @cfg {String} ddGroup
22118      * The drag drop group to handle drop events for
22119      */
22120      
22121     /**
22122      * @cfg {String} dropAllowed
22123      * The CSS class returned to the drag source when drop is allowed (defaults to "x-dd-drop-ok").
22124      */
22125     dropAllowed : "x-dd-drop-ok",
22126     /**
22127      * @cfg {String} dropNotAllowed
22128      * The CSS class returned to the drag source when drop is not allowed (defaults to "x-dd-drop-nodrop").
22129      */
22130     dropNotAllowed : "x-dd-drop-nodrop",
22131     /**
22132      * @cfg {boolean} success
22133      * set this after drop listener.. 
22134      */
22135     success : false,
22136     /**
22137      * @cfg {boolean|String} valid true/false or string (ok-add/ok-sub/ok/nodrop)
22138      * if the drop point is valid for over/enter..
22139      */
22140     valid : false,
22141     // private
22142     isTarget : true,
22143
22144     // private
22145     isNotifyTarget : true,
22146     
22147     /**
22148      * @hide
22149      */
22150     notifyEnter : function(dd, e, data)
22151     {
22152         this.valid = true;
22153         this.fireEvent('enter', dd, e, data);
22154         if(this.overClass){
22155             this.el.addClass(this.overClass);
22156         }
22157         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22158             this.valid ? this.dropAllowed : this.dropNotAllowed
22159         );
22160     },
22161
22162     /**
22163      * @hide
22164      */
22165     notifyOver : function(dd, e, data)
22166     {
22167         this.valid = true;
22168         this.fireEvent('over', dd, e, data);
22169         return typeof(this.valid) == 'string' ? 'x-dd-drop-' + this.valid : (
22170             this.valid ? this.dropAllowed : this.dropNotAllowed
22171         );
22172     },
22173
22174     /**
22175      * @hide
22176      */
22177     notifyOut : function(dd, e, data)
22178     {
22179         this.fireEvent('out', dd, e, data);
22180         if(this.overClass){
22181             this.el.removeClass(this.overClass);
22182         }
22183     },
22184
22185     /**
22186      * @hide
22187      */
22188     notifyDrop : function(dd, e, data)
22189     {
22190         this.success = false;
22191         this.fireEvent('drop', dd, e, data);
22192         return this.success;
22193     }
22194 });/*
22195  * Based on:
22196  * Ext JS Library 1.1.1
22197  * Copyright(c) 2006-2007, Ext JS, LLC.
22198  *
22199  * Originally Released Under LGPL - original licence link has changed is not relivant.
22200  *
22201  * Fork - LGPL
22202  * <script type="text/javascript">
22203  */
22204
22205
22206 /**
22207  * @class Roo.dd.DragZone
22208  * @extends Roo.dd.DragSource
22209  * This class provides a container DD instance that proxies for multiple child node sources.<br />
22210  * By default, this class requires that draggable child nodes are registered with {@link Roo.dd.Registry}.
22211  * @constructor
22212  * @param {String/HTMLElement/Element} el The container element
22213  * @param {Object} config
22214  */
22215 Roo.dd.DragZone = function(el, config){
22216     Roo.dd.DragZone.superclass.constructor.call(this, el, config);
22217     if(this.containerScroll){
22218         Roo.dd.ScrollManager.register(this.el);
22219     }
22220 };
22221
22222 Roo.extend(Roo.dd.DragZone, Roo.dd.DragSource, {
22223     /**
22224      * @cfg {Boolean} containerScroll True to register this container with the Scrollmanager
22225      * for auto scrolling during drag operations.
22226      */
22227     /**
22228      * @cfg {String} hlColor The color to use when visually highlighting the drag source in the afterRepair
22229      * method after a failed drop (defaults to "c3daf9" - light blue)
22230      */
22231
22232     /**
22233      * Called when a mousedown occurs in this container. Looks in {@link Roo.dd.Registry}
22234      * for a valid target to drag based on the mouse down. Override this method
22235      * to provide your own lookup logic (e.g. finding a child by class name). Make sure your returned
22236      * object has a "ddel" attribute (with an HTML Element) for other functions to work.
22237      * @param {EventObject} e The mouse down event
22238      * @return {Object} The dragData
22239      */
22240     getDragData : function(e){
22241         return Roo.dd.Registry.getHandleFromEvent(e);
22242     },
22243     
22244     /**
22245      * Called once drag threshold has been reached to initialize the proxy element. By default, it clones the
22246      * this.dragData.ddel
22247      * @param {Number} x The x position of the click on the dragged object
22248      * @param {Number} y The y position of the click on the dragged object
22249      * @return {Boolean} true to continue the drag, false to cancel
22250      */
22251     onInitDrag : function(x, y){
22252         this.proxy.update(this.dragData.ddel.cloneNode(true));
22253         this.onStartDrag(x, y);
22254         return true;
22255     },
22256     
22257     /**
22258      * Called after a repair of an invalid drop. By default, highlights this.dragData.ddel 
22259      */
22260     afterRepair : function(){
22261         if(Roo.enableFx){
22262             Roo.Element.fly(this.dragData.ddel).highlight(this.hlColor || "c3daf9");
22263         }
22264         this.dragging = false;
22265     },
22266
22267     /**
22268      * Called before a repair of an invalid drop to get the XY to animate to. By default returns
22269      * the XY of this.dragData.ddel
22270      * @param {EventObject} e The mouse up event
22271      * @return {Array} The xy location (e.g. [100, 200])
22272      */
22273     getRepairXY : function(e){
22274         return Roo.Element.fly(this.dragData.ddel).getXY();  
22275     }
22276 });/*
22277  * Based on:
22278  * Ext JS Library 1.1.1
22279  * Copyright(c) 2006-2007, Ext JS, LLC.
22280  *
22281  * Originally Released Under LGPL - original licence link has changed is not relivant.
22282  *
22283  * Fork - LGPL
22284  * <script type="text/javascript">
22285  */
22286 /**
22287  * @class Roo.dd.DropZone
22288  * @extends Roo.dd.DropTarget
22289  * This class provides a container DD instance that proxies for multiple child node targets.<br />
22290  * By default, this class requires that child nodes accepting drop are registered with {@link Roo.dd.Registry}.
22291  * @constructor
22292  * @param {String/HTMLElement/Element} el The container element
22293  * @param {Object} config
22294  */
22295 Roo.dd.DropZone = function(el, config){
22296     Roo.dd.DropZone.superclass.constructor.call(this, el, config);
22297 };
22298
22299 Roo.extend(Roo.dd.DropZone, Roo.dd.DropTarget, {
22300     /**
22301      * Returns a custom data object associated with the DOM node that is the target of the event.  By default
22302      * this looks up the event target in the {@link Roo.dd.Registry}, although you can override this method to
22303      * provide your own custom lookup.
22304      * @param {Event} e The event
22305      * @return {Object} data The custom data
22306      */
22307     getTargetFromEvent : function(e){
22308         return Roo.dd.Registry.getTargetFromEvent(e);
22309     },
22310
22311     /**
22312      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has entered a drop node
22313      * that it has registered.  This method has no default implementation and should be overridden to provide
22314      * node-specific processing if necessary.
22315      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from 
22316      * {@link #getTargetFromEvent} for this node)
22317      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22318      * @param {Event} e The event
22319      * @param {Object} data An object containing arbitrary data supplied by the drag source
22320      */
22321     onNodeEnter : function(n, dd, e, data){
22322         
22323     },
22324
22325     /**
22326      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is over a drop node
22327      * that it has registered.  The default implementation returns this.dropNotAllowed, so it should be
22328      * overridden to provide the proper feedback.
22329      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22330      * {@link #getTargetFromEvent} for this node)
22331      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22332      * @param {Event} e The event
22333      * @param {Object} data An object containing arbitrary data supplied by the drag source
22334      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22335      * underlying {@link Roo.dd.StatusProxy} can be updated
22336      */
22337     onNodeOver : function(n, dd, e, data){
22338         return this.dropAllowed;
22339     },
22340
22341     /**
22342      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dragged out of
22343      * the drop node without dropping.  This method has no default implementation and should be overridden to provide
22344      * node-specific processing if necessary.
22345      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22346      * {@link #getTargetFromEvent} for this node)
22347      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22348      * @param {Event} e The event
22349      * @param {Object} data An object containing arbitrary data supplied by the drag source
22350      */
22351     onNodeOut : function(n, dd, e, data){
22352         
22353     },
22354
22355     /**
22356      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped onto
22357      * the drop node.  The default implementation returns false, so it should be overridden to provide the
22358      * appropriate processing of the drop event and return true so that the drag source's repair action does not run.
22359      * @param {Object} nodeData The custom data associated with the drop node (this is the same value returned from
22360      * {@link #getTargetFromEvent} for this node)
22361      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22362      * @param {Event} e The event
22363      * @param {Object} data An object containing arbitrary data supplied by the drag source
22364      * @return {Boolean} True if the drop was valid, else false
22365      */
22366     onNodeDrop : function(n, dd, e, data){
22367         return false;
22368     },
22369
22370     /**
22371      * Called internally while the DropZone determines that a {@link Roo.dd.DragSource} is being dragged over it,
22372      * but not over any of its registered drop nodes.  The default implementation returns this.dropNotAllowed, so
22373      * it should be overridden to provide the proper feedback if necessary.
22374      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22375      * @param {Event} e The event
22376      * @param {Object} data An object containing arbitrary data supplied by the drag source
22377      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22378      * underlying {@link Roo.dd.StatusProxy} can be updated
22379      */
22380     onContainerOver : function(dd, e, data){
22381         return this.dropNotAllowed;
22382     },
22383
22384     /**
22385      * Called internally when the DropZone determines that a {@link Roo.dd.DragSource} has been dropped on it,
22386      * but not on any of its registered drop nodes.  The default implementation returns false, so it should be
22387      * overridden to provide the appropriate processing of the drop event if you need the drop zone itself to
22388      * be able to accept drops.  It should return true when valid so that the drag source's repair action does not run.
22389      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22390      * @param {Event} e The event
22391      * @param {Object} data An object containing arbitrary data supplied by the drag source
22392      * @return {Boolean} True if the drop was valid, else false
22393      */
22394     onContainerDrop : function(dd, e, data){
22395         return false;
22396     },
22397
22398     /**
22399      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source is now over
22400      * the zone.  The default implementation returns this.dropNotAllowed and expects that only registered drop
22401      * nodes can process drag drop operations, so if you need the drop zone itself to be able to process drops
22402      * you should override this method and provide a custom implementation.
22403      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22404      * @param {Event} e The event
22405      * @param {Object} data An object containing arbitrary data supplied by the drag source
22406      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22407      * underlying {@link Roo.dd.StatusProxy} can be updated
22408      */
22409     notifyEnter : function(dd, e, data){
22410         return this.dropNotAllowed;
22411     },
22412
22413     /**
22414      * The function a {@link Roo.dd.DragSource} calls continuously while it is being dragged over the drop zone.
22415      * This method will be called on every mouse movement while the drag source is over the drop zone.
22416      * It will call {@link #onNodeOver} while the drag source is over a registered node, and will also automatically
22417      * delegate to the appropriate node-specific methods as necessary when the drag source enters and exits
22418      * registered nodes ({@link #onNodeEnter}, {@link #onNodeOut}). If the drag source is not currently over a
22419      * registered node, it will call {@link #onContainerOver}.
22420      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22421      * @param {Event} e The event
22422      * @param {Object} data An object containing arbitrary data supplied by the drag source
22423      * @return {String} status The CSS class that communicates the drop status back to the source so that the
22424      * underlying {@link Roo.dd.StatusProxy} can be updated
22425      */
22426     notifyOver : function(dd, e, data){
22427         var n = this.getTargetFromEvent(e);
22428         if(!n){ // not over valid drop target
22429             if(this.lastOverNode){
22430                 this.onNodeOut(this.lastOverNode, dd, e, data);
22431                 this.lastOverNode = null;
22432             }
22433             return this.onContainerOver(dd, e, data);
22434         }
22435         if(this.lastOverNode != n){
22436             if(this.lastOverNode){
22437                 this.onNodeOut(this.lastOverNode, dd, e, data);
22438             }
22439             this.onNodeEnter(n, dd, e, data);
22440             this.lastOverNode = n;
22441         }
22442         return this.onNodeOver(n, dd, e, data);
22443     },
22444
22445     /**
22446      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the source has been dragged
22447      * out of the zone without dropping.  If the drag source is currently over a registered node, the notification
22448      * will be delegated to {@link #onNodeOut} for node-specific handling, otherwise it will be ignored.
22449      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop target
22450      * @param {Event} e The event
22451      * @param {Object} data An object containing arbitrary data supplied by the drag zone
22452      */
22453     notifyOut : function(dd, e, data){
22454         if(this.lastOverNode){
22455             this.onNodeOut(this.lastOverNode, dd, e, data);
22456             this.lastOverNode = null;
22457         }
22458     },
22459
22460     /**
22461      * The function a {@link Roo.dd.DragSource} calls once to notify this drop zone that the dragged item has
22462      * been dropped on it.  The drag zone will look up the target node based on the event passed in, and if there
22463      * is a node registered for that event, it will delegate to {@link #onNodeDrop} for node-specific handling,
22464      * otherwise it will call {@link #onContainerDrop}.
22465      * @param {Roo.dd.DragSource} source The drag source that was dragged over this drop zone
22466      * @param {Event} e The event
22467      * @param {Object} data An object containing arbitrary data supplied by the drag source
22468      * @return {Boolean} True if the drop was valid, else false
22469      */
22470     notifyDrop : function(dd, e, data){
22471         if(this.lastOverNode){
22472             this.onNodeOut(this.lastOverNode, dd, e, data);
22473             this.lastOverNode = null;
22474         }
22475         var n = this.getTargetFromEvent(e);
22476         return n ?
22477             this.onNodeDrop(n, dd, e, data) :
22478             this.onContainerDrop(dd, e, data);
22479     },
22480
22481     // private
22482     triggerCacheRefresh : function(){
22483         Roo.dd.DDM.refreshCache(this.groups);
22484     }  
22485 });/*
22486  * Based on:
22487  * Ext JS Library 1.1.1
22488  * Copyright(c) 2006-2007, Ext JS, LLC.
22489  *
22490  * Originally Released Under LGPL - original licence link has changed is not relivant.
22491  *
22492  * Fork - LGPL
22493  * <script type="text/javascript">
22494  */
22495
22496
22497 /**
22498  * @class Roo.data.SortTypes
22499  * @singleton
22500  * Defines the default sorting (casting?) comparison functions used when sorting data.
22501  */
22502 Roo.data.SortTypes = {
22503     /**
22504      * Default sort that does nothing
22505      * @param {Mixed} s The value being converted
22506      * @return {Mixed} The comparison value
22507      */
22508     none : function(s){
22509         return s;
22510     },
22511     
22512     /**
22513      * The regular expression used to strip tags
22514      * @type {RegExp}
22515      * @property
22516      */
22517     stripTagsRE : /<\/?[^>]+>/gi,
22518     
22519     /**
22520      * Strips all HTML tags to sort on text only
22521      * @param {Mixed} s The value being converted
22522      * @return {String} The comparison value
22523      */
22524     asText : function(s){
22525         return String(s).replace(this.stripTagsRE, "");
22526     },
22527     
22528     /**
22529      * Strips all HTML tags to sort on text only - Case insensitive
22530      * @param {Mixed} s The value being converted
22531      * @return {String} The comparison value
22532      */
22533     asUCText : function(s){
22534         return String(s).toUpperCase().replace(this.stripTagsRE, "");
22535     },
22536     
22537     /**
22538      * Case insensitive string
22539      * @param {Mixed} s The value being converted
22540      * @return {String} The comparison value
22541      */
22542     asUCString : function(s) {
22543         return String(s).toUpperCase();
22544     },
22545     
22546     /**
22547      * Date sorting
22548      * @param {Mixed} s The value being converted
22549      * @return {Number} The comparison value
22550      */
22551     asDate : function(s) {
22552         if(!s){
22553             return 0;
22554         }
22555         if(s instanceof Date){
22556             return s.getTime();
22557         }
22558         return Date.parse(String(s));
22559     },
22560     
22561     /**
22562      * Float sorting
22563      * @param {Mixed} s The value being converted
22564      * @return {Float} The comparison value
22565      */
22566     asFloat : function(s) {
22567         var val = parseFloat(String(s).replace(/,/g, ""));
22568         if(isNaN(val)) {
22569             val = 0;
22570         }
22571         return val;
22572     },
22573     
22574     /**
22575      * Integer sorting
22576      * @param {Mixed} s The value being converted
22577      * @return {Number} The comparison value
22578      */
22579     asInt : function(s) {
22580         var val = parseInt(String(s).replace(/,/g, ""));
22581         if(isNaN(val)) {
22582             val = 0;
22583         }
22584         return val;
22585     }
22586 };/*
22587  * Based on:
22588  * Ext JS Library 1.1.1
22589  * Copyright(c) 2006-2007, Ext JS, LLC.
22590  *
22591  * Originally Released Under LGPL - original licence link has changed is not relivant.
22592  *
22593  * Fork - LGPL
22594  * <script type="text/javascript">
22595  */
22596
22597 /**
22598 * @class Roo.data.Record
22599  * Instances of this class encapsulate both record <em>definition</em> information, and record
22600  * <em>value</em> information for use in {@link Roo.data.Store} objects, or any code which needs
22601  * to access Records cached in an {@link Roo.data.Store} object.<br>
22602  * <p>
22603  * Constructors for this class are generated by passing an Array of field definition objects to {@link #create}.
22604  * Instances are usually only created by {@link Roo.data.Reader} implementations when processing unformatted data
22605  * objects.<br>
22606  * <p>
22607  * Record objects generated by this constructor inherit all the methods of Roo.data.Record listed below.
22608  * @constructor
22609  * This constructor should not be used to create Record objects. Instead, use the constructor generated by
22610  * {@link #create}. The parameters are the same.
22611  * @param {Array} data An associative Array of data values keyed by the field name.
22612  * @param {Object} id (Optional) The id of the record. This id should be unique, and is used by the
22613  * {@link Roo.data.Store} object which owns the Record to index its collection of Records. If
22614  * not specified an integer id is generated.
22615  */
22616 Roo.data.Record = function(data, id){
22617     this.id = (id || id === 0) ? id : ++Roo.data.Record.AUTO_ID;
22618     this.data = data;
22619 };
22620
22621 /**
22622  * Generate a constructor for a specific record layout.
22623  * @param {Array} o An Array of field definition objects which specify field names, and optionally,
22624  * data types, and a mapping for an {@link Roo.data.Reader} to extract the field's value from a data object.
22625  * Each field definition object may contain the following properties: <ul>
22626  * <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,
22627  * for example the <em>dataIndex</em> property in column definition objects passed to {@link Roo.grid.ColumnModel}</p></li>
22628  * <li><b>mapping</b> : String<p style="margin-left:1em">(Optional) A path specification for use by the {@link Roo.data.Reader} implementation
22629  * that is creating the Record to access the data value from the data object. If an {@link Roo.data.JsonReader}
22630  * is being used, then this is a string containing the javascript expression to reference the data relative to 
22631  * the record item's root. If an {@link Roo.data.XmlReader} is being used, this is an {@link Roo.DomQuery} path
22632  * to the data item relative to the record element. If the mapping expression is the same as the field name,
22633  * this may be omitted.</p></li>
22634  * <li><b>type</b> : String<p style="margin-left:1em">(Optional) The data type for conversion to displayable value. Possible values are
22635  * <ul><li>auto (Default, implies no conversion)</li>
22636  * <li>string</li>
22637  * <li>int</li>
22638  * <li>float</li>
22639  * <li>boolean</li>
22640  * <li>date</li></ul></p></li>
22641  * <li><b>sortType</b> : Mixed<p style="margin-left:1em">(Optional) A member of {@link Roo.data.SortTypes}.</p></li>
22642  * <li><b>sortDir</b> : String<p style="margin-left:1em">(Optional) Initial direction to sort. "ASC" or "DESC"</p></li>
22643  * <li><b>convert</b> : Function<p style="margin-left:1em">(Optional) A function which converts the value provided
22644  * by the Reader into an object that will be stored in the Record. It is passed the
22645  * following parameters:<ul>
22646  * <li><b>v</b> : Mixed<p style="margin-left:1em">The data value as read by the Reader.</p></li>
22647  * </ul></p></li>
22648  * <li><b>dateFormat</b> : String<p style="margin-left:1em">(Optional) A format String for the Date.parseDate function.</p></li>
22649  * </ul>
22650  * <br>usage:<br><pre><code>
22651 var TopicRecord = Roo.data.Record.create(
22652     {name: 'title', mapping: 'topic_title'},
22653     {name: 'author', mapping: 'username'},
22654     {name: 'totalPosts', mapping: 'topic_replies', type: 'int'},
22655     {name: 'lastPost', mapping: 'post_time', type: 'date'},
22656     {name: 'lastPoster', mapping: 'user2'},
22657     {name: 'excerpt', mapping: 'post_text'}
22658 );
22659
22660 var myNewRecord = new TopicRecord({
22661     title: 'Do my job please',
22662     author: 'noobie',
22663     totalPosts: 1,
22664     lastPost: new Date(),
22665     lastPoster: 'Animal',
22666     excerpt: 'No way dude!'
22667 });
22668 myStore.add(myNewRecord);
22669 </code></pre>
22670  * @method create
22671  * @static
22672  */
22673 Roo.data.Record.create = function(o){
22674     var f = function(){
22675         f.superclass.constructor.apply(this, arguments);
22676     };
22677     Roo.extend(f, Roo.data.Record);
22678     var p = f.prototype;
22679     p.fields = new Roo.util.MixedCollection(false, function(field){
22680         return field.name;
22681     });
22682     for(var i = 0, len = o.length; i < len; i++){
22683         p.fields.add(new Roo.data.Field(o[i]));
22684     }
22685     f.getField = function(name){
22686         return p.fields.get(name);  
22687     };
22688     return f;
22689 };
22690
22691 Roo.data.Record.AUTO_ID = 1000;
22692 Roo.data.Record.EDIT = 'edit';
22693 Roo.data.Record.REJECT = 'reject';
22694 Roo.data.Record.COMMIT = 'commit';
22695
22696 Roo.data.Record.prototype = {
22697     /**
22698      * Readonly flag - true if this record has been modified.
22699      * @type Boolean
22700      */
22701     dirty : false,
22702     editing : false,
22703     error: null,
22704     modified: null,
22705
22706     // private
22707     join : function(store){
22708         this.store = store;
22709     },
22710
22711     /**
22712      * Set the named field to the specified value.
22713      * @param {String} name The name of the field to set.
22714      * @param {Object} value The value to set the field to.
22715      */
22716     set : function(name, value){
22717         if(this.data[name] == value){
22718             return;
22719         }
22720         this.dirty = true;
22721         if(!this.modified){
22722             this.modified = {};
22723         }
22724         if(typeof this.modified[name] == 'undefined'){
22725             this.modified[name] = this.data[name];
22726         }
22727         this.data[name] = value;
22728         if(!this.editing && this.store){
22729             this.store.afterEdit(this);
22730         }       
22731     },
22732
22733     /**
22734      * Get the value of the named field.
22735      * @param {String} name The name of the field to get the value of.
22736      * @return {Object} The value of the field.
22737      */
22738     get : function(name){
22739         return this.data[name]; 
22740     },
22741
22742     // private
22743     beginEdit : function(){
22744         this.editing = true;
22745         this.modified = {}; 
22746     },
22747
22748     // private
22749     cancelEdit : function(){
22750         this.editing = false;
22751         delete this.modified;
22752     },
22753
22754     // private
22755     endEdit : function(){
22756         this.editing = false;
22757         if(this.dirty && this.store){
22758             this.store.afterEdit(this);
22759         }
22760     },
22761
22762     /**
22763      * Usually called by the {@link Roo.data.Store} which owns the Record.
22764      * Rejects all changes made to the Record since either creation, or the last commit operation.
22765      * Modified fields are reverted to their original values.
22766      * <p>
22767      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22768      * of reject operations.
22769      */
22770     reject : function(){
22771         var m = this.modified;
22772         for(var n in m){
22773             if(typeof m[n] != "function"){
22774                 this.data[n] = m[n];
22775             }
22776         }
22777         this.dirty = false;
22778         delete this.modified;
22779         this.editing = false;
22780         if(this.store){
22781             this.store.afterReject(this);
22782         }
22783     },
22784
22785     /**
22786      * Usually called by the {@link Roo.data.Store} which owns the Record.
22787      * Commits all changes made to the Record since either creation, or the last commit operation.
22788      * <p>
22789      * Developers should subscribe to the {@link Roo.data.Store#update} event to have their code notified
22790      * of commit operations.
22791      */
22792     commit : function(){
22793         this.dirty = false;
22794         delete this.modified;
22795         this.editing = false;
22796         if(this.store){
22797             this.store.afterCommit(this);
22798         }
22799     },
22800
22801     // private
22802     hasError : function(){
22803         return this.error != null;
22804     },
22805
22806     // private
22807     clearError : function(){
22808         this.error = null;
22809     },
22810
22811     /**
22812      * Creates a copy of this record.
22813      * @param {String} id (optional) A new record id if you don't want to use this record's id
22814      * @return {Record}
22815      */
22816     copy : function(newId) {
22817         return new this.constructor(Roo.apply({}, this.data), newId || this.id);
22818     }
22819 };/*
22820  * Based on:
22821  * Ext JS Library 1.1.1
22822  * Copyright(c) 2006-2007, Ext JS, LLC.
22823  *
22824  * Originally Released Under LGPL - original licence link has changed is not relivant.
22825  *
22826  * Fork - LGPL
22827  * <script type="text/javascript">
22828  */
22829
22830
22831
22832 /**
22833  * @class Roo.data.Store
22834  * @extends Roo.util.Observable
22835  * The Store class encapsulates a client side cache of {@link Roo.data.Record} objects which provide input data
22836  * for widgets such as the Roo.grid.Grid, or the Roo.form.ComboBox.<br>
22837  * <p>
22838  * 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
22839  * has no knowledge of the format of the data returned by the Proxy.<br>
22840  * <p>
22841  * A Store object uses its configured implementation of {@link Roo.data.DataReader} to create {@link Roo.data.Record}
22842  * instances from the data object. These records are cached and made available through accessor functions.
22843  * @constructor
22844  * Creates a new Store.
22845  * @param {Object} config A config object containing the objects needed for the Store to access data,
22846  * and read the data into Records.
22847  */
22848 Roo.data.Store = function(config){
22849     this.data = new Roo.util.MixedCollection(false);
22850     this.data.getKey = function(o){
22851         return o.id;
22852     };
22853     this.baseParams = {};
22854     // private
22855     this.paramNames = {
22856         "start" : "start",
22857         "limit" : "limit",
22858         "sort" : "sort",
22859         "dir" : "dir",
22860         "multisort" : "_multisort"
22861     };
22862
22863     if(config && config.data){
22864         this.inlineData = config.data;
22865         delete config.data;
22866     }
22867
22868     Roo.apply(this, config);
22869     
22870     if(this.reader){ // reader passed
22871         this.reader = Roo.factory(this.reader, Roo.data);
22872         this.reader.xmodule = this.xmodule || false;
22873         if(!this.recordType){
22874             this.recordType = this.reader.recordType;
22875         }
22876         if(this.reader.onMetaChange){
22877             this.reader.onMetaChange = this.onMetaChange.createDelegate(this);
22878         }
22879     }
22880
22881     if(this.recordType){
22882         this.fields = this.recordType.prototype.fields;
22883     }
22884     this.modified = [];
22885
22886     this.addEvents({
22887         /**
22888          * @event datachanged
22889          * Fires when the data cache has changed, and a widget which is using this Store
22890          * as a Record cache should refresh its view.
22891          * @param {Store} this
22892          */
22893         datachanged : true,
22894         /**
22895          * @event metachange
22896          * Fires when this store's reader provides new metadata (fields). This is currently only support for JsonReaders.
22897          * @param {Store} this
22898          * @param {Object} meta The JSON metadata
22899          */
22900         metachange : true,
22901         /**
22902          * @event add
22903          * Fires when Records have been added to the Store
22904          * @param {Store} this
22905          * @param {Roo.data.Record[]} records The array of Records added
22906          * @param {Number} index The index at which the record(s) were added
22907          */
22908         add : true,
22909         /**
22910          * @event remove
22911          * Fires when a Record has been removed from the Store
22912          * @param {Store} this
22913          * @param {Roo.data.Record} record The Record that was removed
22914          * @param {Number} index The index at which the record was removed
22915          */
22916         remove : true,
22917         /**
22918          * @event update
22919          * Fires when a Record has been updated
22920          * @param {Store} this
22921          * @param {Roo.data.Record} record The Record that was updated
22922          * @param {String} operation The update operation being performed.  Value may be one of:
22923          * <pre><code>
22924  Roo.data.Record.EDIT
22925  Roo.data.Record.REJECT
22926  Roo.data.Record.COMMIT
22927          * </code></pre>
22928          */
22929         update : true,
22930         /**
22931          * @event clear
22932          * Fires when the data cache has been cleared.
22933          * @param {Store} this
22934          */
22935         clear : true,
22936         /**
22937          * @event beforeload
22938          * Fires before a request is made for a new data object.  If the beforeload handler returns false
22939          * the load action will be canceled.
22940          * @param {Store} this
22941          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22942          */
22943         beforeload : true,
22944         /**
22945          * @event beforeloadadd
22946          * Fires after a new set of Records has been loaded.
22947          * @param {Store} this
22948          * @param {Roo.data.Record[]} records The Records that were loaded
22949          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22950          */
22951         beforeloadadd : true,
22952         /**
22953          * @event load
22954          * Fires after a new set of Records has been loaded, before they are added to the store.
22955          * @param {Store} this
22956          * @param {Roo.data.Record[]} records The Records that were loaded
22957          * @param {Object} options The loading options that were specified (see {@link #load} for details)
22958          * @params {Object} return from reader
22959          */
22960         load : true,
22961         /**
22962          * @event loadexception
22963          * Fires if an exception occurs in the Proxy during loading.
22964          * Called with the signature of the Proxy's "loadexception" event.
22965          * If you return Json { data: [] , success: false, .... } then this will be thrown with the following args
22966          * 
22967          * @param {Proxy} 
22968          * @param {Object} return from JsonData.reader() - success, totalRecords, records
22969          * @param {Object} load options 
22970          * @param {Object} jsonData from your request (normally this contains the Exception)
22971          */
22972         loadexception : true
22973     });
22974     
22975     if(this.proxy){
22976         this.proxy = Roo.factory(this.proxy, Roo.data);
22977         this.proxy.xmodule = this.xmodule || false;
22978         this.relayEvents(this.proxy,  ["loadexception"]);
22979     }
22980     this.sortToggle = {};
22981     this.sortOrder = []; // array of order of sorting - updated by grid if multisort is enabled.
22982
22983     Roo.data.Store.superclass.constructor.call(this);
22984
22985     if(this.inlineData){
22986         this.loadData(this.inlineData);
22987         delete this.inlineData;
22988     }
22989 };
22990
22991 Roo.extend(Roo.data.Store, Roo.util.Observable, {
22992      /**
22993     * @cfg {boolean} isLocal   flag if data is locally available (and can be always looked up
22994     * without a remote query - used by combo/forms at present.
22995     */
22996     
22997     /**
22998     * @cfg {Roo.data.DataProxy} proxy The Proxy object which provides access to a data object.
22999     */
23000     /**
23001     * @cfg {Array} data Inline data to be loaded when the store is initialized.
23002     */
23003     /**
23004     * @cfg {Roo.data.Reader} reader The Reader object which processes the data object and returns
23005     * an Array of Roo.data.record objects which are cached keyed by their <em>id</em> property.
23006     */
23007     /**
23008     * @cfg {Object} baseParams An object containing properties which are to be sent as parameters
23009     * on any HTTP request
23010     */
23011     /**
23012     * @cfg {Object} sortInfo A config object in the format: {field: "fieldName", direction: "ASC|DESC"}
23013     */
23014     /**
23015     * @cfg {Boolean} multiSort enable multi column sorting (sort is based on the order of columns, remote only at present)
23016     */
23017     multiSort: false,
23018     /**
23019     * @cfg {boolean} remoteSort True if sorting is to be handled by requesting the Proxy to provide a refreshed
23020     * version of the data object in sorted order, as opposed to sorting the Record cache in place (defaults to false).
23021     */
23022     remoteSort : false,
23023
23024     /**
23025     * @cfg {boolean} pruneModifiedRecords True to clear all modified record information each time the store is
23026      * loaded or when a record is removed. (defaults to false).
23027     */
23028     pruneModifiedRecords : false,
23029
23030     // private
23031     lastOptions : null,
23032
23033     /**
23034      * Add Records to the Store and fires the add event.
23035      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23036      */
23037     add : function(records){
23038         records = [].concat(records);
23039         for(var i = 0, len = records.length; i < len; i++){
23040             records[i].join(this);
23041         }
23042         var index = this.data.length;
23043         this.data.addAll(records);
23044         this.fireEvent("add", this, records, index);
23045     },
23046
23047     /**
23048      * Remove a Record from the Store and fires the remove event.
23049      * @param {Ext.data.Record} record The Roo.data.Record object to remove from the cache.
23050      */
23051     remove : function(record){
23052         var index = this.data.indexOf(record);
23053         this.data.removeAt(index);
23054  
23055         if(this.pruneModifiedRecords){
23056             this.modified.remove(record);
23057         }
23058         this.fireEvent("remove", this, record, index);
23059     },
23060
23061     /**
23062      * Remove all Records from the Store and fires the clear event.
23063      */
23064     removeAll : function(){
23065         this.data.clear();
23066         if(this.pruneModifiedRecords){
23067             this.modified = [];
23068         }
23069         this.fireEvent("clear", this);
23070     },
23071
23072     /**
23073      * Inserts Records to the Store at the given index and fires the add event.
23074      * @param {Number} index The start index at which to insert the passed Records.
23075      * @param {Roo.data.Record[]} records An Array of Roo.data.Record objects to add to the cache.
23076      */
23077     insert : function(index, records){
23078         records = [].concat(records);
23079         for(var i = 0, len = records.length; i < len; i++){
23080             this.data.insert(index, records[i]);
23081             records[i].join(this);
23082         }
23083         this.fireEvent("add", this, records, index);
23084     },
23085
23086     /**
23087      * Get the index within the cache of the passed Record.
23088      * @param {Roo.data.Record} record The Roo.data.Record object to to find.
23089      * @return {Number} The index of the passed Record. Returns -1 if not found.
23090      */
23091     indexOf : function(record){
23092         return this.data.indexOf(record);
23093     },
23094
23095     /**
23096      * Get the index within the cache of the Record with the passed id.
23097      * @param {String} id The id of the Record to find.
23098      * @return {Number} The index of the Record. Returns -1 if not found.
23099      */
23100     indexOfId : function(id){
23101         return this.data.indexOfKey(id);
23102     },
23103
23104     /**
23105      * Get the Record with the specified id.
23106      * @param {String} id The id of the Record to find.
23107      * @return {Roo.data.Record} The Record with the passed id. Returns undefined if not found.
23108      */
23109     getById : function(id){
23110         return this.data.key(id);
23111     },
23112
23113     /**
23114      * Get the Record at the specified index.
23115      * @param {Number} index The index of the Record to find.
23116      * @return {Roo.data.Record} The Record at the passed index. Returns undefined if not found.
23117      */
23118     getAt : function(index){
23119         return this.data.itemAt(index);
23120     },
23121
23122     /**
23123      * Returns a range of Records between specified indices.
23124      * @param {Number} startIndex (optional) The starting index (defaults to 0)
23125      * @param {Number} endIndex (optional) The ending index (defaults to the last Record in the Store)
23126      * @return {Roo.data.Record[]} An array of Records
23127      */
23128     getRange : function(start, end){
23129         return this.data.getRange(start, end);
23130     },
23131
23132     // private
23133     storeOptions : function(o){
23134         o = Roo.apply({}, o);
23135         delete o.callback;
23136         delete o.scope;
23137         this.lastOptions = o;
23138     },
23139
23140     /**
23141      * Loads the Record cache from the configured Proxy using the configured Reader.
23142      * <p>
23143      * If using remote paging, then the first load call must specify the <em>start</em>
23144      * and <em>limit</em> properties in the options.params property to establish the initial
23145      * position within the dataset, and the number of Records to cache on each read from the Proxy.
23146      * <p>
23147      * <strong>It is important to note that for remote data sources, loading is asynchronous,
23148      * and this call will return before the new data has been loaded. Perform any post-processing
23149      * in a callback function, or in a "load" event handler.</strong>
23150      * <p>
23151      * @param {Object} options An object containing properties which control loading options:<ul>
23152      * <li>params {Object} An object containing properties to pass as HTTP parameters to a remote data source.</li>
23153      * <li>callback {Function} A function to be called after the Records have been loaded. The callback is
23154      * passed the following arguments:<ul>
23155      * <li>r : Roo.data.Record[]</li>
23156      * <li>options: Options object from the load call</li>
23157      * <li>success: Boolean success indicator</li></ul></li>
23158      * <li>scope {Object} Scope with which to call the callback (defaults to the Store object)</li>
23159      * <li>add {Boolean} indicator to append loaded records rather than replace the current cache.</li>
23160      * </ul>
23161      */
23162     load : function(options){
23163         options = options || {};
23164         if(this.fireEvent("beforeload", this, options) !== false){
23165             this.storeOptions(options);
23166             var p = Roo.apply(options.params || {}, this.baseParams);
23167             // if meta was not loaded from remote source.. try requesting it.
23168             if (!this.reader.metaFromRemote) {
23169                 p._requestMeta = 1;
23170             }
23171             if(this.sortInfo && this.remoteSort){
23172                 var pn = this.paramNames;
23173                 p[pn["sort"]] = this.sortInfo.field;
23174                 p[pn["dir"]] = this.sortInfo.direction;
23175             }
23176             if (this.multiSort) {
23177                 var pn = this.paramNames;
23178                 p[pn["multisort"]] = Roo.encode( { sort : this.sortToggle, order: this.sortOrder });
23179             }
23180             
23181             this.proxy.load(p, this.reader, this.loadRecords, this, options);
23182         }
23183     },
23184
23185     /**
23186      * Reloads the Record cache from the configured Proxy using the configured Reader and
23187      * the options from the last load operation performed.
23188      * @param {Object} options (optional) An object containing properties which may override the options
23189      * used in the last load operation. See {@link #load} for details (defaults to null, in which case
23190      * the most recently used options are reused).
23191      */
23192     reload : function(options){
23193         this.load(Roo.applyIf(options||{}, this.lastOptions));
23194     },
23195
23196     // private
23197     // Called as a callback by the Reader during a load operation.
23198     loadRecords : function(o, options, success){
23199         if(!o || success === false){
23200             if(success !== false){
23201                 this.fireEvent("load", this, [], options, o);
23202             }
23203             if(options.callback){
23204                 options.callback.call(options.scope || this, [], options, false);
23205             }
23206             return;
23207         }
23208         // if data returned failure - throw an exception.
23209         if (o.success === false) {
23210             // show a message if no listener is registered.
23211             if (!this.hasListener('loadexception') && typeof(o.raw.errorMsg) != 'undefined') {
23212                     Roo.MessageBox.alert("Error loading",o.raw.errorMsg);
23213             }
23214             // loadmask wil be hooked into this..
23215             this.fireEvent("loadexception", this, o, options, o.raw.errorMsg);
23216             return;
23217         }
23218         var r = o.records, t = o.totalRecords || r.length;
23219         
23220         this.fireEvent("beforeloadadd", this, r, options, o);
23221         
23222         if(!options || options.add !== true){
23223             if(this.pruneModifiedRecords){
23224                 this.modified = [];
23225             }
23226             for(var i = 0, len = r.length; i < len; i++){
23227                 r[i].join(this);
23228             }
23229             if(this.snapshot){
23230                 this.data = this.snapshot;
23231                 delete this.snapshot;
23232             }
23233             this.data.clear();
23234             this.data.addAll(r);
23235             this.totalLength = t;
23236             this.applySort();
23237             this.fireEvent("datachanged", this);
23238         }else{
23239             this.totalLength = Math.max(t, this.data.length+r.length);
23240             this.add(r);
23241         }
23242         
23243         if(this.parent && !Roo.isIOS && !this.useNativeIOS && this.parent.emptyTitle.length) {
23244                 
23245             var e = new Roo.data.Record({});
23246
23247             e.set(this.parent.displayField, this.parent.emptyTitle);
23248             e.set(this.parent.valueField, '');
23249
23250             this.insert(0, e);
23251         }
23252             
23253         this.fireEvent("load", this, r, options, o);
23254         if(options.callback){
23255             options.callback.call(options.scope || this, r, options, true);
23256         }
23257     },
23258
23259
23260     /**
23261      * Loads data from a passed data block. A Reader which understands the format of the data
23262      * must have been configured in the constructor.
23263      * @param {Object} data The data block from which to read the Records.  The format of the data expected
23264      * is dependent on the type of Reader that is configured and should correspond to that Reader's readRecords parameter.
23265      * @param {Boolean} append (Optional) True to append the new Records rather than replace the existing cache.
23266      */
23267     loadData : function(o, append){
23268         var r = this.reader.readRecords(o);
23269         this.loadRecords(r, {add: append}, true);
23270     },
23271     
23272      /**
23273      * using 'cn' the nested child reader read the child array into it's child stores.
23274      * @param {Object} rec The record with a 'children array
23275      */
23276     loadDataFromChildren : function(rec)
23277     {
23278         this.loadData(this.reader.toLoadData(rec));
23279     },
23280     
23281
23282     /**
23283      * Gets the number of cached records.
23284      * <p>
23285      * <em>If using paging, this may not be the total size of the dataset. If the data object
23286      * used by the Reader contains the dataset size, then the getTotalCount() function returns
23287      * the data set size</em>
23288      */
23289     getCount : function(){
23290         return this.data.length || 0;
23291     },
23292
23293     /**
23294      * Gets the total number of records in the dataset as returned by the server.
23295      * <p>
23296      * <em>If using paging, for this to be accurate, the data object used by the Reader must contain
23297      * the dataset size</em>
23298      */
23299     getTotalCount : function(){
23300         return this.totalLength || 0;
23301     },
23302
23303     /**
23304      * Returns the sort state of the Store as an object with two properties:
23305      * <pre><code>
23306  field {String} The name of the field by which the Records are sorted
23307  direction {String} The sort order, "ASC" or "DESC"
23308      * </code></pre>
23309      */
23310     getSortState : function(){
23311         return this.sortInfo;
23312     },
23313
23314     // private
23315     applySort : function(){
23316         if(this.sortInfo && !this.remoteSort){
23317             var s = this.sortInfo, f = s.field;
23318             var st = this.fields.get(f).sortType;
23319             var fn = function(r1, r2){
23320                 var v1 = st(r1.data[f]), v2 = st(r2.data[f]);
23321                 return v1 > v2 ? 1 : (v1 < v2 ? -1 : 0);
23322             };
23323             this.data.sort(s.direction, fn);
23324             if(this.snapshot && this.snapshot != this.data){
23325                 this.snapshot.sort(s.direction, fn);
23326             }
23327         }
23328     },
23329
23330     /**
23331      * Sets the default sort column and order to be used by the next load operation.
23332      * @param {String} fieldName The name of the field to sort by.
23333      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23334      */
23335     setDefaultSort : function(field, dir){
23336         this.sortInfo = {field: field, direction: dir ? dir.toUpperCase() : "ASC"};
23337     },
23338
23339     /**
23340      * Sort the Records.
23341      * If remote sorting is used, the sort is performed on the server, and the cache is
23342      * reloaded. If local sorting is used, the cache is sorted internally.
23343      * @param {String} fieldName The name of the field to sort by.
23344      * @param {String} dir (optional) The sort order, "ASC" or "DESC" (defaults to "ASC")
23345      */
23346     sort : function(fieldName, dir){
23347         var f = this.fields.get(fieldName);
23348         if(!dir){
23349             this.sortToggle[f.name] = this.sortToggle[f.name] || f.sortDir;
23350             
23351             if(this.multiSort || (this.sortInfo && this.sortInfo.field == f.name) ){ // toggle sort dir
23352                 dir = (this.sortToggle[f.name] || "ASC").toggle("ASC", "DESC");
23353             }else{
23354                 dir = f.sortDir;
23355             }
23356         }
23357         this.sortToggle[f.name] = dir;
23358         this.sortInfo = {field: f.name, direction: dir};
23359         if(!this.remoteSort){
23360             this.applySort();
23361             this.fireEvent("datachanged", this);
23362         }else{
23363             this.load(this.lastOptions);
23364         }
23365     },
23366
23367     /**
23368      * Calls the specified function for each of the Records in the cache.
23369      * @param {Function} fn The function to call. The Record is passed as the first parameter.
23370      * Returning <em>false</em> aborts and exits the iteration.
23371      * @param {Object} scope (optional) The scope in which to call the function (defaults to the Record).
23372      */
23373     each : function(fn, scope){
23374         this.data.each(fn, scope);
23375     },
23376
23377     /**
23378      * Gets all records modified since the last commit.  Modified records are persisted across load operations
23379      * (e.g., during paging).
23380      * @return {Roo.data.Record[]} An array of Records containing outstanding modifications.
23381      */
23382     getModifiedRecords : function(){
23383         return this.modified;
23384     },
23385
23386     // private
23387     createFilterFn : function(property, value, anyMatch){
23388         if(!value.exec){ // not a regex
23389             value = String(value);
23390             if(value.length == 0){
23391                 return false;
23392             }
23393             value = new RegExp((anyMatch === true ? '' : '^') + Roo.escapeRe(value), "i");
23394         }
23395         return function(r){
23396             return value.test(r.data[property]);
23397         };
23398     },
23399
23400     /**
23401      * Sums the value of <i>property</i> for each record between start and end and returns the result.
23402      * @param {String} property A field on your records
23403      * @param {Number} start The record index to start at (defaults to 0)
23404      * @param {Number} end The last record index to include (defaults to length - 1)
23405      * @return {Number} The sum
23406      */
23407     sum : function(property, start, end){
23408         var rs = this.data.items, v = 0;
23409         start = start || 0;
23410         end = (end || end === 0) ? end : rs.length-1;
23411
23412         for(var i = start; i <= end; i++){
23413             v += (rs[i].data[property] || 0);
23414         }
23415         return v;
23416     },
23417
23418     /**
23419      * Filter the records by a specified property.
23420      * @param {String} field A field on your records
23421      * @param {String/RegExp} value Either a string that the field
23422      * should start with or a RegExp to test against the field
23423      * @param {Boolean} anyMatch True to match any part not just the beginning
23424      */
23425     filter : function(property, value, anyMatch){
23426         var fn = this.createFilterFn(property, value, anyMatch);
23427         return fn ? this.filterBy(fn) : this.clearFilter();
23428     },
23429
23430     /**
23431      * Filter by a function. The specified function will be called with each
23432      * record in this data source. If the function returns true the record is included,
23433      * otherwise it is filtered.
23434      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23435      * @param {Object} scope (optional) The scope of the function (defaults to this)
23436      */
23437     filterBy : function(fn, scope){
23438         this.snapshot = this.snapshot || this.data;
23439         this.data = this.queryBy(fn, scope||this);
23440         this.fireEvent("datachanged", this);
23441     },
23442
23443     /**
23444      * Query the records by a specified property.
23445      * @param {String} field A field on your records
23446      * @param {String/RegExp} value Either a string that the field
23447      * should start with or a RegExp to test against the field
23448      * @param {Boolean} anyMatch True to match any part not just the beginning
23449      * @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23450      */
23451     query : function(property, value, anyMatch){
23452         var fn = this.createFilterFn(property, value, anyMatch);
23453         return fn ? this.queryBy(fn) : this.data.clone();
23454     },
23455
23456     /**
23457      * Query by a function. The specified function will be called with each
23458      * record in this data source. If the function returns true the record is included
23459      * in the results.
23460      * @param {Function} fn The function to be called, it will receive 2 args (record, id)
23461      * @param {Object} scope (optional) The scope of the function (defaults to this)
23462       @return {MixedCollection} Returns an Roo.util.MixedCollection of the matched records
23463      **/
23464     queryBy : function(fn, scope){
23465         var data = this.snapshot || this.data;
23466         return data.filterBy(fn, scope||this);
23467     },
23468
23469     /**
23470      * Collects unique values for a particular dataIndex from this store.
23471      * @param {String} dataIndex The property to collect
23472      * @param {Boolean} allowNull (optional) Pass true to allow null, undefined or empty string values
23473      * @param {Boolean} bypassFilter (optional) Pass true to collect from all records, even ones which are filtered
23474      * @return {Array} An array of the unique values
23475      **/
23476     collect : function(dataIndex, allowNull, bypassFilter){
23477         var d = (bypassFilter === true && this.snapshot) ?
23478                 this.snapshot.items : this.data.items;
23479         var v, sv, r = [], l = {};
23480         for(var i = 0, len = d.length; i < len; i++){
23481             v = d[i].data[dataIndex];
23482             sv = String(v);
23483             if((allowNull || !Roo.isEmpty(v)) && !l[sv]){
23484                 l[sv] = true;
23485                 r[r.length] = v;
23486             }
23487         }
23488         return r;
23489     },
23490
23491     /**
23492      * Revert to a view of the Record cache with no filtering applied.
23493      * @param {Boolean} suppressEvent If true the filter is cleared silently without notifying listeners
23494      */
23495     clearFilter : function(suppressEvent){
23496         if(this.snapshot && this.snapshot != this.data){
23497             this.data = this.snapshot;
23498             delete this.snapshot;
23499             if(suppressEvent !== true){
23500                 this.fireEvent("datachanged", this);
23501             }
23502         }
23503     },
23504
23505     // private
23506     afterEdit : function(record){
23507         if(this.modified.indexOf(record) == -1){
23508             this.modified.push(record);
23509         }
23510         this.fireEvent("update", this, record, Roo.data.Record.EDIT);
23511     },
23512     
23513     // private
23514     afterReject : function(record){
23515         this.modified.remove(record);
23516         this.fireEvent("update", this, record, Roo.data.Record.REJECT);
23517     },
23518
23519     // private
23520     afterCommit : function(record){
23521         this.modified.remove(record);
23522         this.fireEvent("update", this, record, Roo.data.Record.COMMIT);
23523     },
23524
23525     /**
23526      * Commit all Records with outstanding changes. To handle updates for changes, subscribe to the
23527      * Store's "update" event, and perform updating when the third parameter is Roo.data.Record.COMMIT.
23528      */
23529     commitChanges : function(){
23530         var m = this.modified.slice(0);
23531         this.modified = [];
23532         for(var i = 0, len = m.length; i < len; i++){
23533             m[i].commit();
23534         }
23535     },
23536
23537     /**
23538      * Cancel outstanding changes on all changed records.
23539      */
23540     rejectChanges : function(){
23541         var m = this.modified.slice(0);
23542         this.modified = [];
23543         for(var i = 0, len = m.length; i < len; i++){
23544             m[i].reject();
23545         }
23546     },
23547
23548     onMetaChange : function(meta, rtype, o){
23549         this.recordType = rtype;
23550         this.fields = rtype.prototype.fields;
23551         delete this.snapshot;
23552         this.sortInfo = meta.sortInfo || this.sortInfo;
23553         this.modified = [];
23554         this.fireEvent('metachange', this, this.reader.meta);
23555     },
23556     
23557     moveIndex : function(data, type)
23558     {
23559         var index = this.indexOf(data);
23560         
23561         var newIndex = index + type;
23562         
23563         this.remove(data);
23564         
23565         this.insert(newIndex, data);
23566         
23567     }
23568 });/*
23569  * Based on:
23570  * Ext JS Library 1.1.1
23571  * Copyright(c) 2006-2007, Ext JS, LLC.
23572  *
23573  * Originally Released Under LGPL - original licence link has changed is not relivant.
23574  *
23575  * Fork - LGPL
23576  * <script type="text/javascript">
23577  */
23578
23579 /**
23580  * @class Roo.data.SimpleStore
23581  * @extends Roo.data.Store
23582  * Small helper class to make creating Stores from Array data easier.
23583  * @cfg {Number} id The array index of the record id. Leave blank to auto generate ids.
23584  * @cfg {Array} fields An array of field definition objects, or field name strings.
23585  * @cfg {Object} an existing reader (eg. copied from another store)
23586  * @cfg {Array} data The multi-dimensional array of data
23587  * @constructor
23588  * @param {Object} config
23589  */
23590 Roo.data.SimpleStore = function(config)
23591 {
23592     Roo.data.SimpleStore.superclass.constructor.call(this, {
23593         isLocal : true,
23594         reader: typeof(config.reader) != 'undefined' ? config.reader : new Roo.data.ArrayReader({
23595                 id: config.id
23596             },
23597             Roo.data.Record.create(config.fields)
23598         ),
23599         proxy : new Roo.data.MemoryProxy(config.data)
23600     });
23601     this.load();
23602 };
23603 Roo.extend(Roo.data.SimpleStore, Roo.data.Store);/*
23604  * Based on:
23605  * Ext JS Library 1.1.1
23606  * Copyright(c) 2006-2007, Ext JS, LLC.
23607  *
23608  * Originally Released Under LGPL - original licence link has changed is not relivant.
23609  *
23610  * Fork - LGPL
23611  * <script type="text/javascript">
23612  */
23613
23614 /**
23615 /**
23616  * @extends Roo.data.Store
23617  * @class Roo.data.JsonStore
23618  * Small helper class to make creating Stores for JSON data easier. <br/>
23619 <pre><code>
23620 var store = new Roo.data.JsonStore({
23621     url: 'get-images.php',
23622     root: 'images',
23623     fields: ['name', 'url', {name:'size', type: 'float'}, {name:'lastmod', type:'date'}]
23624 });
23625 </code></pre>
23626  * <b>Note: Although they are not listed, this class inherits all of the config options of Store,
23627  * JsonReader and HttpProxy (unless inline data is provided).</b>
23628  * @cfg {Array} fields An array of field definition objects, or field name strings.
23629  * @constructor
23630  * @param {Object} config
23631  */
23632 Roo.data.JsonStore = function(c){
23633     Roo.data.JsonStore.superclass.constructor.call(this, Roo.apply(c, {
23634         proxy: !c.data ? new Roo.data.HttpProxy({url: c.url}) : undefined,
23635         reader: new Roo.data.JsonReader(c, c.fields)
23636     }));
23637 };
23638 Roo.extend(Roo.data.JsonStore, Roo.data.Store);/*
23639  * Based on:
23640  * Ext JS Library 1.1.1
23641  * Copyright(c) 2006-2007, Ext JS, LLC.
23642  *
23643  * Originally Released Under LGPL - original licence link has changed is not relivant.
23644  *
23645  * Fork - LGPL
23646  * <script type="text/javascript">
23647  */
23648
23649  
23650 Roo.data.Field = function(config){
23651     if(typeof config == "string"){
23652         config = {name: config};
23653     }
23654     Roo.apply(this, config);
23655     
23656     if(!this.type){
23657         this.type = "auto";
23658     }
23659     
23660     var st = Roo.data.SortTypes;
23661     // named sortTypes are supported, here we look them up
23662     if(typeof this.sortType == "string"){
23663         this.sortType = st[this.sortType];
23664     }
23665     
23666     // set default sortType for strings and dates
23667     if(!this.sortType){
23668         switch(this.type){
23669             case "string":
23670                 this.sortType = st.asUCString;
23671                 break;
23672             case "date":
23673                 this.sortType = st.asDate;
23674                 break;
23675             default:
23676                 this.sortType = st.none;
23677         }
23678     }
23679
23680     // define once
23681     var stripRe = /[\$,%]/g;
23682
23683     // prebuilt conversion function for this field, instead of
23684     // switching every time we're reading a value
23685     if(!this.convert){
23686         var cv, dateFormat = this.dateFormat;
23687         switch(this.type){
23688             case "":
23689             case "auto":
23690             case undefined:
23691                 cv = function(v){ return v; };
23692                 break;
23693             case "string":
23694                 cv = function(v){ return (v === undefined || v === null) ? '' : String(v); };
23695                 break;
23696             case "int":
23697                 cv = function(v){
23698                     return v !== undefined && v !== null && v !== '' ?
23699                            parseInt(String(v).replace(stripRe, ""), 10) : '';
23700                     };
23701                 break;
23702             case "float":
23703                 cv = function(v){
23704                     return v !== undefined && v !== null && v !== '' ?
23705                            parseFloat(String(v).replace(stripRe, ""), 10) : ''; 
23706                     };
23707                 break;
23708             case "bool":
23709             case "boolean":
23710                 cv = function(v){ return v === true || v === "true" || v == 1; };
23711                 break;
23712             case "date":
23713                 cv = function(v){
23714                     if(!v){
23715                         return '';
23716                     }
23717                     if(v instanceof Date){
23718                         return v;
23719                     }
23720                     if(dateFormat){
23721                         if(dateFormat == "timestamp"){
23722                             return new Date(v*1000);
23723                         }
23724                         return Date.parseDate(v, dateFormat);
23725                     }
23726                     var parsed = Date.parse(v);
23727                     return parsed ? new Date(parsed) : null;
23728                 };
23729              break;
23730             
23731         }
23732         this.convert = cv;
23733     }
23734 };
23735
23736 Roo.data.Field.prototype = {
23737     dateFormat: null,
23738     defaultValue: "",
23739     mapping: null,
23740     sortType : null,
23741     sortDir : "ASC"
23742 };/*
23743  * Based on:
23744  * Ext JS Library 1.1.1
23745  * Copyright(c) 2006-2007, Ext JS, LLC.
23746  *
23747  * Originally Released Under LGPL - original licence link has changed is not relivant.
23748  *
23749  * Fork - LGPL
23750  * <script type="text/javascript">
23751  */
23752  
23753 // Base class for reading structured data from a data source.  This class is intended to be
23754 // extended (see ArrayReader, JsonReader and XmlReader) and should not be created directly.
23755
23756 /**
23757  * @class Roo.data.DataReader
23758  * Base class for reading structured data from a data source.  This class is intended to be
23759  * extended (see {Roo.data.ArrayReader}, {Roo.data.JsonReader} and {Roo.data.XmlReader}) and should not be created directly.
23760  */
23761
23762 Roo.data.DataReader = function(meta, recordType){
23763     
23764     this.meta = meta;
23765     
23766     this.recordType = recordType instanceof Array ? 
23767         Roo.data.Record.create(recordType) : recordType;
23768 };
23769
23770 Roo.data.DataReader.prototype = {
23771     
23772     
23773     readerType : 'Data',
23774      /**
23775      * Create an empty record
23776      * @param {Object} data (optional) - overlay some values
23777      * @return {Roo.data.Record} record created.
23778      */
23779     newRow :  function(d) {
23780         var da =  {};
23781         this.recordType.prototype.fields.each(function(c) {
23782             switch( c.type) {
23783                 case 'int' : da[c.name] = 0; break;
23784                 case 'date' : da[c.name] = new Date(); break;
23785                 case 'float' : da[c.name] = 0.0; break;
23786                 case 'boolean' : da[c.name] = false; break;
23787                 default : da[c.name] = ""; break;
23788             }
23789             
23790         });
23791         return new this.recordType(Roo.apply(da, d));
23792     }
23793     
23794     
23795 };/*
23796  * Based on:
23797  * Ext JS Library 1.1.1
23798  * Copyright(c) 2006-2007, Ext JS, LLC.
23799  *
23800  * Originally Released Under LGPL - original licence link has changed is not relivant.
23801  *
23802  * Fork - LGPL
23803  * <script type="text/javascript">
23804  */
23805
23806 /**
23807  * @class Roo.data.DataProxy
23808  * @extends Roo.data.Observable
23809  * This class is an abstract base class for implementations which provide retrieval of
23810  * unformatted data objects.<br>
23811  * <p>
23812  * DataProxy implementations are usually used in conjunction with an implementation of Roo.data.DataReader
23813  * (of the appropriate type which knows how to parse the data object) to provide a block of
23814  * {@link Roo.data.Records} to an {@link Roo.data.Store}.<br>
23815  * <p>
23816  * Custom implementations must implement the load method as described in
23817  * {@link Roo.data.HttpProxy#load}.
23818  */
23819 Roo.data.DataProxy = function(){
23820     this.addEvents({
23821         /**
23822          * @event beforeload
23823          * Fires before a network request is made to retrieve a data object.
23824          * @param {Object} This DataProxy object.
23825          * @param {Object} params The params parameter to the load function.
23826          */
23827         beforeload : true,
23828         /**
23829          * @event load
23830          * Fires before the load method's callback is called.
23831          * @param {Object} This DataProxy object.
23832          * @param {Object} o The data object.
23833          * @param {Object} arg The callback argument object passed to the load function.
23834          */
23835         load : true,
23836         /**
23837          * @event loadexception
23838          * Fires if an Exception occurs during data retrieval.
23839          * @param {Object} This DataProxy object.
23840          * @param {Object} o The data object.
23841          * @param {Object} arg The callback argument object passed to the load function.
23842          * @param {Object} e The Exception.
23843          */
23844         loadexception : true
23845     });
23846     Roo.data.DataProxy.superclass.constructor.call(this);
23847 };
23848
23849 Roo.extend(Roo.data.DataProxy, Roo.util.Observable);
23850
23851     /**
23852      * @cfg {void} listeners (Not available) Constructor blocks listeners from being set
23853      */
23854 /*
23855  * Based on:
23856  * Ext JS Library 1.1.1
23857  * Copyright(c) 2006-2007, Ext JS, LLC.
23858  *
23859  * Originally Released Under LGPL - original licence link has changed is not relivant.
23860  *
23861  * Fork - LGPL
23862  * <script type="text/javascript">
23863  */
23864 /**
23865  * @class Roo.data.MemoryProxy
23866  * An implementation of Roo.data.DataProxy that simply passes the data specified in its constructor
23867  * to the Reader when its load method is called.
23868  * @constructor
23869  * @param {Object} data The data object which the Reader uses to construct a block of Roo.data.Records.
23870  */
23871 Roo.data.MemoryProxy = function(data){
23872     if (data.data) {
23873         data = data.data;
23874     }
23875     Roo.data.MemoryProxy.superclass.constructor.call(this);
23876     this.data = data;
23877 };
23878
23879 Roo.extend(Roo.data.MemoryProxy, Roo.data.DataProxy, {
23880     
23881     /**
23882      * Load data from the requested source (in this case an in-memory
23883      * data object passed to the constructor), read the data object into
23884      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
23885      * process that block using the passed callback.
23886      * @param {Object} params This parameter is not used by the MemoryProxy class.
23887      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23888      * object into a block of Roo.data.Records.
23889      * @param {Function} callback The function into which to pass the block of Roo.data.records.
23890      * The function must be passed <ul>
23891      * <li>The Record block object</li>
23892      * <li>The "arg" argument from the load function</li>
23893      * <li>A boolean success indicator</li>
23894      * </ul>
23895      * @param {Object} scope The scope in which to call the callback
23896      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
23897      */
23898     load : function(params, reader, callback, scope, arg){
23899         params = params || {};
23900         var result;
23901         try {
23902             result = reader.readRecords(params.data ? params.data :this.data);
23903         }catch(e){
23904             this.fireEvent("loadexception", this, arg, null, e);
23905             callback.call(scope, null, arg, false);
23906             return;
23907         }
23908         callback.call(scope, result, arg, true);
23909     },
23910     
23911     // private
23912     update : function(params, records){
23913         
23914     }
23915 });/*
23916  * Based on:
23917  * Ext JS Library 1.1.1
23918  * Copyright(c) 2006-2007, Ext JS, LLC.
23919  *
23920  * Originally Released Under LGPL - original licence link has changed is not relivant.
23921  *
23922  * Fork - LGPL
23923  * <script type="text/javascript">
23924  */
23925 /**
23926  * @class Roo.data.HttpProxy
23927  * @extends Roo.data.DataProxy
23928  * An implementation of {@link Roo.data.DataProxy} that reads a data object from an {@link Roo.data.Connection} object
23929  * configured to reference a certain URL.<br><br>
23930  * <p>
23931  * <em>Note that this class cannot be used to retrieve data from a domain other than the domain
23932  * from which the running page was served.<br><br>
23933  * <p>
23934  * For cross-domain access to remote data, use an {@link Roo.data.ScriptTagProxy}.</em><br><br>
23935  * <p>
23936  * Be aware that to enable the browser to parse an XML document, the server must set
23937  * the Content-Type header in the HTTP response to "text/xml".
23938  * @constructor
23939  * @param {Object} conn Connection config options to add to each request (e.g. {url: 'foo.php'} or
23940  * an {@link Roo.data.Connection} object.  If a Connection config is passed, the singleton {@link Roo.Ajax} object
23941  * will be used to make the request.
23942  */
23943 Roo.data.HttpProxy = function(conn){
23944     Roo.data.HttpProxy.superclass.constructor.call(this);
23945     // is conn a conn config or a real conn?
23946     this.conn = conn;
23947     this.useAjax = !conn || !conn.events;
23948   
23949 };
23950
23951 Roo.extend(Roo.data.HttpProxy, Roo.data.DataProxy, {
23952     // thse are take from connection...
23953     
23954     /**
23955      * @cfg {String} url (Optional) The default URL to be used for requests to the server. (defaults to undefined)
23956      */
23957     /**
23958      * @cfg {Object} extraParams (Optional) An object containing properties which are used as
23959      * extra parameters to each request made by this object. (defaults to undefined)
23960      */
23961     /**
23962      * @cfg {Object} defaultHeaders (Optional) An object containing request headers which are added
23963      *  to each request made by this object. (defaults to undefined)
23964      */
23965     /**
23966      * @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)
23967      */
23968     /**
23969      * @cfg {Number} timeout (Optional) The timeout in milliseconds to be used for requests. (defaults to 30000)
23970      */
23971      /**
23972      * @cfg {Boolean} autoAbort (Optional) Whether this request should abort any pending requests. (defaults to false)
23973      * @type Boolean
23974      */
23975   
23976
23977     /**
23978      * @cfg {Boolean} disableCaching (Optional) True to add a unique cache-buster param to GET requests. (defaults to true)
23979      * @type Boolean
23980      */
23981     /**
23982      * Return the {@link Roo.data.Connection} object being used by this Proxy.
23983      * @return {Connection} The Connection object. This object may be used to subscribe to events on
23984      * a finer-grained basis than the DataProxy events.
23985      */
23986     getConnection : function(){
23987         return this.useAjax ? Roo.Ajax : this.conn;
23988     },
23989
23990     /**
23991      * Load data from the configured {@link Roo.data.Connection}, read the data object into
23992      * a block of Roo.data.Records using the passed {@link Roo.data.DataReader} implementation, and
23993      * process that block using the passed callback.
23994      * @param {Object} params An object containing properties which are to be used as HTTP parameters
23995      * for the request to the remote server.
23996      * @param {Roo.data.DataReader} reader The Reader object which converts the data
23997      * object into a block of Roo.data.Records.
23998      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
23999      * The function must be passed <ul>
24000      * <li>The Record block object</li>
24001      * <li>The "arg" argument from the load function</li>
24002      * <li>A boolean success indicator</li>
24003      * </ul>
24004      * @param {Object} scope The scope in which to call the callback
24005      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24006      */
24007     load : function(params, reader, callback, scope, arg){
24008         if(this.fireEvent("beforeload", this, params) !== false){
24009             var  o = {
24010                 params : params || {},
24011                 request: {
24012                     callback : callback,
24013                     scope : scope,
24014                     arg : arg
24015                 },
24016                 reader: reader,
24017                 callback : this.loadResponse,
24018                 scope: this
24019             };
24020             if(this.useAjax){
24021                 Roo.applyIf(o, this.conn);
24022                 if(this.activeRequest){
24023                     Roo.Ajax.abort(this.activeRequest);
24024                 }
24025                 this.activeRequest = Roo.Ajax.request(o);
24026             }else{
24027                 this.conn.request(o);
24028             }
24029         }else{
24030             callback.call(scope||this, null, arg, false);
24031         }
24032     },
24033
24034     // private
24035     loadResponse : function(o, success, response){
24036         delete this.activeRequest;
24037         if(!success){
24038             this.fireEvent("loadexception", this, o, response);
24039             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24040             return;
24041         }
24042         var result;
24043         try {
24044             result = o.reader.read(response);
24045         }catch(e){
24046             this.fireEvent("loadexception", this, o, response, e);
24047             o.request.callback.call(o.request.scope, null, o.request.arg, false);
24048             return;
24049         }
24050         
24051         this.fireEvent("load", this, o, o.request.arg);
24052         o.request.callback.call(o.request.scope, result, o.request.arg, true);
24053     },
24054
24055     // private
24056     update : function(dataSet){
24057
24058     },
24059
24060     // private
24061     updateResponse : function(dataSet){
24062
24063     }
24064 });/*
24065  * Based on:
24066  * Ext JS Library 1.1.1
24067  * Copyright(c) 2006-2007, Ext JS, LLC.
24068  *
24069  * Originally Released Under LGPL - original licence link has changed is not relivant.
24070  *
24071  * Fork - LGPL
24072  * <script type="text/javascript">
24073  */
24074
24075 /**
24076  * @class Roo.data.ScriptTagProxy
24077  * An implementation of Roo.data.DataProxy that reads a data object from a URL which may be in a domain
24078  * other than the originating domain of the running page.<br><br>
24079  * <p>
24080  * <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
24081  * of the running page, you must use this class, rather than DataProxy.</em><br><br>
24082  * <p>
24083  * The content passed back from a server resource requested by a ScriptTagProxy is executable JavaScript
24084  * source code that is used as the source inside a &lt;script> tag.<br><br>
24085  * <p>
24086  * In order for the browser to process the returned data, the server must wrap the data object
24087  * with a call to a callback function, the name of which is passed as a parameter by the ScriptTagProxy.
24088  * Below is a Java example for a servlet which returns data for either a ScriptTagProxy, or an HttpProxy
24089  * depending on whether the callback name was passed:
24090  * <p>
24091  * <pre><code>
24092 boolean scriptTag = false;
24093 String cb = request.getParameter("callback");
24094 if (cb != null) {
24095     scriptTag = true;
24096     response.setContentType("text/javascript");
24097 } else {
24098     response.setContentType("application/x-json");
24099 }
24100 Writer out = response.getWriter();
24101 if (scriptTag) {
24102     out.write(cb + "(");
24103 }
24104 out.print(dataBlock.toJsonString());
24105 if (scriptTag) {
24106     out.write(");");
24107 }
24108 </pre></code>
24109  *
24110  * @constructor
24111  * @param {Object} config A configuration object.
24112  */
24113 Roo.data.ScriptTagProxy = function(config){
24114     Roo.data.ScriptTagProxy.superclass.constructor.call(this);
24115     Roo.apply(this, config);
24116     this.head = document.getElementsByTagName("head")[0];
24117 };
24118
24119 Roo.data.ScriptTagProxy.TRANS_ID = 1000;
24120
24121 Roo.extend(Roo.data.ScriptTagProxy, Roo.data.DataProxy, {
24122     /**
24123      * @cfg {String} url The URL from which to request the data object.
24124      */
24125     /**
24126      * @cfg {Number} timeout (Optional) The number of milliseconds to wait for a response. Defaults to 30 seconds.
24127      */
24128     timeout : 30000,
24129     /**
24130      * @cfg {String} callbackParam (Optional) The name of the parameter to pass to the server which tells
24131      * the server the name of the callback function set up by the load call to process the returned data object.
24132      * Defaults to "callback".<p>The server-side processing must read this parameter value, and generate
24133      * javascript output which calls this named function passing the data object as its only parameter.
24134      */
24135     callbackParam : "callback",
24136     /**
24137      *  @cfg {Boolean} nocache (Optional) Defaults to true. Disable cacheing by adding a unique parameter
24138      * name to the request.
24139      */
24140     nocache : true,
24141
24142     /**
24143      * Load data from the configured URL, read the data object into
24144      * a block of Roo.data.Records using the passed Roo.data.DataReader implementation, and
24145      * process that block using the passed callback.
24146      * @param {Object} params An object containing properties which are to be used as HTTP parameters
24147      * for the request to the remote server.
24148      * @param {Roo.data.DataReader} reader The Reader object which converts the data
24149      * object into a block of Roo.data.Records.
24150      * @param {Function} callback The function into which to pass the block of Roo.data.Records.
24151      * The function must be passed <ul>
24152      * <li>The Record block object</li>
24153      * <li>The "arg" argument from the load function</li>
24154      * <li>A boolean success indicator</li>
24155      * </ul>
24156      * @param {Object} scope The scope in which to call the callback
24157      * @param {Object} arg An optional argument which is passed to the callback as its second parameter.
24158      */
24159     load : function(params, reader, callback, scope, arg){
24160         if(this.fireEvent("beforeload", this, params) !== false){
24161
24162             var p = Roo.urlEncode(Roo.apply(params, this.extraParams));
24163
24164             var url = this.url;
24165             url += (url.indexOf("?") != -1 ? "&" : "?") + p;
24166             if(this.nocache){
24167                 url += "&_dc=" + (new Date().getTime());
24168             }
24169             var transId = ++Roo.data.ScriptTagProxy.TRANS_ID;
24170             var trans = {
24171                 id : transId,
24172                 cb : "stcCallback"+transId,
24173                 scriptId : "stcScript"+transId,
24174                 params : params,
24175                 arg : arg,
24176                 url : url,
24177                 callback : callback,
24178                 scope : scope,
24179                 reader : reader
24180             };
24181             var conn = this;
24182
24183             window[trans.cb] = function(o){
24184                 conn.handleResponse(o, trans);
24185             };
24186
24187             url += String.format("&{0}={1}", this.callbackParam, trans.cb);
24188
24189             if(this.autoAbort !== false){
24190                 this.abort();
24191             }
24192
24193             trans.timeoutId = this.handleFailure.defer(this.timeout, this, [trans]);
24194
24195             var script = document.createElement("script");
24196             script.setAttribute("src", url);
24197             script.setAttribute("type", "text/javascript");
24198             script.setAttribute("id", trans.scriptId);
24199             this.head.appendChild(script);
24200
24201             this.trans = trans;
24202         }else{
24203             callback.call(scope||this, null, arg, false);
24204         }
24205     },
24206
24207     // private
24208     isLoading : function(){
24209         return this.trans ? true : false;
24210     },
24211
24212     /**
24213      * Abort the current server request.
24214      */
24215     abort : function(){
24216         if(this.isLoading()){
24217             this.destroyTrans(this.trans);
24218         }
24219     },
24220
24221     // private
24222     destroyTrans : function(trans, isLoaded){
24223         this.head.removeChild(document.getElementById(trans.scriptId));
24224         clearTimeout(trans.timeoutId);
24225         if(isLoaded){
24226             window[trans.cb] = undefined;
24227             try{
24228                 delete window[trans.cb];
24229             }catch(e){}
24230         }else{
24231             // if hasn't been loaded, wait for load to remove it to prevent script error
24232             window[trans.cb] = function(){
24233                 window[trans.cb] = undefined;
24234                 try{
24235                     delete window[trans.cb];
24236                 }catch(e){}
24237             };
24238         }
24239     },
24240
24241     // private
24242     handleResponse : function(o, trans){
24243         this.trans = false;
24244         this.destroyTrans(trans, true);
24245         var result;
24246         try {
24247             result = trans.reader.readRecords(o);
24248         }catch(e){
24249             this.fireEvent("loadexception", this, o, trans.arg, e);
24250             trans.callback.call(trans.scope||window, null, trans.arg, false);
24251             return;
24252         }
24253         this.fireEvent("load", this, o, trans.arg);
24254         trans.callback.call(trans.scope||window, result, trans.arg, true);
24255     },
24256
24257     // private
24258     handleFailure : function(trans){
24259         this.trans = false;
24260         this.destroyTrans(trans, false);
24261         this.fireEvent("loadexception", this, null, trans.arg);
24262         trans.callback.call(trans.scope||window, null, trans.arg, false);
24263     }
24264 });/*
24265  * Based on:
24266  * Ext JS Library 1.1.1
24267  * Copyright(c) 2006-2007, Ext JS, LLC.
24268  *
24269  * Originally Released Under LGPL - original licence link has changed is not relivant.
24270  *
24271  * Fork - LGPL
24272  * <script type="text/javascript">
24273  */
24274
24275 /**
24276  * @class Roo.data.JsonReader
24277  * @extends Roo.data.DataReader
24278  * Data reader class to create an Array of Roo.data.Record objects from a JSON response
24279  * based on mappings in a provided Roo.data.Record constructor.
24280  * 
24281  * The default behaviour of a store is to send ?_requestMeta=1, unless the class has recieved 'metaData' property
24282  * in the reply previously. 
24283  * 
24284  * <p>
24285  * Example code:
24286  * <pre><code>
24287 var RecordDef = Roo.data.Record.create([
24288     {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24289     {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24290 ]);
24291 var myReader = new Roo.data.JsonReader({
24292     totalProperty: "results",    // The property which contains the total dataset size (optional)
24293     root: "rows",                // The property which contains an Array of row objects
24294     id: "id"                     // The property within each row object that provides an ID for the record (optional)
24295 }, RecordDef);
24296 </code></pre>
24297  * <p>
24298  * This would consume a JSON file like this:
24299  * <pre><code>
24300 { 'results': 2, 'rows': [
24301     { 'id': 1, 'name': 'Bill', occupation: 'Gardener' },
24302     { 'id': 2, 'name': 'Ben', occupation: 'Horticulturalist' } ]
24303 }
24304 </code></pre>
24305  * @cfg {String} totalProperty Name of the property from which to retrieve the total number of records
24306  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24307  * paged from the remote server.
24308  * @cfg {String} successProperty Name of the property from which to retrieve the success attribute used by forms.
24309  * @cfg {String} root name of the property which contains the Array of row objects.
24310  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24311  * @cfg {Array} fields Array of field definition objects
24312  * @constructor
24313  * Create a new JsonReader
24314  * @param {Object} meta Metadata configuration options
24315  * @param {Object} recordType Either an Array of field definition objects,
24316  * or an {@link Roo.data.Record} object created using {@link Roo.data.Record#create}.
24317  */
24318 Roo.data.JsonReader = function(meta, recordType){
24319     
24320     meta = meta || {};
24321     // set some defaults:
24322     Roo.applyIf(meta, {
24323         totalProperty: 'total',
24324         successProperty : 'success',
24325         root : 'data',
24326         id : 'id'
24327     });
24328     
24329     Roo.data.JsonReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24330 };
24331 Roo.extend(Roo.data.JsonReader, Roo.data.DataReader, {
24332     
24333     readerType : 'Json',
24334     
24335     /**
24336      * @prop {Boolean} metaFromRemote  - if the meta data was loaded from the remote source.
24337      * Used by Store query builder to append _requestMeta to params.
24338      * 
24339      */
24340     metaFromRemote : false,
24341     /**
24342      * This method is only used by a DataProxy which has retrieved data from a remote server.
24343      * @param {Object} response The XHR object which contains the JSON data in its responseText.
24344      * @return {Object} data A data block which is used by an Roo.data.Store object as
24345      * a cache of Roo.data.Records.
24346      */
24347     read : function(response){
24348         var json = response.responseText;
24349        
24350         var o = /* eval:var:o */ eval("("+json+")");
24351         if(!o) {
24352             throw {message: "JsonReader.read: Json object not found"};
24353         }
24354         
24355         if(o.metaData){
24356             
24357             delete this.ef;
24358             this.metaFromRemote = true;
24359             this.meta = o.metaData;
24360             this.recordType = Roo.data.Record.create(o.metaData.fields);
24361             this.onMetaChange(this.meta, this.recordType, o);
24362         }
24363         return this.readRecords(o);
24364     },
24365
24366     // private function a store will implement
24367     onMetaChange : function(meta, recordType, o){
24368
24369     },
24370
24371     /**
24372          * @ignore
24373          */
24374     simpleAccess: function(obj, subsc) {
24375         return obj[subsc];
24376     },
24377
24378         /**
24379          * @ignore
24380          */
24381     getJsonAccessor: function(){
24382         var re = /[\[\.]/;
24383         return function(expr) {
24384             try {
24385                 return(re.test(expr))
24386                     ? new Function("obj", "return obj." + expr)
24387                     : function(obj){
24388                         return obj[expr];
24389                     };
24390             } catch(e){}
24391             return Roo.emptyFn;
24392         };
24393     }(),
24394
24395     /**
24396      * Create a data block containing Roo.data.Records from an XML document.
24397      * @param {Object} o An object which contains an Array of row objects in the property specified
24398      * in the config as 'root, and optionally a property, specified in the config as 'totalProperty'
24399      * which contains the total size of the dataset.
24400      * @return {Object} data A data block which is used by an Roo.data.Store object as
24401      * a cache of Roo.data.Records.
24402      */
24403     readRecords : function(o){
24404         /**
24405          * After any data loads, the raw JSON data is available for further custom processing.
24406          * @type Object
24407          */
24408         this.o = o;
24409         var s = this.meta, Record = this.recordType,
24410             f = Record ? Record.prototype.fields : null, fi = f ? f.items : [], fl = f ? f.length : 0;
24411
24412 //      Generate extraction functions for the totalProperty, the root, the id, and for each field
24413         if (!this.ef) {
24414             if(s.totalProperty) {
24415                     this.getTotal = this.getJsonAccessor(s.totalProperty);
24416                 }
24417                 if(s.successProperty) {
24418                     this.getSuccess = this.getJsonAccessor(s.successProperty);
24419                 }
24420                 this.getRoot = s.root ? this.getJsonAccessor(s.root) : function(p){return p;};
24421                 if (s.id) {
24422                         var g = this.getJsonAccessor(s.id);
24423                         this.getId = function(rec) {
24424                                 var r = g(rec);  
24425                                 return (r === undefined || r === "") ? null : r;
24426                         };
24427                 } else {
24428                         this.getId = function(){return null;};
24429                 }
24430             this.ef = [];
24431             for(var jj = 0; jj < fl; jj++){
24432                 f = fi[jj];
24433                 var map = (f.mapping !== undefined && f.mapping !== null) ? f.mapping : f.name;
24434                 this.ef[jj] = this.getJsonAccessor(map);
24435             }
24436         }
24437
24438         var root = this.getRoot(o), c = root.length, totalRecords = c, success = true;
24439         if(s.totalProperty){
24440             var vt = parseInt(this.getTotal(o), 10);
24441             if(!isNaN(vt)){
24442                 totalRecords = vt;
24443             }
24444         }
24445         if(s.successProperty){
24446             var vs = this.getSuccess(o);
24447             if(vs === false || vs === 'false'){
24448                 success = false;
24449             }
24450         }
24451         var records = [];
24452         for(var i = 0; i < c; i++){
24453                 var n = root[i];
24454             var values = {};
24455             var id = this.getId(n);
24456             for(var j = 0; j < fl; j++){
24457                 f = fi[j];
24458             var v = this.ef[j](n);
24459             if (!f.convert) {
24460                 Roo.log('missing convert for ' + f.name);
24461                 Roo.log(f);
24462                 continue;
24463             }
24464             values[f.name] = f.convert((v !== undefined) ? v : f.defaultValue);
24465             }
24466             var record = new Record(values, id);
24467             record.json = n;
24468             records[i] = record;
24469         }
24470         return {
24471             raw : o,
24472             success : success,
24473             records : records,
24474             totalRecords : totalRecords
24475         };
24476     },
24477     // used when loading children.. @see loadDataFromChildren
24478     toLoadData: function(rec)
24479     {
24480         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
24481         var data = typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
24482         return { data : data, total : data.length };
24483         
24484     }
24485 });/*
24486  * Based on:
24487  * Ext JS Library 1.1.1
24488  * Copyright(c) 2006-2007, Ext JS, LLC.
24489  *
24490  * Originally Released Under LGPL - original licence link has changed is not relivant.
24491  *
24492  * Fork - LGPL
24493  * <script type="text/javascript">
24494  */
24495
24496 /**
24497  * @class Roo.data.XmlReader
24498  * @extends Roo.data.DataReader
24499  * Data reader class to create an Array of {@link Roo.data.Record} objects from an XML document
24500  * based on mappings in a provided Roo.data.Record constructor.<br><br>
24501  * <p>
24502  * <em>Note that in order for the browser to parse a returned XML document, the Content-Type
24503  * header in the HTTP response must be set to "text/xml".</em>
24504  * <p>
24505  * Example code:
24506  * <pre><code>
24507 var RecordDef = Roo.data.Record.create([
24508    {name: 'name', mapping: 'name'},     // "mapping" property not needed if it's the same as "name"
24509    {name: 'occupation'}                 // This field will use "occupation" as the mapping.
24510 ]);
24511 var myReader = new Roo.data.XmlReader({
24512    totalRecords: "results", // The element which contains the total dataset size (optional)
24513    record: "row",           // The repeated element which contains row information
24514    id: "id"                 // The element within the row that provides an ID for the record (optional)
24515 }, RecordDef);
24516 </code></pre>
24517  * <p>
24518  * This would consume an XML file like this:
24519  * <pre><code>
24520 &lt;?xml?>
24521 &lt;dataset>
24522  &lt;results>2&lt;/results>
24523  &lt;row>
24524    &lt;id>1&lt;/id>
24525    &lt;name>Bill&lt;/name>
24526    &lt;occupation>Gardener&lt;/occupation>
24527  &lt;/row>
24528  &lt;row>
24529    &lt;id>2&lt;/id>
24530    &lt;name>Ben&lt;/name>
24531    &lt;occupation>Horticulturalist&lt;/occupation>
24532  &lt;/row>
24533 &lt;/dataset>
24534 </code></pre>
24535  * @cfg {String} totalRecords The DomQuery path from which to retrieve the total number of records
24536  * in the dataset. This is only needed if the whole dataset is not passed in one go, but is being
24537  * paged from the remote server.
24538  * @cfg {String} record The DomQuery path to the repeated element which contains record information.
24539  * @cfg {String} success The DomQuery path to the success attribute used by forms.
24540  * @cfg {String} id The DomQuery path relative from the record element to the element that contains
24541  * a record identifier value.
24542  * @constructor
24543  * Create a new XmlReader
24544  * @param {Object} meta Metadata configuration options
24545  * @param {Mixed} recordType The definition of the data record type to produce.  This can be either a valid
24546  * Record subclass created with {@link Roo.data.Record#create}, or an array of objects with which to call
24547  * Roo.data.Record.create.  See the {@link Roo.data.Record} class for more details.
24548  */
24549 Roo.data.XmlReader = function(meta, recordType){
24550     meta = meta || {};
24551     Roo.data.XmlReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24552 };
24553 Roo.extend(Roo.data.XmlReader, Roo.data.DataReader, {
24554     
24555     readerType : 'Xml',
24556     
24557     /**
24558      * This method is only used by a DataProxy which has retrieved data from a remote server.
24559          * @param {Object} response The XHR object which contains the parsed XML document.  The response is expected
24560          * to contain a method called 'responseXML' that returns an XML document object.
24561      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24562      * a cache of Roo.data.Records.
24563      */
24564     read : function(response){
24565         var doc = response.responseXML;
24566         if(!doc) {
24567             throw {message: "XmlReader.read: XML Document not available"};
24568         }
24569         return this.readRecords(doc);
24570     },
24571
24572     /**
24573      * Create a data block containing Roo.data.Records from an XML document.
24574          * @param {Object} doc A parsed XML document.
24575      * @return {Object} records A data block which is used by an {@link Roo.data.Store} as
24576      * a cache of Roo.data.Records.
24577      */
24578     readRecords : function(doc){
24579         /**
24580          * After any data loads/reads, the raw XML Document is available for further custom processing.
24581          * @type XMLDocument
24582          */
24583         this.xmlData = doc;
24584         var root = doc.documentElement || doc;
24585         var q = Roo.DomQuery;
24586         var recordType = this.recordType, fields = recordType.prototype.fields;
24587         var sid = this.meta.id;
24588         var totalRecords = 0, success = true;
24589         if(this.meta.totalRecords){
24590             totalRecords = q.selectNumber(this.meta.totalRecords, root, 0);
24591         }
24592         
24593         if(this.meta.success){
24594             var sv = q.selectValue(this.meta.success, root, true);
24595             success = sv !== false && sv !== 'false';
24596         }
24597         var records = [];
24598         var ns = q.select(this.meta.record, root);
24599         for(var i = 0, len = ns.length; i < len; i++) {
24600                 var n = ns[i];
24601                 var values = {};
24602                 var id = sid ? q.selectValue(sid, n) : undefined;
24603                 for(var j = 0, jlen = fields.length; j < jlen; j++){
24604                     var f = fields.items[j];
24605                 var v = q.selectValue(f.mapping || f.name, n, f.defaultValue);
24606                     v = f.convert(v);
24607                     values[f.name] = v;
24608                 }
24609                 var record = new recordType(values, id);
24610                 record.node = n;
24611                 records[records.length] = record;
24612             }
24613
24614             return {
24615                 success : success,
24616                 records : records,
24617                 totalRecords : totalRecords || records.length
24618             };
24619     }
24620 });/*
24621  * Based on:
24622  * Ext JS Library 1.1.1
24623  * Copyright(c) 2006-2007, Ext JS, LLC.
24624  *
24625  * Originally Released Under LGPL - original licence link has changed is not relivant.
24626  *
24627  * Fork - LGPL
24628  * <script type="text/javascript">
24629  */
24630
24631 /**
24632  * @class Roo.data.ArrayReader
24633  * @extends Roo.data.DataReader
24634  * Data reader class to create an Array of Roo.data.Record objects from an Array.
24635  * Each element of that Array represents a row of data fields. The
24636  * fields are pulled into a Record object using as a subscript, the <em>mapping</em> property
24637  * of the field definition if it exists, or the field's ordinal position in the definition.<br>
24638  * <p>
24639  * Example code:.
24640  * <pre><code>
24641 var RecordDef = Roo.data.Record.create([
24642     {name: 'name', mapping: 1},         // "mapping" only needed if an "id" field is present which
24643     {name: 'occupation', mapping: 2}    // precludes using the ordinal position as the index.
24644 ]);
24645 var myReader = new Roo.data.ArrayReader({
24646     id: 0                     // The subscript within row Array that provides an ID for the Record (optional)
24647 }, RecordDef);
24648 </code></pre>
24649  * <p>
24650  * This would consume an Array like this:
24651  * <pre><code>
24652 [ [1, 'Bill', 'Gardener'], [2, 'Ben', 'Horticulturalist'] ]
24653   </code></pre>
24654  
24655  * @constructor
24656  * Create a new JsonReader
24657  * @param {Object} meta Metadata configuration options.
24658  * @param {Object|Array} recordType Either an Array of field definition objects
24659  * 
24660  * @cfg {Array} fields Array of field definition objects
24661  * @cfg {String} id Name of the property within a row object that contains a record identifier value.
24662  * as specified to {@link Roo.data.Record#create},
24663  * or an {@link Roo.data.Record} object
24664  *
24665  * 
24666  * created using {@link Roo.data.Record#create}.
24667  */
24668 Roo.data.ArrayReader = function(meta, recordType)
24669 {    
24670     Roo.data.ArrayReader.superclass.constructor.call(this, meta, recordType||meta.fields);
24671 };
24672
24673 Roo.extend(Roo.data.ArrayReader, Roo.data.JsonReader, {
24674     
24675       /**
24676      * Create a data block containing Roo.data.Records from an XML document.
24677      * @param {Object} o An Array of row objects which represents the dataset.
24678      * @return {Object} A data block which is used by an {@link Roo.data.Store} object as
24679      * a cache of Roo.data.Records.
24680      */
24681     readRecords : function(o)
24682     {
24683         var sid = this.meta ? this.meta.id : null;
24684         var recordType = this.recordType, fields = recordType.prototype.fields;
24685         var records = [];
24686         var root = o;
24687         for(var i = 0; i < root.length; i++){
24688                 var n = root[i];
24689             var values = {};
24690             var id = ((sid || sid === 0) && n[sid] !== undefined && n[sid] !== "" ? n[sid] : null);
24691             for(var j = 0, jlen = fields.length; j < jlen; j++){
24692                 var f = fields.items[j];
24693                 var k = f.mapping !== undefined && f.mapping !== null ? f.mapping : j;
24694                 var v = n[k] !== undefined ? n[k] : f.defaultValue;
24695                 v = f.convert(v);
24696                 values[f.name] = v;
24697             }
24698             var record = new recordType(values, id);
24699             record.json = n;
24700             records[records.length] = record;
24701         }
24702         return {
24703             records : records,
24704             totalRecords : records.length
24705         };
24706     },
24707     /**
24708      * using 'cn' the nested child reader read the child array into it's child stores.
24709      * @param {Object} rec The record with a 'children array
24710      */
24711     toLoadData: function(rec)
24712     {
24713         // expect rec just to be an array.. eg [a,b,c, [...] << cn ]
24714         return typeof(rec.data.cn) == 'undefined' ? [] : rec.data.cn;
24715         
24716     }
24717     
24718     
24719 });/*
24720  * Based on:
24721  * Ext JS Library 1.1.1
24722  * Copyright(c) 2006-2007, Ext JS, LLC.
24723  *
24724  * Originally Released Under LGPL - original licence link has changed is not relivant.
24725  *
24726  * Fork - LGPL
24727  * <script type="text/javascript">
24728  */
24729
24730
24731 /**
24732  * @class Roo.data.Tree
24733  * @extends Roo.util.Observable
24734  * Represents a tree data structure and bubbles all the events for its nodes. The nodes
24735  * in the tree have most standard DOM functionality.
24736  * @constructor
24737  * @param {Node} root (optional) The root node
24738  */
24739 Roo.data.Tree = function(root){
24740    this.nodeHash = {};
24741    /**
24742     * The root node for this tree
24743     * @type Node
24744     */
24745    this.root = null;
24746    if(root){
24747        this.setRootNode(root);
24748    }
24749    this.addEvents({
24750        /**
24751         * @event append
24752         * Fires when a new child node is appended to a node in this tree.
24753         * @param {Tree} tree The owner tree
24754         * @param {Node} parent The parent node
24755         * @param {Node} node The newly appended node
24756         * @param {Number} index The index of the newly appended node
24757         */
24758        "append" : true,
24759        /**
24760         * @event remove
24761         * Fires when a child node is removed from a node in this tree.
24762         * @param {Tree} tree The owner tree
24763         * @param {Node} parent The parent node
24764         * @param {Node} node The child node removed
24765         */
24766        "remove" : true,
24767        /**
24768         * @event move
24769         * Fires when a node is moved to a new location in the tree
24770         * @param {Tree} tree The owner tree
24771         * @param {Node} node The node moved
24772         * @param {Node} oldParent The old parent of this node
24773         * @param {Node} newParent The new parent of this node
24774         * @param {Number} index The index it was moved to
24775         */
24776        "move" : true,
24777        /**
24778         * @event insert
24779         * Fires when a new child node is inserted in a node in this tree.
24780         * @param {Tree} tree The owner tree
24781         * @param {Node} parent The parent node
24782         * @param {Node} node The child node inserted
24783         * @param {Node} refNode The child node the node was inserted before
24784         */
24785        "insert" : true,
24786        /**
24787         * @event beforeappend
24788         * Fires before a new child is appended to a node in this tree, return false to cancel the append.
24789         * @param {Tree} tree The owner tree
24790         * @param {Node} parent The parent node
24791         * @param {Node} node The child node to be appended
24792         */
24793        "beforeappend" : true,
24794        /**
24795         * @event beforeremove
24796         * Fires before a child is removed from a node in this tree, return false to cancel the remove.
24797         * @param {Tree} tree The owner tree
24798         * @param {Node} parent The parent node
24799         * @param {Node} node The child node to be removed
24800         */
24801        "beforeremove" : true,
24802        /**
24803         * @event beforemove
24804         * Fires before a node is moved to a new location in the tree. Return false to cancel the move.
24805         * @param {Tree} tree The owner tree
24806         * @param {Node} node The node being moved
24807         * @param {Node} oldParent The parent of the node
24808         * @param {Node} newParent The new parent the node is moving to
24809         * @param {Number} index The index it is being moved to
24810         */
24811        "beforemove" : true,
24812        /**
24813         * @event beforeinsert
24814         * Fires before a new child is inserted in a node in this tree, return false to cancel the insert.
24815         * @param {Tree} tree The owner tree
24816         * @param {Node} parent The parent node
24817         * @param {Node} node The child node to be inserted
24818         * @param {Node} refNode The child node the node is being inserted before
24819         */
24820        "beforeinsert" : true
24821    });
24822
24823     Roo.data.Tree.superclass.constructor.call(this);
24824 };
24825
24826 Roo.extend(Roo.data.Tree, Roo.util.Observable, {
24827     pathSeparator: "/",
24828
24829     proxyNodeEvent : function(){
24830         return this.fireEvent.apply(this, arguments);
24831     },
24832
24833     /**
24834      * Returns the root node for this tree.
24835      * @return {Node}
24836      */
24837     getRootNode : function(){
24838         return this.root;
24839     },
24840
24841     /**
24842      * Sets the root node for this tree.
24843      * @param {Node} node
24844      * @return {Node}
24845      */
24846     setRootNode : function(node){
24847         this.root = node;
24848         node.ownerTree = this;
24849         node.isRoot = true;
24850         this.registerNode(node);
24851         return node;
24852     },
24853
24854     /**
24855      * Gets a node in this tree by its id.
24856      * @param {String} id
24857      * @return {Node}
24858      */
24859     getNodeById : function(id){
24860         return this.nodeHash[id];
24861     },
24862
24863     registerNode : function(node){
24864         this.nodeHash[node.id] = node;
24865     },
24866
24867     unregisterNode : function(node){
24868         delete this.nodeHash[node.id];
24869     },
24870
24871     toString : function(){
24872         return "[Tree"+(this.id?" "+this.id:"")+"]";
24873     }
24874 });
24875
24876 /**
24877  * @class Roo.data.Node
24878  * @extends Roo.util.Observable
24879  * @cfg {Boolean} leaf true if this node is a leaf and does not have children
24880  * @cfg {String} id The id for this node. If one is not specified, one is generated.
24881  * @constructor
24882  * @param {Object} attributes The attributes/config for the node
24883  */
24884 Roo.data.Node = function(attributes){
24885     /**
24886      * The attributes supplied for the node. You can use this property to access any custom attributes you supplied.
24887      * @type {Object}
24888      */
24889     this.attributes = attributes || {};
24890     this.leaf = this.attributes.leaf;
24891     /**
24892      * The node id. @type String
24893      */
24894     this.id = this.attributes.id;
24895     if(!this.id){
24896         this.id = Roo.id(null, "ynode-");
24897         this.attributes.id = this.id;
24898     }
24899      
24900     
24901     /**
24902      * All child nodes of this node. @type Array
24903      */
24904     this.childNodes = [];
24905     if(!this.childNodes.indexOf){ // indexOf is a must
24906         this.childNodes.indexOf = function(o){
24907             for(var i = 0, len = this.length; i < len; i++){
24908                 if(this[i] == o) {
24909                     return i;
24910                 }
24911             }
24912             return -1;
24913         };
24914     }
24915     /**
24916      * The parent node for this node. @type Node
24917      */
24918     this.parentNode = null;
24919     /**
24920      * The first direct child node of this node, or null if this node has no child nodes. @type Node
24921      */
24922     this.firstChild = null;
24923     /**
24924      * The last direct child node of this node, or null if this node has no child nodes. @type Node
24925      */
24926     this.lastChild = null;
24927     /**
24928      * The node immediately preceding this node in the tree, or null if there is no sibling node. @type Node
24929      */
24930     this.previousSibling = null;
24931     /**
24932      * The node immediately following this node in the tree, or null if there is no sibling node. @type Node
24933      */
24934     this.nextSibling = null;
24935
24936     this.addEvents({
24937        /**
24938         * @event append
24939         * Fires when a new child node is appended
24940         * @param {Tree} tree The owner tree
24941         * @param {Node} this This node
24942         * @param {Node} node The newly appended node
24943         * @param {Number} index The index of the newly appended node
24944         */
24945        "append" : true,
24946        /**
24947         * @event remove
24948         * Fires when a child node is removed
24949         * @param {Tree} tree The owner tree
24950         * @param {Node} this This node
24951         * @param {Node} node The removed node
24952         */
24953        "remove" : true,
24954        /**
24955         * @event move
24956         * Fires when this node is moved to a new location in the tree
24957         * @param {Tree} tree The owner tree
24958         * @param {Node} this This node
24959         * @param {Node} oldParent The old parent of this node
24960         * @param {Node} newParent The new parent of this node
24961         * @param {Number} index The index it was moved to
24962         */
24963        "move" : true,
24964        /**
24965         * @event insert
24966         * Fires when a new child node is inserted.
24967         * @param {Tree} tree The owner tree
24968         * @param {Node} this This node
24969         * @param {Node} node The child node inserted
24970         * @param {Node} refNode The child node the node was inserted before
24971         */
24972        "insert" : true,
24973        /**
24974         * @event beforeappend
24975         * Fires before a new child is appended, return false to cancel the append.
24976         * @param {Tree} tree The owner tree
24977         * @param {Node} this This node
24978         * @param {Node} node The child node to be appended
24979         */
24980        "beforeappend" : true,
24981        /**
24982         * @event beforeremove
24983         * Fires before a child is removed, return false to cancel the remove.
24984         * @param {Tree} tree The owner tree
24985         * @param {Node} this This node
24986         * @param {Node} node The child node to be removed
24987         */
24988        "beforeremove" : true,
24989        /**
24990         * @event beforemove
24991         * Fires before this node is moved to a new location in the tree. Return false to cancel the move.
24992         * @param {Tree} tree The owner tree
24993         * @param {Node} this This node
24994         * @param {Node} oldParent The parent of this node
24995         * @param {Node} newParent The new parent this node is moving to
24996         * @param {Number} index The index it is being moved to
24997         */
24998        "beforemove" : true,
24999        /**
25000         * @event beforeinsert
25001         * Fires before a new child is inserted, return false to cancel the insert.
25002         * @param {Tree} tree The owner tree
25003         * @param {Node} this This node
25004         * @param {Node} node The child node to be inserted
25005         * @param {Node} refNode The child node the node is being inserted before
25006         */
25007        "beforeinsert" : true
25008    });
25009     this.listeners = this.attributes.listeners;
25010     Roo.data.Node.superclass.constructor.call(this);
25011 };
25012
25013 Roo.extend(Roo.data.Node, Roo.util.Observable, {
25014     fireEvent : function(evtName){
25015         // first do standard event for this node
25016         if(Roo.data.Node.superclass.fireEvent.apply(this, arguments) === false){
25017             return false;
25018         }
25019         // then bubble it up to the tree if the event wasn't cancelled
25020         var ot = this.getOwnerTree();
25021         if(ot){
25022             if(ot.proxyNodeEvent.apply(ot, arguments) === false){
25023                 return false;
25024             }
25025         }
25026         return true;
25027     },
25028
25029     /**
25030      * Returns true if this node is a leaf
25031      * @return {Boolean}
25032      */
25033     isLeaf : function(){
25034         return this.leaf === true;
25035     },
25036
25037     // private
25038     setFirstChild : function(node){
25039         this.firstChild = node;
25040     },
25041
25042     //private
25043     setLastChild : function(node){
25044         this.lastChild = node;
25045     },
25046
25047
25048     /**
25049      * Returns true if this node is the last child of its parent
25050      * @return {Boolean}
25051      */
25052     isLast : function(){
25053        return (!this.parentNode ? true : this.parentNode.lastChild == this);
25054     },
25055
25056     /**
25057      * Returns true if this node is the first child of its parent
25058      * @return {Boolean}
25059      */
25060     isFirst : function(){
25061        return (!this.parentNode ? true : this.parentNode.firstChild == this);
25062     },
25063
25064     hasChildNodes : function(){
25065         return !this.isLeaf() && this.childNodes.length > 0;
25066     },
25067
25068     /**
25069      * Insert node(s) as the last child node of this node.
25070      * @param {Node/Array} node The node or Array of nodes to append
25071      * @return {Node} The appended node if single append, or null if an array was passed
25072      */
25073     appendChild : function(node){
25074         var multi = false;
25075         if(node instanceof Array){
25076             multi = node;
25077         }else if(arguments.length > 1){
25078             multi = arguments;
25079         }
25080         
25081         // if passed an array or multiple args do them one by one
25082         if(multi){
25083             for(var i = 0, len = multi.length; i < len; i++) {
25084                 this.appendChild(multi[i]);
25085             }
25086         }else{
25087             if(this.fireEvent("beforeappend", this.ownerTree, this, node) === false){
25088                 return false;
25089             }
25090             var index = this.childNodes.length;
25091             var oldParent = node.parentNode;
25092             // it's a move, make sure we move it cleanly
25093             if(oldParent){
25094                 if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index) === false){
25095                     return false;
25096                 }
25097                 oldParent.removeChild(node);
25098             }
25099             
25100             index = this.childNodes.length;
25101             if(index == 0){
25102                 this.setFirstChild(node);
25103             }
25104             this.childNodes.push(node);
25105             node.parentNode = this;
25106             var ps = this.childNodes[index-1];
25107             if(ps){
25108                 node.previousSibling = ps;
25109                 ps.nextSibling = node;
25110             }else{
25111                 node.previousSibling = null;
25112             }
25113             node.nextSibling = null;
25114             this.setLastChild(node);
25115             node.setOwnerTree(this.getOwnerTree());
25116             this.fireEvent("append", this.ownerTree, this, node, index);
25117             if(this.ownerTree) {
25118                 this.ownerTree.fireEvent("appendnode", this, node, index);
25119             }
25120             if(oldParent){
25121                 node.fireEvent("move", this.ownerTree, node, oldParent, this, index);
25122             }
25123             return node;
25124         }
25125     },
25126
25127     /**
25128      * Removes a child node from this node.
25129      * @param {Node} node The node to remove
25130      * @return {Node} The removed node
25131      */
25132     removeChild : function(node){
25133         var index = this.childNodes.indexOf(node);
25134         if(index == -1){
25135             return false;
25136         }
25137         if(this.fireEvent("beforeremove", this.ownerTree, this, node) === false){
25138             return false;
25139         }
25140
25141         // remove it from childNodes collection
25142         this.childNodes.splice(index, 1);
25143
25144         // update siblings
25145         if(node.previousSibling){
25146             node.previousSibling.nextSibling = node.nextSibling;
25147         }
25148         if(node.nextSibling){
25149             node.nextSibling.previousSibling = node.previousSibling;
25150         }
25151
25152         // update child refs
25153         if(this.firstChild == node){
25154             this.setFirstChild(node.nextSibling);
25155         }
25156         if(this.lastChild == node){
25157             this.setLastChild(node.previousSibling);
25158         }
25159
25160         node.setOwnerTree(null);
25161         // clear any references from the node
25162         node.parentNode = null;
25163         node.previousSibling = null;
25164         node.nextSibling = null;
25165         this.fireEvent("remove", this.ownerTree, this, node);
25166         return node;
25167     },
25168
25169     /**
25170      * Inserts the first node before the second node in this nodes childNodes collection.
25171      * @param {Node} node The node to insert
25172      * @param {Node} refNode The node to insert before (if null the node is appended)
25173      * @return {Node} The inserted node
25174      */
25175     insertBefore : function(node, refNode){
25176         if(!refNode){ // like standard Dom, refNode can be null for append
25177             return this.appendChild(node);
25178         }
25179         // nothing to do
25180         if(node == refNode){
25181             return false;
25182         }
25183
25184         if(this.fireEvent("beforeinsert", this.ownerTree, this, node, refNode) === false){
25185             return false;
25186         }
25187         var index = this.childNodes.indexOf(refNode);
25188         var oldParent = node.parentNode;
25189         var refIndex = index;
25190
25191         // when moving internally, indexes will change after remove
25192         if(oldParent == this && this.childNodes.indexOf(node) < index){
25193             refIndex--;
25194         }
25195
25196         // it's a move, make sure we move it cleanly
25197         if(oldParent){
25198             if(node.fireEvent("beforemove", node.getOwnerTree(), node, oldParent, this, index, refNode) === false){
25199                 return false;
25200             }
25201             oldParent.removeChild(node);
25202         }
25203         if(refIndex == 0){
25204             this.setFirstChild(node);
25205         }
25206         this.childNodes.splice(refIndex, 0, node);
25207         node.parentNode = this;
25208         var ps = this.childNodes[refIndex-1];
25209         if(ps){
25210             node.previousSibling = ps;
25211             ps.nextSibling = node;
25212         }else{
25213             node.previousSibling = null;
25214         }
25215         node.nextSibling = refNode;
25216         refNode.previousSibling = node;
25217         node.setOwnerTree(this.getOwnerTree());
25218         this.fireEvent("insert", this.ownerTree, this, node, refNode);
25219         if(oldParent){
25220             node.fireEvent("move", this.ownerTree, node, oldParent, this, refIndex, refNode);
25221         }
25222         return node;
25223     },
25224
25225     /**
25226      * Returns the child node at the specified index.
25227      * @param {Number} index
25228      * @return {Node}
25229      */
25230     item : function(index){
25231         return this.childNodes[index];
25232     },
25233
25234     /**
25235      * Replaces one child node in this node with another.
25236      * @param {Node} newChild The replacement node
25237      * @param {Node} oldChild The node to replace
25238      * @return {Node} The replaced node
25239      */
25240     replaceChild : function(newChild, oldChild){
25241         this.insertBefore(newChild, oldChild);
25242         this.removeChild(oldChild);
25243         return oldChild;
25244     },
25245
25246     /**
25247      * Returns the index of a child node
25248      * @param {Node} node
25249      * @return {Number} The index of the node or -1 if it was not found
25250      */
25251     indexOf : function(child){
25252         return this.childNodes.indexOf(child);
25253     },
25254
25255     /**
25256      * Returns the tree this node is in.
25257      * @return {Tree}
25258      */
25259     getOwnerTree : function(){
25260         // if it doesn't have one, look for one
25261         if(!this.ownerTree){
25262             var p = this;
25263             while(p){
25264                 if(p.ownerTree){
25265                     this.ownerTree = p.ownerTree;
25266                     break;
25267                 }
25268                 p = p.parentNode;
25269             }
25270         }
25271         return this.ownerTree;
25272     },
25273
25274     /**
25275      * Returns depth of this node (the root node has a depth of 0)
25276      * @return {Number}
25277      */
25278     getDepth : function(){
25279         var depth = 0;
25280         var p = this;
25281         while(p.parentNode){
25282             ++depth;
25283             p = p.parentNode;
25284         }
25285         return depth;
25286     },
25287
25288     // private
25289     setOwnerTree : function(tree){
25290         // if it's move, we need to update everyone
25291         if(tree != this.ownerTree){
25292             if(this.ownerTree){
25293                 this.ownerTree.unregisterNode(this);
25294             }
25295             this.ownerTree = tree;
25296             var cs = this.childNodes;
25297             for(var i = 0, len = cs.length; i < len; i++) {
25298                 cs[i].setOwnerTree(tree);
25299             }
25300             if(tree){
25301                 tree.registerNode(this);
25302             }
25303         }
25304     },
25305
25306     /**
25307      * Returns the path for this node. The path can be used to expand or select this node programmatically.
25308      * @param {String} attr (optional) The attr to use for the path (defaults to the node's id)
25309      * @return {String} The path
25310      */
25311     getPath : function(attr){
25312         attr = attr || "id";
25313         var p = this.parentNode;
25314         var b = [this.attributes[attr]];
25315         while(p){
25316             b.unshift(p.attributes[attr]);
25317             p = p.parentNode;
25318         }
25319         var sep = this.getOwnerTree().pathSeparator;
25320         return sep + b.join(sep);
25321     },
25322
25323     /**
25324      * Bubbles up the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25325      * function call will be the scope provided or the current node. The arguments to the function
25326      * will be the args provided or the current node. If the function returns false at any point,
25327      * the bubble is stopped.
25328      * @param {Function} fn The function to call
25329      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25330      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25331      */
25332     bubble : function(fn, scope, args){
25333         var p = this;
25334         while(p){
25335             if(fn.call(scope || p, args || p) === false){
25336                 break;
25337             }
25338             p = p.parentNode;
25339         }
25340     },
25341
25342     /**
25343      * Cascades down the tree from this node, calling the specified function with each node. The scope (<i>this</i>) of
25344      * function call will be the scope provided or the current node. The arguments to the function
25345      * will be the args provided or the current node. If the function returns false at any point,
25346      * the cascade is stopped on that branch.
25347      * @param {Function} fn The function to call
25348      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25349      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25350      */
25351     cascade : function(fn, scope, args){
25352         if(fn.call(scope || this, args || this) !== false){
25353             var cs = this.childNodes;
25354             for(var i = 0, len = cs.length; i < len; i++) {
25355                 cs[i].cascade(fn, scope, args);
25356             }
25357         }
25358     },
25359
25360     /**
25361      * Interates the child nodes of this node, calling the specified function with each node. The scope (<i>this</i>) of
25362      * function call will be the scope provided or the current node. The arguments to the function
25363      * will be the args provided or the current node. If the function returns false at any point,
25364      * the iteration stops.
25365      * @param {Function} fn The function to call
25366      * @param {Object} scope (optional) The scope of the function (defaults to current node)
25367      * @param {Array} args (optional) The args to call the function with (default to passing the current node)
25368      */
25369     eachChild : function(fn, scope, args){
25370         var cs = this.childNodes;
25371         for(var i = 0, len = cs.length; i < len; i++) {
25372                 if(fn.call(scope || this, args || cs[i]) === false){
25373                     break;
25374                 }
25375         }
25376     },
25377
25378     /**
25379      * Finds the first child that has the attribute with the specified value.
25380      * @param {String} attribute The attribute name
25381      * @param {Mixed} value The value to search for
25382      * @return {Node} The found child or null if none was found
25383      */
25384     findChild : function(attribute, value){
25385         var cs = this.childNodes;
25386         for(var i = 0, len = cs.length; i < len; i++) {
25387                 if(cs[i].attributes[attribute] == value){
25388                     return cs[i];
25389                 }
25390         }
25391         return null;
25392     },
25393
25394     /**
25395      * Finds the first child by a custom function. The child matches if the function passed
25396      * returns true.
25397      * @param {Function} fn
25398      * @param {Object} scope (optional)
25399      * @return {Node} The found child or null if none was found
25400      */
25401     findChildBy : function(fn, scope){
25402         var cs = this.childNodes;
25403         for(var i = 0, len = cs.length; i < len; i++) {
25404                 if(fn.call(scope||cs[i], cs[i]) === true){
25405                     return cs[i];
25406                 }
25407         }
25408         return null;
25409     },
25410
25411     /**
25412      * Sorts this nodes children using the supplied sort function
25413      * @param {Function} fn
25414      * @param {Object} scope (optional)
25415      */
25416     sort : function(fn, scope){
25417         var cs = this.childNodes;
25418         var len = cs.length;
25419         if(len > 0){
25420             var sortFn = scope ? function(){fn.apply(scope, arguments);} : fn;
25421             cs.sort(sortFn);
25422             for(var i = 0; i < len; i++){
25423                 var n = cs[i];
25424                 n.previousSibling = cs[i-1];
25425                 n.nextSibling = cs[i+1];
25426                 if(i == 0){
25427                     this.setFirstChild(n);
25428                 }
25429                 if(i == len-1){
25430                     this.setLastChild(n);
25431                 }
25432             }
25433         }
25434     },
25435
25436     /**
25437      * Returns true if this node is an ancestor (at any point) of the passed node.
25438      * @param {Node} node
25439      * @return {Boolean}
25440      */
25441     contains : function(node){
25442         return node.isAncestor(this);
25443     },
25444
25445     /**
25446      * Returns true if the passed node is an ancestor (at any point) of this node.
25447      * @param {Node} node
25448      * @return {Boolean}
25449      */
25450     isAncestor : function(node){
25451         var p = this.parentNode;
25452         while(p){
25453             if(p == node){
25454                 return true;
25455             }
25456             p = p.parentNode;
25457         }
25458         return false;
25459     },
25460
25461     toString : function(){
25462         return "[Node"+(this.id?" "+this.id:"")+"]";
25463     }
25464 });/*
25465  * Based on:
25466  * Ext JS Library 1.1.1
25467  * Copyright(c) 2006-2007, Ext JS, LLC.
25468  *
25469  * Originally Released Under LGPL - original licence link has changed is not relivant.
25470  *
25471  * Fork - LGPL
25472  * <script type="text/javascript">
25473  */
25474  (function(){ 
25475 /**
25476  * @class Roo.Layer
25477  * @extends Roo.Element
25478  * An extended {@link Roo.Element} object that supports a shadow and shim, constrain to viewport and
25479  * automatic maintaining of shadow/shim positions.
25480  * @cfg {Boolean} shim False to disable the iframe shim in browsers which need one (defaults to true)
25481  * @cfg {String/Boolean} shadow True to create a shadow element with default class "x-layer-shadow", or
25482  * you can pass a string with a CSS class name. False turns off the shadow.
25483  * @cfg {Object} dh DomHelper object config to create element with (defaults to {tag: "div", cls: "x-layer"}).
25484  * @cfg {Boolean} constrain False to disable constrain to viewport (defaults to true)
25485  * @cfg {String} cls CSS class to add to the element
25486  * @cfg {Number} zindex Starting z-index (defaults to 11000)
25487  * @cfg {Number} shadowOffset Number of pixels to offset the shadow (defaults to 3)
25488  * @constructor
25489  * @param {Object} config An object with config options.
25490  * @param {String/HTMLElement} existingEl (optional) Uses an existing DOM element. If the element is not found it creates it.
25491  */
25492
25493 Roo.Layer = function(config, existingEl){
25494     config = config || {};
25495     var dh = Roo.DomHelper;
25496     var cp = config.parentEl, pel = cp ? Roo.getDom(cp) : document.body;
25497     if(existingEl){
25498         this.dom = Roo.getDom(existingEl);
25499     }
25500     if(!this.dom){
25501         var o = config.dh || {tag: "div", cls: "x-layer"};
25502         this.dom = dh.append(pel, o);
25503     }
25504     if(config.cls){
25505         this.addClass(config.cls);
25506     }
25507     this.constrain = config.constrain !== false;
25508     this.visibilityMode = Roo.Element.VISIBILITY;
25509     if(config.id){
25510         this.id = this.dom.id = config.id;
25511     }else{
25512         this.id = Roo.id(this.dom);
25513     }
25514     this.zindex = config.zindex || this.getZIndex();
25515     this.position("absolute", this.zindex);
25516     if(config.shadow){
25517         this.shadowOffset = config.shadowOffset || 4;
25518         this.shadow = new Roo.Shadow({
25519             offset : this.shadowOffset,
25520             mode : config.shadow
25521         });
25522     }else{
25523         this.shadowOffset = 0;
25524     }
25525     this.useShim = config.shim !== false && Roo.useShims;
25526     this.useDisplay = config.useDisplay;
25527     this.hide();
25528 };
25529
25530 var supr = Roo.Element.prototype;
25531
25532 // shims are shared among layer to keep from having 100 iframes
25533 var shims = [];
25534
25535 Roo.extend(Roo.Layer, Roo.Element, {
25536
25537     getZIndex : function(){
25538         return this.zindex || parseInt(this.getStyle("z-index"), 10) || 11000;
25539     },
25540
25541     getShim : function(){
25542         if(!this.useShim){
25543             return null;
25544         }
25545         if(this.shim){
25546             return this.shim;
25547         }
25548         var shim = shims.shift();
25549         if(!shim){
25550             shim = this.createShim();
25551             shim.enableDisplayMode('block');
25552             shim.dom.style.display = 'none';
25553             shim.dom.style.visibility = 'visible';
25554         }
25555         var pn = this.dom.parentNode;
25556         if(shim.dom.parentNode != pn){
25557             pn.insertBefore(shim.dom, this.dom);
25558         }
25559         shim.setStyle('z-index', this.getZIndex()-2);
25560         this.shim = shim;
25561         return shim;
25562     },
25563
25564     hideShim : function(){
25565         if(this.shim){
25566             this.shim.setDisplayed(false);
25567             shims.push(this.shim);
25568             delete this.shim;
25569         }
25570     },
25571
25572     disableShadow : function(){
25573         if(this.shadow){
25574             this.shadowDisabled = true;
25575             this.shadow.hide();
25576             this.lastShadowOffset = this.shadowOffset;
25577             this.shadowOffset = 0;
25578         }
25579     },
25580
25581     enableShadow : function(show){
25582         if(this.shadow){
25583             this.shadowDisabled = false;
25584             this.shadowOffset = this.lastShadowOffset;
25585             delete this.lastShadowOffset;
25586             if(show){
25587                 this.sync(true);
25588             }
25589         }
25590     },
25591
25592     // private
25593     // this code can execute repeatedly in milliseconds (i.e. during a drag) so
25594     // code size was sacrificed for effeciency (e.g. no getBox/setBox, no XY calls)
25595     sync : function(doShow){
25596         var sw = this.shadow;
25597         if(!this.updating && this.isVisible() && (sw || this.useShim)){
25598             var sh = this.getShim();
25599
25600             var w = this.getWidth(),
25601                 h = this.getHeight();
25602
25603             var l = this.getLeft(true),
25604                 t = this.getTop(true);
25605
25606             if(sw && !this.shadowDisabled){
25607                 if(doShow && !sw.isVisible()){
25608                     sw.show(this);
25609                 }else{
25610                     sw.realign(l, t, w, h);
25611                 }
25612                 if(sh){
25613                     if(doShow){
25614                        sh.show();
25615                     }
25616                     // fit the shim behind the shadow, so it is shimmed too
25617                     var a = sw.adjusts, s = sh.dom.style;
25618                     s.left = (Math.min(l, l+a.l))+"px";
25619                     s.top = (Math.min(t, t+a.t))+"px";
25620                     s.width = (w+a.w)+"px";
25621                     s.height = (h+a.h)+"px";
25622                 }
25623             }else if(sh){
25624                 if(doShow){
25625                    sh.show();
25626                 }
25627                 sh.setSize(w, h);
25628                 sh.setLeftTop(l, t);
25629             }
25630             
25631         }
25632     },
25633
25634     // private
25635     destroy : function(){
25636         this.hideShim();
25637         if(this.shadow){
25638             this.shadow.hide();
25639         }
25640         this.removeAllListeners();
25641         var pn = this.dom.parentNode;
25642         if(pn){
25643             pn.removeChild(this.dom);
25644         }
25645         Roo.Element.uncache(this.id);
25646     },
25647
25648     remove : function(){
25649         this.destroy();
25650     },
25651
25652     // private
25653     beginUpdate : function(){
25654         this.updating = true;
25655     },
25656
25657     // private
25658     endUpdate : function(){
25659         this.updating = false;
25660         this.sync(true);
25661     },
25662
25663     // private
25664     hideUnders : function(negOffset){
25665         if(this.shadow){
25666             this.shadow.hide();
25667         }
25668         this.hideShim();
25669     },
25670
25671     // private
25672     constrainXY : function(){
25673         if(this.constrain){
25674             var vw = Roo.lib.Dom.getViewWidth(),
25675                 vh = Roo.lib.Dom.getViewHeight();
25676             var s = Roo.get(document).getScroll();
25677
25678             var xy = this.getXY();
25679             var x = xy[0], y = xy[1];   
25680             var w = this.dom.offsetWidth+this.shadowOffset, h = this.dom.offsetHeight+this.shadowOffset;
25681             // only move it if it needs it
25682             var moved = false;
25683             // first validate right/bottom
25684             if((x + w) > vw+s.left){
25685                 x = vw - w - this.shadowOffset;
25686                 moved = true;
25687             }
25688             if((y + h) > vh+s.top){
25689                 y = vh - h - this.shadowOffset;
25690                 moved = true;
25691             }
25692             // then make sure top/left isn't negative
25693             if(x < s.left){
25694                 x = s.left;
25695                 moved = true;
25696             }
25697             if(y < s.top){
25698                 y = s.top;
25699                 moved = true;
25700             }
25701             if(moved){
25702                 if(this.avoidY){
25703                     var ay = this.avoidY;
25704                     if(y <= ay && (y+h) >= ay){
25705                         y = ay-h-5;   
25706                     }
25707                 }
25708                 xy = [x, y];
25709                 this.storeXY(xy);
25710                 supr.setXY.call(this, xy);
25711                 this.sync();
25712             }
25713         }
25714     },
25715
25716     isVisible : function(){
25717         return this.visible;    
25718     },
25719
25720     // private
25721     showAction : function(){
25722         this.visible = true; // track visibility to prevent getStyle calls
25723         if(this.useDisplay === true){
25724             this.setDisplayed("");
25725         }else if(this.lastXY){
25726             supr.setXY.call(this, this.lastXY);
25727         }else if(this.lastLT){
25728             supr.setLeftTop.call(this, this.lastLT[0], this.lastLT[1]);
25729         }
25730     },
25731
25732     // private
25733     hideAction : function(){
25734         this.visible = false;
25735         if(this.useDisplay === true){
25736             this.setDisplayed(false);
25737         }else{
25738             this.setLeftTop(-10000,-10000);
25739         }
25740     },
25741
25742     // overridden Element method
25743     setVisible : function(v, a, d, c, e){
25744         if(v){
25745             this.showAction();
25746         }
25747         if(a && v){
25748             var cb = function(){
25749                 this.sync(true);
25750                 if(c){
25751                     c();
25752                 }
25753             }.createDelegate(this);
25754             supr.setVisible.call(this, true, true, d, cb, e);
25755         }else{
25756             if(!v){
25757                 this.hideUnders(true);
25758             }
25759             var cb = c;
25760             if(a){
25761                 cb = function(){
25762                     this.hideAction();
25763                     if(c){
25764                         c();
25765                     }
25766                 }.createDelegate(this);
25767             }
25768             supr.setVisible.call(this, v, a, d, cb, e);
25769             if(v){
25770                 this.sync(true);
25771             }else if(!a){
25772                 this.hideAction();
25773             }
25774         }
25775     },
25776
25777     storeXY : function(xy){
25778         delete this.lastLT;
25779         this.lastXY = xy;
25780     },
25781
25782     storeLeftTop : function(left, top){
25783         delete this.lastXY;
25784         this.lastLT = [left, top];
25785     },
25786
25787     // private
25788     beforeFx : function(){
25789         this.beforeAction();
25790         return Roo.Layer.superclass.beforeFx.apply(this, arguments);
25791     },
25792
25793     // private
25794     afterFx : function(){
25795         Roo.Layer.superclass.afterFx.apply(this, arguments);
25796         this.sync(this.isVisible());
25797     },
25798
25799     // private
25800     beforeAction : function(){
25801         if(!this.updating && this.shadow){
25802             this.shadow.hide();
25803         }
25804     },
25805
25806     // overridden Element method
25807     setLeft : function(left){
25808         this.storeLeftTop(left, this.getTop(true));
25809         supr.setLeft.apply(this, arguments);
25810         this.sync();
25811     },
25812
25813     setTop : function(top){
25814         this.storeLeftTop(this.getLeft(true), top);
25815         supr.setTop.apply(this, arguments);
25816         this.sync();
25817     },
25818
25819     setLeftTop : function(left, top){
25820         this.storeLeftTop(left, top);
25821         supr.setLeftTop.apply(this, arguments);
25822         this.sync();
25823     },
25824
25825     setXY : function(xy, a, d, c, e){
25826         this.fixDisplay();
25827         this.beforeAction();
25828         this.storeXY(xy);
25829         var cb = this.createCB(c);
25830         supr.setXY.call(this, xy, a, d, cb, e);
25831         if(!a){
25832             cb();
25833         }
25834     },
25835
25836     // private
25837     createCB : function(c){
25838         var el = this;
25839         return function(){
25840             el.constrainXY();
25841             el.sync(true);
25842             if(c){
25843                 c();
25844             }
25845         };
25846     },
25847
25848     // overridden Element method
25849     setX : function(x, a, d, c, e){
25850         this.setXY([x, this.getY()], a, d, c, e);
25851     },
25852
25853     // overridden Element method
25854     setY : function(y, a, d, c, e){
25855         this.setXY([this.getX(), y], a, d, c, e);
25856     },
25857
25858     // overridden Element method
25859     setSize : function(w, h, a, d, c, e){
25860         this.beforeAction();
25861         var cb = this.createCB(c);
25862         supr.setSize.call(this, w, h, a, d, cb, e);
25863         if(!a){
25864             cb();
25865         }
25866     },
25867
25868     // overridden Element method
25869     setWidth : function(w, a, d, c, e){
25870         this.beforeAction();
25871         var cb = this.createCB(c);
25872         supr.setWidth.call(this, w, a, d, cb, e);
25873         if(!a){
25874             cb();
25875         }
25876     },
25877
25878     // overridden Element method
25879     setHeight : function(h, a, d, c, e){
25880         this.beforeAction();
25881         var cb = this.createCB(c);
25882         supr.setHeight.call(this, h, a, d, cb, e);
25883         if(!a){
25884             cb();
25885         }
25886     },
25887
25888     // overridden Element method
25889     setBounds : function(x, y, w, h, a, d, c, e){
25890         this.beforeAction();
25891         var cb = this.createCB(c);
25892         if(!a){
25893             this.storeXY([x, y]);
25894             supr.setXY.call(this, [x, y]);
25895             supr.setSize.call(this, w, h, a, d, cb, e);
25896             cb();
25897         }else{
25898             supr.setBounds.call(this, x, y, w, h, a, d, cb, e);
25899         }
25900         return this;
25901     },
25902     
25903     /**
25904      * Sets the z-index of this layer and adjusts any shadow and shim z-indexes. The layer z-index is automatically
25905      * incremented by two more than the value passed in so that it always shows above any shadow or shim (the shadow
25906      * element, if any, will be assigned z-index + 1, and the shim element, if any, will be assigned the unmodified z-index).
25907      * @param {Number} zindex The new z-index to set
25908      * @return {this} The Layer
25909      */
25910     setZIndex : function(zindex){
25911         this.zindex = zindex;
25912         this.setStyle("z-index", zindex + 2);
25913         if(this.shadow){
25914             this.shadow.setZIndex(zindex + 1);
25915         }
25916         if(this.shim){
25917             this.shim.setStyle("z-index", zindex);
25918         }
25919     }
25920 });
25921 })();/*
25922  * Based on:
25923  * Ext JS Library 1.1.1
25924  * Copyright(c) 2006-2007, Ext JS, LLC.
25925  *
25926  * Originally Released Under LGPL - original licence link has changed is not relivant.
25927  *
25928  * Fork - LGPL
25929  * <script type="text/javascript">
25930  */
25931
25932
25933 /**
25934  * @class Roo.Shadow
25935  * Simple class that can provide a shadow effect for any element.  Note that the element MUST be absolutely positioned,
25936  * and the shadow does not provide any shimming.  This should be used only in simple cases -- for more advanced
25937  * functionality that can also provide the same shadow effect, see the {@link Roo.Layer} class.
25938  * @constructor
25939  * Create a new Shadow
25940  * @param {Object} config The config object
25941  */
25942 Roo.Shadow = function(config){
25943     Roo.apply(this, config);
25944     if(typeof this.mode != "string"){
25945         this.mode = this.defaultMode;
25946     }
25947     var o = this.offset, a = {h: 0};
25948     var rad = Math.floor(this.offset/2);
25949     switch(this.mode.toLowerCase()){ // all this hideous nonsense calculates the various offsets for shadows
25950         case "drop":
25951             a.w = 0;
25952             a.l = a.t = o;
25953             a.t -= 1;
25954             if(Roo.isIE){
25955                 a.l -= this.offset + rad;
25956                 a.t -= this.offset + rad;
25957                 a.w -= rad;
25958                 a.h -= rad;
25959                 a.t += 1;
25960             }
25961         break;
25962         case "sides":
25963             a.w = (o*2);
25964             a.l = -o;
25965             a.t = o-1;
25966             if(Roo.isIE){
25967                 a.l -= (this.offset - rad);
25968                 a.t -= this.offset + rad;
25969                 a.l += 1;
25970                 a.w -= (this.offset - rad)*2;
25971                 a.w -= rad + 1;
25972                 a.h -= 1;
25973             }
25974         break;
25975         case "frame":
25976             a.w = a.h = (o*2);
25977             a.l = a.t = -o;
25978             a.t += 1;
25979             a.h -= 2;
25980             if(Roo.isIE){
25981                 a.l -= (this.offset - rad);
25982                 a.t -= (this.offset - rad);
25983                 a.l += 1;
25984                 a.w -= (this.offset + rad + 1);
25985                 a.h -= (this.offset + rad);
25986                 a.h += 1;
25987             }
25988         break;
25989     };
25990
25991     this.adjusts = a;
25992 };
25993
25994 Roo.Shadow.prototype = {
25995     /**
25996      * @cfg {String} mode
25997      * The shadow display mode.  Supports the following options:<br />
25998      * sides: Shadow displays on both sides and bottom only<br />
25999      * frame: Shadow displays equally on all four sides<br />
26000      * drop: Traditional bottom-right drop shadow (default)
26001      */
26002     /**
26003      * @cfg {String} offset
26004      * The number of pixels to offset the shadow from the element (defaults to 4)
26005      */
26006     offset: 4,
26007
26008     // private
26009     defaultMode: "drop",
26010
26011     /**
26012      * Displays the shadow under the target element
26013      * @param {String/HTMLElement/Element} targetEl The id or element under which the shadow should display
26014      */
26015     show : function(target){
26016         target = Roo.get(target);
26017         if(!this.el){
26018             this.el = Roo.Shadow.Pool.pull();
26019             if(this.el.dom.nextSibling != target.dom){
26020                 this.el.insertBefore(target);
26021             }
26022         }
26023         this.el.setStyle("z-index", this.zIndex || parseInt(target.getStyle("z-index"), 10)-1);
26024         if(Roo.isIE){
26025             this.el.dom.style.filter="progid:DXImageTransform.Microsoft.alpha(opacity=50) progid:DXImageTransform.Microsoft.Blur(pixelradius="+(this.offset)+")";
26026         }
26027         this.realign(
26028             target.getLeft(true),
26029             target.getTop(true),
26030             target.getWidth(),
26031             target.getHeight()
26032         );
26033         this.el.dom.style.display = "block";
26034     },
26035
26036     /**
26037      * Returns true if the shadow is visible, else false
26038      */
26039     isVisible : function(){
26040         return this.el ? true : false;  
26041     },
26042
26043     /**
26044      * Direct alignment when values are already available. Show must be called at least once before
26045      * calling this method to ensure it is initialized.
26046      * @param {Number} left The target element left position
26047      * @param {Number} top The target element top position
26048      * @param {Number} width The target element width
26049      * @param {Number} height The target element height
26050      */
26051     realign : function(l, t, w, h){
26052         if(!this.el){
26053             return;
26054         }
26055         var a = this.adjusts, d = this.el.dom, s = d.style;
26056         var iea = 0;
26057         s.left = (l+a.l)+"px";
26058         s.top = (t+a.t)+"px";
26059         var sw = (w+a.w), sh = (h+a.h), sws = sw +"px", shs = sh + "px";
26060  
26061         if(s.width != sws || s.height != shs){
26062             s.width = sws;
26063             s.height = shs;
26064             if(!Roo.isIE){
26065                 var cn = d.childNodes;
26066                 var sww = Math.max(0, (sw-12))+"px";
26067                 cn[0].childNodes[1].style.width = sww;
26068                 cn[1].childNodes[1].style.width = sww;
26069                 cn[2].childNodes[1].style.width = sww;
26070                 cn[1].style.height = Math.max(0, (sh-12))+"px";
26071             }
26072         }
26073     },
26074
26075     /**
26076      * Hides this shadow
26077      */
26078     hide : function(){
26079         if(this.el){
26080             this.el.dom.style.display = "none";
26081             Roo.Shadow.Pool.push(this.el);
26082             delete this.el;
26083         }
26084     },
26085
26086     /**
26087      * Adjust the z-index of this shadow
26088      * @param {Number} zindex The new z-index
26089      */
26090     setZIndex : function(z){
26091         this.zIndex = z;
26092         if(this.el){
26093             this.el.setStyle("z-index", z);
26094         }
26095     }
26096 };
26097
26098 // Private utility class that manages the internal Shadow cache
26099 Roo.Shadow.Pool = function(){
26100     var p = [];
26101     var markup = Roo.isIE ?
26102                  '<div class="x-ie-shadow"></div>' :
26103                  '<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>';
26104     return {
26105         pull : function(){
26106             var sh = p.shift();
26107             if(!sh){
26108                 sh = Roo.get(Roo.DomHelper.insertHtml("beforeBegin", document.body.firstChild, markup));
26109                 sh.autoBoxAdjust = false;
26110             }
26111             return sh;
26112         },
26113
26114         push : function(sh){
26115             p.push(sh);
26116         }
26117     };
26118 }();/*
26119  * Based on:
26120  * Ext JS Library 1.1.1
26121  * Copyright(c) 2006-2007, Ext JS, LLC.
26122  *
26123  * Originally Released Under LGPL - original licence link has changed is not relivant.
26124  *
26125  * Fork - LGPL
26126  * <script type="text/javascript">
26127  */
26128
26129
26130 /**
26131  * @class Roo.SplitBar
26132  * @extends Roo.util.Observable
26133  * Creates draggable splitter bar functionality from two elements (element to be dragged and element to be resized).
26134  * <br><br>
26135  * Usage:
26136  * <pre><code>
26137 var split = new Roo.SplitBar("elementToDrag", "elementToSize",
26138                    Roo.SplitBar.HORIZONTAL, Roo.SplitBar.LEFT);
26139 split.setAdapter(new Roo.SplitBar.AbsoluteLayoutAdapter("container"));
26140 split.minSize = 100;
26141 split.maxSize = 600;
26142 split.animate = true;
26143 split.on('moved', splitterMoved);
26144 </code></pre>
26145  * @constructor
26146  * Create a new SplitBar
26147  * @param {String/HTMLElement/Roo.Element} dragElement The element to be dragged and act as the SplitBar. 
26148  * @param {String/HTMLElement/Roo.Element} resizingElement The element to be resized based on where the SplitBar element is dragged 
26149  * @param {Number} orientation (optional) Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26150  * @param {Number} placement (optional) Either Roo.SplitBar.LEFT or Roo.SplitBar.RIGHT for horizontal or  
26151                         Roo.SplitBar.TOP or Roo.SplitBar.BOTTOM for vertical. (By default, this is determined automatically by the initial
26152                         position of the SplitBar).
26153  */
26154 Roo.SplitBar = function(dragElement, resizingElement, orientation, placement, existingProxy){
26155     
26156     /** @private */
26157     this.el = Roo.get(dragElement, true);
26158     this.el.dom.unselectable = "on";
26159     /** @private */
26160     this.resizingEl = Roo.get(resizingElement, true);
26161
26162     /**
26163      * @private
26164      * The orientation of the split. Either Roo.SplitBar.HORIZONTAL or Roo.SplitBar.VERTICAL. (Defaults to HORIZONTAL)
26165      * Note: If this is changed after creating the SplitBar, the placement property must be manually updated
26166      * @type Number
26167      */
26168     this.orientation = orientation || Roo.SplitBar.HORIZONTAL;
26169     
26170     /**
26171      * The minimum size of the resizing element. (Defaults to 0)
26172      * @type Number
26173      */
26174     this.minSize = 0;
26175     
26176     /**
26177      * The maximum size of the resizing element. (Defaults to 2000)
26178      * @type Number
26179      */
26180     this.maxSize = 2000;
26181     
26182     /**
26183      * Whether to animate the transition to the new size
26184      * @type Boolean
26185      */
26186     this.animate = false;
26187     
26188     /**
26189      * Whether to create a transparent shim that overlays the page when dragging, enables dragging across iframes.
26190      * @type Boolean
26191      */
26192     this.useShim = false;
26193     
26194     /** @private */
26195     this.shim = null;
26196     
26197     if(!existingProxy){
26198         /** @private */
26199         this.proxy = Roo.SplitBar.createProxy(this.orientation);
26200     }else{
26201         this.proxy = Roo.get(existingProxy).dom;
26202     }
26203     /** @private */
26204     this.dd = new Roo.dd.DDProxy(this.el.dom.id, "XSplitBars", {dragElId : this.proxy.id});
26205     
26206     /** @private */
26207     this.dd.b4StartDrag = this.onStartProxyDrag.createDelegate(this);
26208     
26209     /** @private */
26210     this.dd.endDrag = this.onEndProxyDrag.createDelegate(this);
26211     
26212     /** @private */
26213     this.dragSpecs = {};
26214     
26215     /**
26216      * @private The adapter to use to positon and resize elements
26217      */
26218     this.adapter = new Roo.SplitBar.BasicLayoutAdapter();
26219     this.adapter.init(this);
26220     
26221     if(this.orientation == Roo.SplitBar.HORIZONTAL){
26222         /** @private */
26223         this.placement = placement || (this.el.getX() > this.resizingEl.getX() ? Roo.SplitBar.LEFT : Roo.SplitBar.RIGHT);
26224         this.el.addClass("x-splitbar-h");
26225     }else{
26226         /** @private */
26227         this.placement = placement || (this.el.getY() > this.resizingEl.getY() ? Roo.SplitBar.TOP : Roo.SplitBar.BOTTOM);
26228         this.el.addClass("x-splitbar-v");
26229     }
26230     
26231     this.addEvents({
26232         /**
26233          * @event resize
26234          * Fires when the splitter is moved (alias for {@link #event-moved})
26235          * @param {Roo.SplitBar} this
26236          * @param {Number} newSize the new width or height
26237          */
26238         "resize" : true,
26239         /**
26240          * @event moved
26241          * Fires when the splitter is moved
26242          * @param {Roo.SplitBar} this
26243          * @param {Number} newSize the new width or height
26244          */
26245         "moved" : true,
26246         /**
26247          * @event beforeresize
26248          * Fires before the splitter is dragged
26249          * @param {Roo.SplitBar} this
26250          */
26251         "beforeresize" : true,
26252
26253         "beforeapply" : true
26254     });
26255
26256     Roo.util.Observable.call(this);
26257 };
26258
26259 Roo.extend(Roo.SplitBar, Roo.util.Observable, {
26260     onStartProxyDrag : function(x, y){
26261         this.fireEvent("beforeresize", this);
26262         if(!this.overlay){
26263             var o = Roo.DomHelper.insertFirst(document.body,  {cls: "x-drag-overlay", html: "&#160;"}, true);
26264             o.unselectable();
26265             o.enableDisplayMode("block");
26266             // all splitbars share the same overlay
26267             Roo.SplitBar.prototype.overlay = o;
26268         }
26269         this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
26270         this.overlay.show();
26271         Roo.get(this.proxy).setDisplayed("block");
26272         var size = this.adapter.getElementSize(this);
26273         this.activeMinSize = this.getMinimumSize();;
26274         this.activeMaxSize = this.getMaximumSize();;
26275         var c1 = size - this.activeMinSize;
26276         var c2 = Math.max(this.activeMaxSize - size, 0);
26277         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26278             this.dd.resetConstraints();
26279             this.dd.setXConstraint(
26280                 this.placement == Roo.SplitBar.LEFT ? c1 : c2, 
26281                 this.placement == Roo.SplitBar.LEFT ? c2 : c1
26282             );
26283             this.dd.setYConstraint(0, 0);
26284         }else{
26285             this.dd.resetConstraints();
26286             this.dd.setXConstraint(0, 0);
26287             this.dd.setYConstraint(
26288                 this.placement == Roo.SplitBar.TOP ? c1 : c2, 
26289                 this.placement == Roo.SplitBar.TOP ? c2 : c1
26290             );
26291          }
26292         this.dragSpecs.startSize = size;
26293         this.dragSpecs.startPoint = [x, y];
26294         Roo.dd.DDProxy.prototype.b4StartDrag.call(this.dd, x, y);
26295     },
26296     
26297     /** 
26298      * @private Called after the drag operation by the DDProxy
26299      */
26300     onEndProxyDrag : function(e){
26301         Roo.get(this.proxy).setDisplayed(false);
26302         var endPoint = Roo.lib.Event.getXY(e);
26303         if(this.overlay){
26304             this.overlay.hide();
26305         }
26306         var newSize;
26307         if(this.orientation == Roo.SplitBar.HORIZONTAL){
26308             newSize = this.dragSpecs.startSize + 
26309                 (this.placement == Roo.SplitBar.LEFT ?
26310                     endPoint[0] - this.dragSpecs.startPoint[0] :
26311                     this.dragSpecs.startPoint[0] - endPoint[0]
26312                 );
26313         }else{
26314             newSize = this.dragSpecs.startSize + 
26315                 (this.placement == Roo.SplitBar.TOP ?
26316                     endPoint[1] - this.dragSpecs.startPoint[1] :
26317                     this.dragSpecs.startPoint[1] - endPoint[1]
26318                 );
26319         }
26320         newSize = Math.min(Math.max(newSize, this.activeMinSize), this.activeMaxSize);
26321         if(newSize != this.dragSpecs.startSize){
26322             if(this.fireEvent('beforeapply', this, newSize) !== false){
26323                 this.adapter.setElementSize(this, newSize);
26324                 this.fireEvent("moved", this, newSize);
26325                 this.fireEvent("resize", this, newSize);
26326             }
26327         }
26328     },
26329     
26330     /**
26331      * Get the adapter this SplitBar uses
26332      * @return The adapter object
26333      */
26334     getAdapter : function(){
26335         return this.adapter;
26336     },
26337     
26338     /**
26339      * Set the adapter this SplitBar uses
26340      * @param {Object} adapter A SplitBar adapter object
26341      */
26342     setAdapter : function(adapter){
26343         this.adapter = adapter;
26344         this.adapter.init(this);
26345     },
26346     
26347     /**
26348      * Gets the minimum size for the resizing element
26349      * @return {Number} The minimum size
26350      */
26351     getMinimumSize : function(){
26352         return this.minSize;
26353     },
26354     
26355     /**
26356      * Sets the minimum size for the resizing element
26357      * @param {Number} minSize The minimum size
26358      */
26359     setMinimumSize : function(minSize){
26360         this.minSize = minSize;
26361     },
26362     
26363     /**
26364      * Gets the maximum size for the resizing element
26365      * @return {Number} The maximum size
26366      */
26367     getMaximumSize : function(){
26368         return this.maxSize;
26369     },
26370     
26371     /**
26372      * Sets the maximum size for the resizing element
26373      * @param {Number} maxSize The maximum size
26374      */
26375     setMaximumSize : function(maxSize){
26376         this.maxSize = maxSize;
26377     },
26378     
26379     /**
26380      * Sets the initialize size for the resizing element
26381      * @param {Number} size The initial size
26382      */
26383     setCurrentSize : function(size){
26384         var oldAnimate = this.animate;
26385         this.animate = false;
26386         this.adapter.setElementSize(this, size);
26387         this.animate = oldAnimate;
26388     },
26389     
26390     /**
26391      * Destroy this splitbar. 
26392      * @param {Boolean} removeEl True to remove the element
26393      */
26394     destroy : function(removeEl){
26395         if(this.shim){
26396             this.shim.remove();
26397         }
26398         this.dd.unreg();
26399         this.proxy.parentNode.removeChild(this.proxy);
26400         if(removeEl){
26401             this.el.remove();
26402         }
26403     }
26404 });
26405
26406 /**
26407  * @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.
26408  */
26409 Roo.SplitBar.createProxy = function(dir){
26410     var proxy = new Roo.Element(document.createElement("div"));
26411     proxy.unselectable();
26412     var cls = 'x-splitbar-proxy';
26413     proxy.addClass(cls + ' ' + (dir == Roo.SplitBar.HORIZONTAL ? cls +'-h' : cls + '-v'));
26414     document.body.appendChild(proxy.dom);
26415     return proxy.dom;
26416 };
26417
26418 /** 
26419  * @class Roo.SplitBar.BasicLayoutAdapter
26420  * Default Adapter. It assumes the splitter and resizing element are not positioned
26421  * elements and only gets/sets the width of the element. Generally used for table based layouts.
26422  */
26423 Roo.SplitBar.BasicLayoutAdapter = function(){
26424 };
26425
26426 Roo.SplitBar.BasicLayoutAdapter.prototype = {
26427     // do nothing for now
26428     init : function(s){
26429     
26430     },
26431     /**
26432      * Called before drag operations to get the current size of the resizing element. 
26433      * @param {Roo.SplitBar} s The SplitBar using this adapter
26434      */
26435      getElementSize : function(s){
26436         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26437             return s.resizingEl.getWidth();
26438         }else{
26439             return s.resizingEl.getHeight();
26440         }
26441     },
26442     
26443     /**
26444      * Called after drag operations to set the size of the resizing element.
26445      * @param {Roo.SplitBar} s The SplitBar using this adapter
26446      * @param {Number} newSize The new size to set
26447      * @param {Function} onComplete A function to be invoked when resizing is complete
26448      */
26449     setElementSize : function(s, newSize, onComplete){
26450         if(s.orientation == Roo.SplitBar.HORIZONTAL){
26451             if(!s.animate){
26452                 s.resizingEl.setWidth(newSize);
26453                 if(onComplete){
26454                     onComplete(s, newSize);
26455                 }
26456             }else{
26457                 s.resizingEl.setWidth(newSize, true, .1, onComplete, 'easeOut');
26458             }
26459         }else{
26460             
26461             if(!s.animate){
26462                 s.resizingEl.setHeight(newSize);
26463                 if(onComplete){
26464                     onComplete(s, newSize);
26465                 }
26466             }else{
26467                 s.resizingEl.setHeight(newSize, true, .1, onComplete, 'easeOut');
26468             }
26469         }
26470     }
26471 };
26472
26473 /** 
26474  *@class Roo.SplitBar.AbsoluteLayoutAdapter
26475  * @extends Roo.SplitBar.BasicLayoutAdapter
26476  * Adapter that  moves the splitter element to align with the resized sizing element. 
26477  * Used with an absolute positioned SplitBar.
26478  * @param {String/HTMLElement/Roo.Element} container The container that wraps around the absolute positioned content. If it's
26479  * document.body, make sure you assign an id to the body element.
26480  */
26481 Roo.SplitBar.AbsoluteLayoutAdapter = function(container){
26482     this.basic = new Roo.SplitBar.BasicLayoutAdapter();
26483     this.container = Roo.get(container);
26484 };
26485
26486 Roo.SplitBar.AbsoluteLayoutAdapter.prototype = {
26487     init : function(s){
26488         this.basic.init(s);
26489     },
26490     
26491     getElementSize : function(s){
26492         return this.basic.getElementSize(s);
26493     },
26494     
26495     setElementSize : function(s, newSize, onComplete){
26496         this.basic.setElementSize(s, newSize, this.moveSplitter.createDelegate(this, [s]));
26497     },
26498     
26499     moveSplitter : function(s){
26500         var yes = Roo.SplitBar;
26501         switch(s.placement){
26502             case yes.LEFT:
26503                 s.el.setX(s.resizingEl.getRight());
26504                 break;
26505             case yes.RIGHT:
26506                 s.el.setStyle("right", (this.container.getWidth() - s.resizingEl.getLeft()) + "px");
26507                 break;
26508             case yes.TOP:
26509                 s.el.setY(s.resizingEl.getBottom());
26510                 break;
26511             case yes.BOTTOM:
26512                 s.el.setY(s.resizingEl.getTop() - s.el.getHeight());
26513                 break;
26514         }
26515     }
26516 };
26517
26518 /**
26519  * Orientation constant - Create a vertical SplitBar
26520  * @static
26521  * @type Number
26522  */
26523 Roo.SplitBar.VERTICAL = 1;
26524
26525 /**
26526  * Orientation constant - Create a horizontal SplitBar
26527  * @static
26528  * @type Number
26529  */
26530 Roo.SplitBar.HORIZONTAL = 2;
26531
26532 /**
26533  * Placement constant - The resizing element is to the left of the splitter element
26534  * @static
26535  * @type Number
26536  */
26537 Roo.SplitBar.LEFT = 1;
26538
26539 /**
26540  * Placement constant - The resizing element is to the right of the splitter element
26541  * @static
26542  * @type Number
26543  */
26544 Roo.SplitBar.RIGHT = 2;
26545
26546 /**
26547  * Placement constant - The resizing element is positioned above the splitter element
26548  * @static
26549  * @type Number
26550  */
26551 Roo.SplitBar.TOP = 3;
26552
26553 /**
26554  * Placement constant - The resizing element is positioned under splitter element
26555  * @static
26556  * @type Number
26557  */
26558 Roo.SplitBar.BOTTOM = 4;
26559 /*
26560  * Based on:
26561  * Ext JS Library 1.1.1
26562  * Copyright(c) 2006-2007, Ext JS, LLC.
26563  *
26564  * Originally Released Under LGPL - original licence link has changed is not relivant.
26565  *
26566  * Fork - LGPL
26567  * <script type="text/javascript">
26568  */
26569
26570 /**
26571  * @class Roo.View
26572  * @extends Roo.util.Observable
26573  * Create a "View" for an element based on a data model or UpdateManager and the supplied DomHelper template. 
26574  * This class also supports single and multi selection modes. <br>
26575  * Create a data model bound view:
26576  <pre><code>
26577  var store = new Roo.data.Store(...);
26578
26579  var view = new Roo.View({
26580     el : "my-element",
26581     tpl : '&lt;div id="{0}"&gt;{2} - {1}&lt;/div&gt;', // auto create template
26582  
26583     singleSelect: true,
26584     selectedClass: "ydataview-selected",
26585     store: store
26586  });
26587
26588  // listen for node click?
26589  view.on("click", function(vw, index, node, e){
26590  alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
26591  });
26592
26593  // load XML data
26594  dataModel.load("foobar.xml");
26595  </code></pre>
26596  For an example of creating a JSON/UpdateManager view, see {@link Roo.JsonView}.
26597  * <br><br>
26598  * <b>Note: The root of your template must be a single node. Table/row implementations may work but are not supported due to
26599  * IE"s limited insertion support with tables and Opera"s faulty event bubbling.</b>
26600  * 
26601  * Note: old style constructor is still suported (container, template, config)
26602  * 
26603  * @constructor
26604  * Create a new View
26605  * @param {Object} config The config object
26606  * 
26607  */
26608 Roo.View = function(config, depreciated_tpl, depreciated_config){
26609     
26610     this.parent = false;
26611     
26612     if (typeof(depreciated_tpl) == 'undefined') {
26613         // new way.. - universal constructor.
26614         Roo.apply(this, config);
26615         this.el  = Roo.get(this.el);
26616     } else {
26617         // old format..
26618         this.el  = Roo.get(config);
26619         this.tpl = depreciated_tpl;
26620         Roo.apply(this, depreciated_config);
26621     }
26622     this.wrapEl  = this.el.wrap().wrap();
26623     ///this.el = this.wrapEla.appendChild(document.createElement("div"));
26624     
26625     
26626     if(typeof(this.tpl) == "string"){
26627         this.tpl = new Roo.Template(this.tpl);
26628     } else {
26629         // support xtype ctors..
26630         this.tpl = new Roo.factory(this.tpl, Roo);
26631     }
26632     
26633     
26634     this.tpl.compile();
26635     
26636     /** @private */
26637     this.addEvents({
26638         /**
26639          * @event beforeclick
26640          * Fires before a click is processed. Returns false to cancel the default action.
26641          * @param {Roo.View} this
26642          * @param {Number} index The index of the target node
26643          * @param {HTMLElement} node The target node
26644          * @param {Roo.EventObject} e The raw event object
26645          */
26646             "beforeclick" : true,
26647         /**
26648          * @event click
26649          * Fires when a template node is clicked.
26650          * @param {Roo.View} this
26651          * @param {Number} index The index of the target node
26652          * @param {HTMLElement} node The target node
26653          * @param {Roo.EventObject} e The raw event object
26654          */
26655             "click" : true,
26656         /**
26657          * @event dblclick
26658          * Fires when a template node is double clicked.
26659          * @param {Roo.View} this
26660          * @param {Number} index The index of the target node
26661          * @param {HTMLElement} node The target node
26662          * @param {Roo.EventObject} e The raw event object
26663          */
26664             "dblclick" : true,
26665         /**
26666          * @event contextmenu
26667          * Fires when a template node is right clicked.
26668          * @param {Roo.View} this
26669          * @param {Number} index The index of the target node
26670          * @param {HTMLElement} node The target node
26671          * @param {Roo.EventObject} e The raw event object
26672          */
26673             "contextmenu" : true,
26674         /**
26675          * @event selectionchange
26676          * Fires when the selected nodes change.
26677          * @param {Roo.View} this
26678          * @param {Array} selections Array of the selected nodes
26679          */
26680             "selectionchange" : true,
26681     
26682         /**
26683          * @event beforeselect
26684          * Fires before a selection is made. If any handlers return false, the selection is cancelled.
26685          * @param {Roo.View} this
26686          * @param {HTMLElement} node The node to be selected
26687          * @param {Array} selections Array of currently selected nodes
26688          */
26689             "beforeselect" : true,
26690         /**
26691          * @event preparedata
26692          * Fires on every row to render, to allow you to change the data.
26693          * @param {Roo.View} this
26694          * @param {Object} data to be rendered (change this)
26695          */
26696           "preparedata" : true
26697           
26698           
26699         });
26700
26701
26702
26703     this.el.on({
26704         "click": this.onClick,
26705         "dblclick": this.onDblClick,
26706         "contextmenu": this.onContextMenu,
26707         scope:this
26708     });
26709
26710     this.selections = [];
26711     this.nodes = [];
26712     this.cmp = new Roo.CompositeElementLite([]);
26713     if(this.store){
26714         this.store = Roo.factory(this.store, Roo.data);
26715         this.setStore(this.store, true);
26716     }
26717     
26718     if ( this.footer && this.footer.xtype) {
26719            
26720          var fctr = this.wrapEl.appendChild(document.createElement("div"));
26721         
26722         this.footer.dataSource = this.store;
26723         this.footer.container = fctr;
26724         this.footer = Roo.factory(this.footer, Roo);
26725         fctr.insertFirst(this.el);
26726         
26727         // this is a bit insane - as the paging toolbar seems to detach the el..
26728 //        dom.parentNode.parentNode.parentNode
26729          // they get detached?
26730     }
26731     
26732     
26733     Roo.View.superclass.constructor.call(this);
26734     
26735     
26736 };
26737
26738 Roo.extend(Roo.View, Roo.util.Observable, {
26739     
26740      /**
26741      * @cfg {Roo.data.Store} store Data store to load data from.
26742      */
26743     store : false,
26744     
26745     /**
26746      * @cfg {String|Roo.Element} el The container element.
26747      */
26748     el : '',
26749     
26750     /**
26751      * @cfg {String|Roo.Template} tpl The template used by this View 
26752      */
26753     tpl : false,
26754     /**
26755      * @cfg {String} dataName the named area of the template to use as the data area
26756      *                          Works with domtemplates roo-name="name"
26757      */
26758     dataName: false,
26759     /**
26760      * @cfg {String} selectedClass The css class to add to selected nodes
26761      */
26762     selectedClass : "x-view-selected",
26763      /**
26764      * @cfg {String} emptyText The empty text to show when nothing is loaded.
26765      */
26766     emptyText : "",
26767     
26768     /**
26769      * @cfg {String} text to display on mask (default Loading)
26770      */
26771     mask : false,
26772     /**
26773      * @cfg {Boolean} multiSelect Allow multiple selection
26774      */
26775     multiSelect : false,
26776     /**
26777      * @cfg {Boolean} singleSelect Allow single selection
26778      */
26779     singleSelect:  false,
26780     
26781     /**
26782      * @cfg {Boolean} toggleSelect - selecting 
26783      */
26784     toggleSelect : false,
26785     
26786     /**
26787      * @cfg {Boolean} tickable - selecting 
26788      */
26789     tickable : false,
26790     
26791     /**
26792      * Returns the element this view is bound to.
26793      * @return {Roo.Element}
26794      */
26795     getEl : function(){
26796         return this.wrapEl;
26797     },
26798     
26799     
26800
26801     /**
26802      * Refreshes the view. - called by datachanged on the store. - do not call directly.
26803      */
26804     refresh : function(){
26805         //Roo.log('refresh');
26806         var t = this.tpl;
26807         
26808         // if we are using something like 'domtemplate', then
26809         // the what gets used is:
26810         // t.applySubtemplate(NAME, data, wrapping data..)
26811         // the outer template then get' applied with
26812         //     the store 'extra data'
26813         // and the body get's added to the
26814         //      roo-name="data" node?
26815         //      <span class='roo-tpl-{name}'></span> ?????
26816         
26817         
26818         
26819         this.clearSelections();
26820         this.el.update("");
26821         var html = [];
26822         var records = this.store.getRange();
26823         if(records.length < 1) {
26824             
26825             // is this valid??  = should it render a template??
26826             
26827             this.el.update(this.emptyText);
26828             return;
26829         }
26830         var el = this.el;
26831         if (this.dataName) {
26832             this.el.update(t.apply(this.store.meta)); //????
26833             el = this.el.child('.roo-tpl-' + this.dataName);
26834         }
26835         
26836         for(var i = 0, len = records.length; i < len; i++){
26837             var data = this.prepareData(records[i].data, i, records[i]);
26838             this.fireEvent("preparedata", this, data, i, records[i]);
26839             
26840             var d = Roo.apply({}, data);
26841             
26842             if(this.tickable){
26843                 Roo.apply(d, {'roo-id' : Roo.id()});
26844                 
26845                 var _this = this;
26846             
26847                 Roo.each(this.parent.item, function(item){
26848                     if(item[_this.parent.valueField] != data[_this.parent.valueField]){
26849                         return;
26850                     }
26851                     Roo.apply(d, {'roo-data-checked' : 'checked'});
26852                 });
26853             }
26854             
26855             html[html.length] = Roo.util.Format.trim(
26856                 this.dataName ?
26857                     t.applySubtemplate(this.dataName, d, this.store.meta) :
26858                     t.apply(d)
26859             );
26860         }
26861         
26862         
26863         
26864         el.update(html.join(""));
26865         this.nodes = el.dom.childNodes;
26866         this.updateIndexes(0);
26867     },
26868     
26869
26870     /**
26871      * Function to override to reformat the data that is sent to
26872      * the template for each node.
26873      * DEPRICATED - use the preparedata event handler.
26874      * @param {Array/Object} data The raw data (array of colData for a data model bound view or
26875      * a JSON object for an UpdateManager bound view).
26876      */
26877     prepareData : function(data, index, record)
26878     {
26879         this.fireEvent("preparedata", this, data, index, record);
26880         return data;
26881     },
26882
26883     onUpdate : function(ds, record){
26884         // Roo.log('on update');   
26885         this.clearSelections();
26886         var index = this.store.indexOf(record);
26887         var n = this.nodes[index];
26888         this.tpl.insertBefore(n, this.prepareData(record.data, index, record));
26889         n.parentNode.removeChild(n);
26890         this.updateIndexes(index, index);
26891     },
26892
26893     
26894     
26895 // --------- FIXME     
26896     onAdd : function(ds, records, index)
26897     {
26898         //Roo.log(['on Add', ds, records, index] );        
26899         this.clearSelections();
26900         if(this.nodes.length == 0){
26901             this.refresh();
26902             return;
26903         }
26904         var n = this.nodes[index];
26905         for(var i = 0, len = records.length; i < len; i++){
26906             var d = this.prepareData(records[i].data, i, records[i]);
26907             if(n){
26908                 this.tpl.insertBefore(n, d);
26909             }else{
26910                 
26911                 this.tpl.append(this.el, d);
26912             }
26913         }
26914         this.updateIndexes(index);
26915     },
26916
26917     onRemove : function(ds, record, index){
26918        // Roo.log('onRemove');
26919         this.clearSelections();
26920         var el = this.dataName  ?
26921             this.el.child('.roo-tpl-' + this.dataName) :
26922             this.el; 
26923         
26924         el.dom.removeChild(this.nodes[index]);
26925         this.updateIndexes(index);
26926     },
26927
26928     /**
26929      * Refresh an individual node.
26930      * @param {Number} index
26931      */
26932     refreshNode : function(index){
26933         this.onUpdate(this.store, this.store.getAt(index));
26934     },
26935
26936     updateIndexes : function(startIndex, endIndex){
26937         var ns = this.nodes;
26938         startIndex = startIndex || 0;
26939         endIndex = endIndex || ns.length - 1;
26940         for(var i = startIndex; i <= endIndex; i++){
26941             ns[i].nodeIndex = i;
26942         }
26943     },
26944
26945     /**
26946      * Changes the data store this view uses and refresh the view.
26947      * @param {Store} store
26948      */
26949     setStore : function(store, initial){
26950         if(!initial && this.store){
26951             this.store.un("datachanged", this.refresh);
26952             this.store.un("add", this.onAdd);
26953             this.store.un("remove", this.onRemove);
26954             this.store.un("update", this.onUpdate);
26955             this.store.un("clear", this.refresh);
26956             this.store.un("beforeload", this.onBeforeLoad);
26957             this.store.un("load", this.onLoad);
26958             this.store.un("loadexception", this.onLoad);
26959         }
26960         if(store){
26961           
26962             store.on("datachanged", this.refresh, this);
26963             store.on("add", this.onAdd, this);
26964             store.on("remove", this.onRemove, this);
26965             store.on("update", this.onUpdate, this);
26966             store.on("clear", this.refresh, this);
26967             store.on("beforeload", this.onBeforeLoad, this);
26968             store.on("load", this.onLoad, this);
26969             store.on("loadexception", this.onLoad, this);
26970         }
26971         
26972         if(store){
26973             this.refresh();
26974         }
26975     },
26976     /**
26977      * onbeforeLoad - masks the loading area.
26978      *
26979      */
26980     onBeforeLoad : function(store,opts)
26981     {
26982          //Roo.log('onBeforeLoad');   
26983         if (!opts.add) {
26984             this.el.update("");
26985         }
26986         this.el.mask(this.mask ? this.mask : "Loading" ); 
26987     },
26988     onLoad : function ()
26989     {
26990         this.el.unmask();
26991     },
26992     
26993
26994     /**
26995      * Returns the template node the passed child belongs to or null if it doesn't belong to one.
26996      * @param {HTMLElement} node
26997      * @return {HTMLElement} The template node
26998      */
26999     findItemFromChild : function(node){
27000         var el = this.dataName  ?
27001             this.el.child('.roo-tpl-' + this.dataName,true) :
27002             this.el.dom; 
27003         
27004         if(!node || node.parentNode == el){
27005                     return node;
27006             }
27007             var p = node.parentNode;
27008             while(p && p != el){
27009             if(p.parentNode == el){
27010                 return p;
27011             }
27012             p = p.parentNode;
27013         }
27014             return null;
27015     },
27016
27017     /** @ignore */
27018     onClick : function(e){
27019         var item = this.findItemFromChild(e.getTarget());
27020         if(item){
27021             var index = this.indexOf(item);
27022             if(this.onItemClick(item, index, e) !== false){
27023                 this.fireEvent("click", this, index, item, e);
27024             }
27025         }else{
27026             this.clearSelections();
27027         }
27028     },
27029
27030     /** @ignore */
27031     onContextMenu : function(e){
27032         var item = this.findItemFromChild(e.getTarget());
27033         if(item){
27034             this.fireEvent("contextmenu", this, this.indexOf(item), item, e);
27035         }
27036     },
27037
27038     /** @ignore */
27039     onDblClick : function(e){
27040         var item = this.findItemFromChild(e.getTarget());
27041         if(item){
27042             this.fireEvent("dblclick", this, this.indexOf(item), item, e);
27043         }
27044     },
27045
27046     onItemClick : function(item, index, e)
27047     {
27048         if(this.fireEvent("beforeclick", this, index, item, e) === false){
27049             return false;
27050         }
27051         if (this.toggleSelect) {
27052             var m = this.isSelected(item) ? 'unselect' : 'select';
27053             //Roo.log(m);
27054             var _t = this;
27055             _t[m](item, true, false);
27056             return true;
27057         }
27058         if(this.multiSelect || this.singleSelect){
27059             if(this.multiSelect && e.shiftKey && this.lastSelection){
27060                 this.select(this.getNodes(this.indexOf(this.lastSelection), index), false);
27061             }else{
27062                 this.select(item, this.multiSelect && e.ctrlKey);
27063                 this.lastSelection = item;
27064             }
27065             
27066             if(!this.tickable){
27067                 e.preventDefault();
27068             }
27069             
27070         }
27071         return true;
27072     },
27073
27074     /**
27075      * Get the number of selected nodes.
27076      * @return {Number}
27077      */
27078     getSelectionCount : function(){
27079         return this.selections.length;
27080     },
27081
27082     /**
27083      * Get the currently selected nodes.
27084      * @return {Array} An array of HTMLElements
27085      */
27086     getSelectedNodes : function(){
27087         return this.selections;
27088     },
27089
27090     /**
27091      * Get the indexes of the selected nodes.
27092      * @return {Array}
27093      */
27094     getSelectedIndexes : function(){
27095         var indexes = [], s = this.selections;
27096         for(var i = 0, len = s.length; i < len; i++){
27097             indexes.push(s[i].nodeIndex);
27098         }
27099         return indexes;
27100     },
27101
27102     /**
27103      * Clear all selections
27104      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange event
27105      */
27106     clearSelections : function(suppressEvent){
27107         if(this.nodes && (this.multiSelect || this.singleSelect) && this.selections.length > 0){
27108             this.cmp.elements = this.selections;
27109             this.cmp.removeClass(this.selectedClass);
27110             this.selections = [];
27111             if(!suppressEvent){
27112                 this.fireEvent("selectionchange", this, this.selections);
27113             }
27114         }
27115     },
27116
27117     /**
27118      * Returns true if the passed node is selected
27119      * @param {HTMLElement/Number} node The node or node index
27120      * @return {Boolean}
27121      */
27122     isSelected : function(node){
27123         var s = this.selections;
27124         if(s.length < 1){
27125             return false;
27126         }
27127         node = this.getNode(node);
27128         return s.indexOf(node) !== -1;
27129     },
27130
27131     /**
27132      * Selects nodes.
27133      * @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
27134      * @param {Boolean} keepExisting (optional) true to keep existing selections
27135      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27136      */
27137     select : function(nodeInfo, keepExisting, suppressEvent){
27138         if(nodeInfo instanceof Array){
27139             if(!keepExisting){
27140                 this.clearSelections(true);
27141             }
27142             for(var i = 0, len = nodeInfo.length; i < len; i++){
27143                 this.select(nodeInfo[i], true, true);
27144             }
27145             return;
27146         } 
27147         var node = this.getNode(nodeInfo);
27148         if(!node || this.isSelected(node)){
27149             return; // already selected.
27150         }
27151         if(!keepExisting){
27152             this.clearSelections(true);
27153         }
27154         
27155         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
27156             Roo.fly(node).addClass(this.selectedClass);
27157             this.selections.push(node);
27158             if(!suppressEvent){
27159                 this.fireEvent("selectionchange", this, this.selections);
27160             }
27161         }
27162         
27163         
27164     },
27165       /**
27166      * Unselects nodes.
27167      * @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
27168      * @param {Boolean} keepExisting (optional) true IGNORED (for campatibility with select)
27169      * @param {Boolean} suppressEvent (optional) true to skip firing of the selectionchange vent
27170      */
27171     unselect : function(nodeInfo, keepExisting, suppressEvent)
27172     {
27173         if(nodeInfo instanceof Array){
27174             Roo.each(this.selections, function(s) {
27175                 this.unselect(s, nodeInfo);
27176             }, this);
27177             return;
27178         }
27179         var node = this.getNode(nodeInfo);
27180         if(!node || !this.isSelected(node)){
27181             //Roo.log("not selected");
27182             return; // not selected.
27183         }
27184         // fireevent???
27185         var ns = [];
27186         Roo.each(this.selections, function(s) {
27187             if (s == node ) {
27188                 Roo.fly(node).removeClass(this.selectedClass);
27189
27190                 return;
27191             }
27192             ns.push(s);
27193         },this);
27194         
27195         this.selections= ns;
27196         this.fireEvent("selectionchange", this, this.selections);
27197     },
27198
27199     /**
27200      * Gets a template node.
27201      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27202      * @return {HTMLElement} The node or null if it wasn't found
27203      */
27204     getNode : function(nodeInfo){
27205         if(typeof nodeInfo == "string"){
27206             return document.getElementById(nodeInfo);
27207         }else if(typeof nodeInfo == "number"){
27208             return this.nodes[nodeInfo];
27209         }
27210         return nodeInfo;
27211     },
27212
27213     /**
27214      * Gets a range template nodes.
27215      * @param {Number} startIndex
27216      * @param {Number} endIndex
27217      * @return {Array} An array of nodes
27218      */
27219     getNodes : function(start, end){
27220         var ns = this.nodes;
27221         start = start || 0;
27222         end = typeof end == "undefined" ? ns.length - 1 : end;
27223         var nodes = [];
27224         if(start <= end){
27225             for(var i = start; i <= end; i++){
27226                 nodes.push(ns[i]);
27227             }
27228         } else{
27229             for(var i = start; i >= end; i--){
27230                 nodes.push(ns[i]);
27231             }
27232         }
27233         return nodes;
27234     },
27235
27236     /**
27237      * Finds the index of the passed node
27238      * @param {HTMLElement/String/Number} nodeInfo An HTMLElement template node, index of a template node or the id of a template node
27239      * @return {Number} The index of the node or -1
27240      */
27241     indexOf : function(node){
27242         node = this.getNode(node);
27243         if(typeof node.nodeIndex == "number"){
27244             return node.nodeIndex;
27245         }
27246         var ns = this.nodes;
27247         for(var i = 0, len = ns.length; i < len; i++){
27248             if(ns[i] == node){
27249                 return i;
27250             }
27251         }
27252         return -1;
27253     }
27254 });
27255 /*
27256  * Based on:
27257  * Ext JS Library 1.1.1
27258  * Copyright(c) 2006-2007, Ext JS, LLC.
27259  *
27260  * Originally Released Under LGPL - original licence link has changed is not relivant.
27261  *
27262  * Fork - LGPL
27263  * <script type="text/javascript">
27264  */
27265
27266 /**
27267  * @class Roo.JsonView
27268  * @extends Roo.View
27269  * Shortcut class to create a JSON + {@link Roo.UpdateManager} template view. Usage:
27270 <pre><code>
27271 var view = new Roo.JsonView({
27272     container: "my-element",
27273     tpl: '&lt;div id="{id}"&gt;{foo} - {bar}&lt;/div&gt;', // auto create template
27274     multiSelect: true, 
27275     jsonRoot: "data" 
27276 });
27277
27278 // listen for node click?
27279 view.on("click", function(vw, index, node, e){
27280     alert('Node "' + node.id + '" at index: ' + index + " was clicked.");
27281 });
27282
27283 // direct load of JSON data
27284 view.load("foobar.php");
27285
27286 // Example from my blog list
27287 var tpl = new Roo.Template(
27288     '&lt;div class="entry"&gt;' +
27289     '&lt;a class="entry-title" href="{link}"&gt;{title}&lt;/a&gt;' +
27290     "&lt;h4&gt;{date} by {author} | {comments} Comments&lt;/h4&gt;{description}" +
27291     "&lt;/div&gt;&lt;hr /&gt;"
27292 );
27293
27294 var moreView = new Roo.JsonView({
27295     container :  "entry-list", 
27296     template : tpl,
27297     jsonRoot: "posts"
27298 });
27299 moreView.on("beforerender", this.sortEntries, this);
27300 moreView.load({
27301     url: "/blog/get-posts.php",
27302     params: "allposts=true",
27303     text: "Loading Blog Entries..."
27304 });
27305 </code></pre>
27306
27307 * Note: old code is supported with arguments : (container, template, config)
27308
27309
27310  * @constructor
27311  * Create a new JsonView
27312  * 
27313  * @param {Object} config The config object
27314  * 
27315  */
27316 Roo.JsonView = function(config, depreciated_tpl, depreciated_config){
27317     
27318     
27319     Roo.JsonView.superclass.constructor.call(this, config, depreciated_tpl, depreciated_config);
27320
27321     var um = this.el.getUpdateManager();
27322     um.setRenderer(this);
27323     um.on("update", this.onLoad, this);
27324     um.on("failure", this.onLoadException, this);
27325
27326     /**
27327      * @event beforerender
27328      * Fires before rendering of the downloaded JSON data.
27329      * @param {Roo.JsonView} this
27330      * @param {Object} data The JSON data loaded
27331      */
27332     /**
27333      * @event load
27334      * Fires when data is loaded.
27335      * @param {Roo.JsonView} this
27336      * @param {Object} data The JSON data loaded
27337      * @param {Object} response The raw Connect response object
27338      */
27339     /**
27340      * @event loadexception
27341      * Fires when loading fails.
27342      * @param {Roo.JsonView} this
27343      * @param {Object} response The raw Connect response object
27344      */
27345     this.addEvents({
27346         'beforerender' : true,
27347         'load' : true,
27348         'loadexception' : true
27349     });
27350 };
27351 Roo.extend(Roo.JsonView, Roo.View, {
27352     /**
27353      * @type {String} The root property in the loaded JSON object that contains the data
27354      */
27355     jsonRoot : "",
27356
27357     /**
27358      * Refreshes the view.
27359      */
27360     refresh : function(){
27361         this.clearSelections();
27362         this.el.update("");
27363         var html = [];
27364         var o = this.jsonData;
27365         if(o && o.length > 0){
27366             for(var i = 0, len = o.length; i < len; i++){
27367                 var data = this.prepareData(o[i], i, o);
27368                 html[html.length] = this.tpl.apply(data);
27369             }
27370         }else{
27371             html.push(this.emptyText);
27372         }
27373         this.el.update(html.join(""));
27374         this.nodes = this.el.dom.childNodes;
27375         this.updateIndexes(0);
27376     },
27377
27378     /**
27379      * 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.
27380      * @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:
27381      <pre><code>
27382      view.load({
27383          url: "your-url.php",
27384          params: {param1: "foo", param2: "bar"}, // or a URL encoded string
27385          callback: yourFunction,
27386          scope: yourObject, //(optional scope)
27387          discardUrl: false,
27388          nocache: false,
27389          text: "Loading...",
27390          timeout: 30,
27391          scripts: false
27392      });
27393      </code></pre>
27394      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
27395      * 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.
27396      * @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}
27397      * @param {Function} callback (optional) Callback when transaction is complete - called with signature (oElement, bSuccess)
27398      * @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.
27399      */
27400     load : function(){
27401         var um = this.el.getUpdateManager();
27402         um.update.apply(um, arguments);
27403     },
27404
27405     // note - render is a standard framework call...
27406     // using it for the response is really flaky... - it's called by UpdateManager normally, except when called by the XComponent/addXtype.
27407     render : function(el, response){
27408         
27409         this.clearSelections();
27410         this.el.update("");
27411         var o;
27412         try{
27413             if (response != '') {
27414                 o = Roo.util.JSON.decode(response.responseText);
27415                 if(this.jsonRoot){
27416                     
27417                     o = o[this.jsonRoot];
27418                 }
27419             }
27420         } catch(e){
27421         }
27422         /**
27423          * The current JSON data or null
27424          */
27425         this.jsonData = o;
27426         this.beforeRender();
27427         this.refresh();
27428     },
27429
27430 /**
27431  * Get the number of records in the current JSON dataset
27432  * @return {Number}
27433  */
27434     getCount : function(){
27435         return this.jsonData ? this.jsonData.length : 0;
27436     },
27437
27438 /**
27439  * Returns the JSON object for the specified node(s)
27440  * @param {HTMLElement/Array} node The node or an array of nodes
27441  * @return {Object/Array} If you pass in an array, you get an array back, otherwise
27442  * you get the JSON object for the node
27443  */
27444     getNodeData : function(node){
27445         if(node instanceof Array){
27446             var data = [];
27447             for(var i = 0, len = node.length; i < len; i++){
27448                 data.push(this.getNodeData(node[i]));
27449             }
27450             return data;
27451         }
27452         return this.jsonData[this.indexOf(node)] || null;
27453     },
27454
27455     beforeRender : function(){
27456         this.snapshot = this.jsonData;
27457         if(this.sortInfo){
27458             this.sort.apply(this, this.sortInfo);
27459         }
27460         this.fireEvent("beforerender", this, this.jsonData);
27461     },
27462
27463     onLoad : function(el, o){
27464         this.fireEvent("load", this, this.jsonData, o);
27465     },
27466
27467     onLoadException : function(el, o){
27468         this.fireEvent("loadexception", this, o);
27469     },
27470
27471 /**
27472  * Filter the data by a specific property.
27473  * @param {String} property A property on your JSON objects
27474  * @param {String/RegExp} value Either string that the property values
27475  * should start with, or a RegExp to test against the property
27476  */
27477     filter : function(property, value){
27478         if(this.jsonData){
27479             var data = [];
27480             var ss = this.snapshot;
27481             if(typeof value == "string"){
27482                 var vlen = value.length;
27483                 if(vlen == 0){
27484                     this.clearFilter();
27485                     return;
27486                 }
27487                 value = value.toLowerCase();
27488                 for(var i = 0, len = ss.length; i < len; i++){
27489                     var o = ss[i];
27490                     if(o[property].substr(0, vlen).toLowerCase() == value){
27491                         data.push(o);
27492                     }
27493                 }
27494             } else if(value.exec){ // regex?
27495                 for(var i = 0, len = ss.length; i < len; i++){
27496                     var o = ss[i];
27497                     if(value.test(o[property])){
27498                         data.push(o);
27499                     }
27500                 }
27501             } else{
27502                 return;
27503             }
27504             this.jsonData = data;
27505             this.refresh();
27506         }
27507     },
27508
27509 /**
27510  * Filter by a function. The passed function will be called with each
27511  * object in the current dataset. If the function returns true the value is kept,
27512  * otherwise it is filtered.
27513  * @param {Function} fn
27514  * @param {Object} scope (optional) The scope of the function (defaults to this JsonView)
27515  */
27516     filterBy : function(fn, scope){
27517         if(this.jsonData){
27518             var data = [];
27519             var ss = this.snapshot;
27520             for(var i = 0, len = ss.length; i < len; i++){
27521                 var o = ss[i];
27522                 if(fn.call(scope || this, o)){
27523                     data.push(o);
27524                 }
27525             }
27526             this.jsonData = data;
27527             this.refresh();
27528         }
27529     },
27530
27531 /**
27532  * Clears the current filter.
27533  */
27534     clearFilter : function(){
27535         if(this.snapshot && this.jsonData != this.snapshot){
27536             this.jsonData = this.snapshot;
27537             this.refresh();
27538         }
27539     },
27540
27541
27542 /**
27543  * Sorts the data for this view and refreshes it.
27544  * @param {String} property A property on your JSON objects to sort on
27545  * @param {String} direction (optional) "desc" or "asc" (defaults to "asc")
27546  * @param {Function} sortType (optional) A function to call to convert the data to a sortable value.
27547  */
27548     sort : function(property, dir, sortType){
27549         this.sortInfo = Array.prototype.slice.call(arguments, 0);
27550         if(this.jsonData){
27551             var p = property;
27552             var dsc = dir && dir.toLowerCase() == "desc";
27553             var f = function(o1, o2){
27554                 var v1 = sortType ? sortType(o1[p]) : o1[p];
27555                 var v2 = sortType ? sortType(o2[p]) : o2[p];
27556                 ;
27557                 if(v1 < v2){
27558                     return dsc ? +1 : -1;
27559                 } else if(v1 > v2){
27560                     return dsc ? -1 : +1;
27561                 } else{
27562                     return 0;
27563                 }
27564             };
27565             this.jsonData.sort(f);
27566             this.refresh();
27567             if(this.jsonData != this.snapshot){
27568                 this.snapshot.sort(f);
27569             }
27570         }
27571     }
27572 });/*
27573  * Based on:
27574  * Ext JS Library 1.1.1
27575  * Copyright(c) 2006-2007, Ext JS, LLC.
27576  *
27577  * Originally Released Under LGPL - original licence link has changed is not relivant.
27578  *
27579  * Fork - LGPL
27580  * <script type="text/javascript">
27581  */
27582  
27583
27584 /**
27585  * @class Roo.ColorPalette
27586  * @extends Roo.Component
27587  * Simple color palette class for choosing colors.  The palette can be rendered to any container.<br />
27588  * Here's an example of typical usage:
27589  * <pre><code>
27590 var cp = new Roo.ColorPalette({value:'993300'});  // initial selected color
27591 cp.render('my-div');
27592
27593 cp.on('select', function(palette, selColor){
27594     // do something with selColor
27595 });
27596 </code></pre>
27597  * @constructor
27598  * Create a new ColorPalette
27599  * @param {Object} config The config object
27600  */
27601 Roo.ColorPalette = function(config){
27602     Roo.ColorPalette.superclass.constructor.call(this, config);
27603     this.addEvents({
27604         /**
27605              * @event select
27606              * Fires when a color is selected
27607              * @param {ColorPalette} this
27608              * @param {String} color The 6-digit color hex code (without the # symbol)
27609              */
27610         select: true
27611     });
27612
27613     if(this.handler){
27614         this.on("select", this.handler, this.scope, true);
27615     }
27616 };
27617 Roo.extend(Roo.ColorPalette, Roo.Component, {
27618     /**
27619      * @cfg {String} itemCls
27620      * The CSS class to apply to the containing element (defaults to "x-color-palette")
27621      */
27622     itemCls : "x-color-palette",
27623     /**
27624      * @cfg {String} value
27625      * The initial color to highlight (should be a valid 6-digit color hex code without the # symbol).  Note that
27626      * the hex codes are case-sensitive.
27627      */
27628     value : null,
27629     clickEvent:'click',
27630     // private
27631     ctype: "Roo.ColorPalette",
27632
27633     /**
27634      * @cfg {Boolean} allowReselect If set to true then reselecting a color that is already selected fires the selection event
27635      */
27636     allowReselect : false,
27637
27638     /**
27639      * <p>An array of 6-digit color hex code strings (without the # symbol).  This array can contain any number
27640      * of colors, and each hex code should be unique.  The width of the palette is controlled via CSS by adjusting
27641      * the width property of the 'x-color-palette' class (or assigning a custom class), so you can balance the number
27642      * of colors with the width setting until the box is symmetrical.</p>
27643      * <p>You can override individual colors if needed:</p>
27644      * <pre><code>
27645 var cp = new Roo.ColorPalette();
27646 cp.colors[0] = "FF0000";  // change the first box to red
27647 </code></pre>
27648
27649 Or you can provide a custom array of your own for complete control:
27650 <pre><code>
27651 var cp = new Roo.ColorPalette();
27652 cp.colors = ["000000", "993300", "333300"];
27653 </code></pre>
27654      * @type Array
27655      */
27656     colors : [
27657         "000000", "993300", "333300", "003300", "003366", "000080", "333399", "333333",
27658         "800000", "FF6600", "808000", "008000", "008080", "0000FF", "666699", "808080",
27659         "FF0000", "FF9900", "99CC00", "339966", "33CCCC", "3366FF", "800080", "969696",
27660         "FF00FF", "FFCC00", "FFFF00", "00FF00", "00FFFF", "00CCFF", "993366", "C0C0C0",
27661         "FF99CC", "FFCC99", "FFFF99", "CCFFCC", "CCFFFF", "99CCFF", "CC99FF", "FFFFFF"
27662     ],
27663
27664     // private
27665     onRender : function(container, position){
27666         var t = new Roo.MasterTemplate(
27667             '<tpl><a href="#" class="color-{0}" hidefocus="on"><em><span style="background:#{0}" unselectable="on">&#160;</span></em></a></tpl>'
27668         );
27669         var c = this.colors;
27670         for(var i = 0, len = c.length; i < len; i++){
27671             t.add([c[i]]);
27672         }
27673         var el = document.createElement("div");
27674         el.className = this.itemCls;
27675         t.overwrite(el);
27676         container.dom.insertBefore(el, position);
27677         this.el = Roo.get(el);
27678         this.el.on(this.clickEvent, this.handleClick,  this, {delegate: "a"});
27679         if(this.clickEvent != 'click'){
27680             this.el.on('click', Roo.emptyFn,  this, {delegate: "a", preventDefault:true});
27681         }
27682     },
27683
27684     // private
27685     afterRender : function(){
27686         Roo.ColorPalette.superclass.afterRender.call(this);
27687         if(this.value){
27688             var s = this.value;
27689             this.value = null;
27690             this.select(s);
27691         }
27692     },
27693
27694     // private
27695     handleClick : function(e, t){
27696         e.preventDefault();
27697         if(!this.disabled){
27698             var c = t.className.match(/(?:^|\s)color-(.{6})(?:\s|$)/)[1];
27699             this.select(c.toUpperCase());
27700         }
27701     },
27702
27703     /**
27704      * Selects the specified color in the palette (fires the select event)
27705      * @param {String} color A valid 6-digit color hex code (# will be stripped if included)
27706      */
27707     select : function(color){
27708         color = color.replace("#", "");
27709         if(color != this.value || this.allowReselect){
27710             var el = this.el;
27711             if(this.value){
27712                 el.child("a.color-"+this.value).removeClass("x-color-palette-sel");
27713             }
27714             el.child("a.color-"+color).addClass("x-color-palette-sel");
27715             this.value = color;
27716             this.fireEvent("select", this, color);
27717         }
27718     }
27719 });/*
27720  * Based on:
27721  * Ext JS Library 1.1.1
27722  * Copyright(c) 2006-2007, Ext JS, LLC.
27723  *
27724  * Originally Released Under LGPL - original licence link has changed is not relivant.
27725  *
27726  * Fork - LGPL
27727  * <script type="text/javascript">
27728  */
27729  
27730 /**
27731  * @class Roo.DatePicker
27732  * @extends Roo.Component
27733  * Simple date picker class.
27734  * @constructor
27735  * Create a new DatePicker
27736  * @param {Object} config The config object
27737  */
27738 Roo.DatePicker = function(config){
27739     Roo.DatePicker.superclass.constructor.call(this, config);
27740
27741     this.value = config && config.value ?
27742                  config.value.clearTime() : new Date().clearTime();
27743
27744     this.addEvents({
27745         /**
27746              * @event select
27747              * Fires when a date is selected
27748              * @param {DatePicker} this
27749              * @param {Date} date The selected date
27750              */
27751         'select': true,
27752         /**
27753              * @event monthchange
27754              * Fires when the displayed month changes 
27755              * @param {DatePicker} this
27756              * @param {Date} date The selected month
27757              */
27758         'monthchange': true
27759     });
27760
27761     if(this.handler){
27762         this.on("select", this.handler,  this.scope || this);
27763     }
27764     // build the disabledDatesRE
27765     if(!this.disabledDatesRE && this.disabledDates){
27766         var dd = this.disabledDates;
27767         var re = "(?:";
27768         for(var i = 0; i < dd.length; i++){
27769             re += dd[i];
27770             if(i != dd.length-1) {
27771                 re += "|";
27772             }
27773         }
27774         this.disabledDatesRE = new RegExp(re + ")");
27775     }
27776 };
27777
27778 Roo.extend(Roo.DatePicker, Roo.Component, {
27779     /**
27780      * @cfg {String} todayText
27781      * The text to display on the button that selects the current date (defaults to "Today")
27782      */
27783     todayText : "Today",
27784     /**
27785      * @cfg {String} okText
27786      * The text to display on the ok button
27787      */
27788     okText : "&#160;OK&#160;", // &#160; to give the user extra clicking room
27789     /**
27790      * @cfg {String} cancelText
27791      * The text to display on the cancel button
27792      */
27793     cancelText : "Cancel",
27794     /**
27795      * @cfg {String} todayTip
27796      * The tooltip to display for the button that selects the current date (defaults to "{current date} (Spacebar)")
27797      */
27798     todayTip : "{0} (Spacebar)",
27799     /**
27800      * @cfg {Date} minDate
27801      * Minimum allowable date (JavaScript date object, defaults to null)
27802      */
27803     minDate : null,
27804     /**
27805      * @cfg {Date} maxDate
27806      * Maximum allowable date (JavaScript date object, defaults to null)
27807      */
27808     maxDate : null,
27809     /**
27810      * @cfg {String} minText
27811      * The error text to display if the minDate validation fails (defaults to "This date is before the minimum date")
27812      */
27813     minText : "This date is before the minimum date",
27814     /**
27815      * @cfg {String} maxText
27816      * The error text to display if the maxDate validation fails (defaults to "This date is after the maximum date")
27817      */
27818     maxText : "This date is after the maximum date",
27819     /**
27820      * @cfg {String} format
27821      * The default date format string which can be overriden for localization support.  The format must be
27822      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
27823      */
27824     format : "m/d/y",
27825     /**
27826      * @cfg {Array} disabledDays
27827      * An array of days to disable, 0-based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
27828      */
27829     disabledDays : null,
27830     /**
27831      * @cfg {String} disabledDaysText
27832      * The tooltip to display when the date falls on a disabled day (defaults to "")
27833      */
27834     disabledDaysText : "",
27835     /**
27836      * @cfg {RegExp} disabledDatesRE
27837      * JavaScript regular expression used to disable a pattern of dates (defaults to null)
27838      */
27839     disabledDatesRE : null,
27840     /**
27841      * @cfg {String} disabledDatesText
27842      * The tooltip text to display when the date falls on a disabled date (defaults to "")
27843      */
27844     disabledDatesText : "",
27845     /**
27846      * @cfg {Boolean} constrainToViewport
27847      * True to constrain the date picker to the viewport (defaults to true)
27848      */
27849     constrainToViewport : true,
27850     /**
27851      * @cfg {Array} monthNames
27852      * An array of textual month names which can be overriden for localization support (defaults to Date.monthNames)
27853      */
27854     monthNames : Date.monthNames,
27855     /**
27856      * @cfg {Array} dayNames
27857      * An array of textual day names which can be overriden for localization support (defaults to Date.dayNames)
27858      */
27859     dayNames : Date.dayNames,
27860     /**
27861      * @cfg {String} nextText
27862      * The next month navigation button tooltip (defaults to 'Next Month (Control+Right)')
27863      */
27864     nextText: 'Next Month (Control+Right)',
27865     /**
27866      * @cfg {String} prevText
27867      * The previous month navigation button tooltip (defaults to 'Previous Month (Control+Left)')
27868      */
27869     prevText: 'Previous Month (Control+Left)',
27870     /**
27871      * @cfg {String} monthYearText
27872      * The header month selector tooltip (defaults to 'Choose a month (Control+Up/Down to move years)')
27873      */
27874     monthYearText: 'Choose a month (Control+Up/Down to move years)',
27875     /**
27876      * @cfg {Number} startDay
27877      * Day index at which the week should begin, 0-based (defaults to 0, which is Sunday)
27878      */
27879     startDay : 0,
27880     /**
27881      * @cfg {Bool} showClear
27882      * Show a clear button (usefull for date form elements that can be blank.)
27883      */
27884     
27885     showClear: false,
27886     
27887     /**
27888      * Sets the value of the date field
27889      * @param {Date} value The date to set
27890      */
27891     setValue : function(value){
27892         var old = this.value;
27893         
27894         if (typeof(value) == 'string') {
27895          
27896             value = Date.parseDate(value, this.format);
27897         }
27898         if (!value) {
27899             value = new Date();
27900         }
27901         
27902         this.value = value.clearTime(true);
27903         if(this.el){
27904             this.update(this.value);
27905         }
27906     },
27907
27908     /**
27909      * Gets the current selected value of the date field
27910      * @return {Date} The selected date
27911      */
27912     getValue : function(){
27913         return this.value;
27914     },
27915
27916     // private
27917     focus : function(){
27918         if(this.el){
27919             this.update(this.activeDate);
27920         }
27921     },
27922
27923     // privateval
27924     onRender : function(container, position){
27925         
27926         var m = [
27927              '<table cellspacing="0">',
27928                 '<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>',
27929                 '<tr><td colspan="3"><table class="x-date-inner" cellspacing="0"><thead><tr>'];
27930         var dn = this.dayNames;
27931         for(var i = 0; i < 7; i++){
27932             var d = this.startDay+i;
27933             if(d > 6){
27934                 d = d-7;
27935             }
27936             m.push("<th><span>", dn[d].substr(0,1), "</span></th>");
27937         }
27938         m[m.length] = "</tr></thead><tbody><tr>";
27939         for(var i = 0; i < 42; i++) {
27940             if(i % 7 == 0 && i != 0){
27941                 m[m.length] = "</tr><tr>";
27942             }
27943             m[m.length] = '<td><a href="#" hidefocus="on" class="x-date-date" tabIndex="1"><em><span></span></em></a></td>';
27944         }
27945         m[m.length] = '</tr></tbody></table></td></tr><tr>'+
27946             '<td colspan="3" class="x-date-bottom" align="center"></td></tr></table><div class="x-date-mp"></div>';
27947
27948         var el = document.createElement("div");
27949         el.className = "x-date-picker";
27950         el.innerHTML = m.join("");
27951
27952         container.dom.insertBefore(el, position);
27953
27954         this.el = Roo.get(el);
27955         this.eventEl = Roo.get(el.firstChild);
27956
27957         new Roo.util.ClickRepeater(this.el.child("td.x-date-left a"), {
27958             handler: this.showPrevMonth,
27959             scope: this,
27960             preventDefault:true,
27961             stopDefault:true
27962         });
27963
27964         new Roo.util.ClickRepeater(this.el.child("td.x-date-right a"), {
27965             handler: this.showNextMonth,
27966             scope: this,
27967             preventDefault:true,
27968             stopDefault:true
27969         });
27970
27971         this.eventEl.on("mousewheel", this.handleMouseWheel,  this);
27972
27973         this.monthPicker = this.el.down('div.x-date-mp');
27974         this.monthPicker.enableDisplayMode('block');
27975         
27976         var kn = new Roo.KeyNav(this.eventEl, {
27977             "left" : function(e){
27978                 e.ctrlKey ?
27979                     this.showPrevMonth() :
27980                     this.update(this.activeDate.add("d", -1));
27981             },
27982
27983             "right" : function(e){
27984                 e.ctrlKey ?
27985                     this.showNextMonth() :
27986                     this.update(this.activeDate.add("d", 1));
27987             },
27988
27989             "up" : function(e){
27990                 e.ctrlKey ?
27991                     this.showNextYear() :
27992                     this.update(this.activeDate.add("d", -7));
27993             },
27994
27995             "down" : function(e){
27996                 e.ctrlKey ?
27997                     this.showPrevYear() :
27998                     this.update(this.activeDate.add("d", 7));
27999             },
28000
28001             "pageUp" : function(e){
28002                 this.showNextMonth();
28003             },
28004
28005             "pageDown" : function(e){
28006                 this.showPrevMonth();
28007             },
28008
28009             "enter" : function(e){
28010                 e.stopPropagation();
28011                 return true;
28012             },
28013
28014             scope : this
28015         });
28016
28017         this.eventEl.on("click", this.handleDateClick,  this, {delegate: "a.x-date-date"});
28018
28019         this.eventEl.addKeyListener(Roo.EventObject.SPACE, this.selectToday,  this);
28020
28021         this.el.unselectable();
28022         
28023         this.cells = this.el.select("table.x-date-inner tbody td");
28024         this.textNodes = this.el.query("table.x-date-inner tbody span");
28025
28026         this.mbtn = new Roo.Button(this.el.child("td.x-date-middle", true), {
28027             text: "&#160;",
28028             tooltip: this.monthYearText
28029         });
28030
28031         this.mbtn.on('click', this.showMonthPicker, this);
28032         this.mbtn.el.child(this.mbtn.menuClassTarget).addClass("x-btn-with-menu");
28033
28034
28035         var today = (new Date()).dateFormat(this.format);
28036         
28037         var baseTb = new Roo.Toolbar(this.el.child("td.x-date-bottom", true));
28038         if (this.showClear) {
28039             baseTb.add( new Roo.Toolbar.Fill());
28040         }
28041         baseTb.add({
28042             text: String.format(this.todayText, today),
28043             tooltip: String.format(this.todayTip, today),
28044             handler: this.selectToday,
28045             scope: this
28046         });
28047         
28048         //var todayBtn = new Roo.Button(this.el.child("td.x-date-bottom", true), {
28049             
28050         //});
28051         if (this.showClear) {
28052             
28053             baseTb.add( new Roo.Toolbar.Fill());
28054             baseTb.add({
28055                 text: '&#160;',
28056                 cls: 'x-btn-icon x-btn-clear',
28057                 handler: function() {
28058                     //this.value = '';
28059                     this.fireEvent("select", this, '');
28060                 },
28061                 scope: this
28062             });
28063         }
28064         
28065         
28066         if(Roo.isIE){
28067             this.el.repaint();
28068         }
28069         this.update(this.value);
28070     },
28071
28072     createMonthPicker : function(){
28073         if(!this.monthPicker.dom.firstChild){
28074             var buf = ['<table border="0" cellspacing="0">'];
28075             for(var i = 0; i < 6; i++){
28076                 buf.push(
28077                     '<tr><td class="x-date-mp-month"><a href="#">', this.monthNames[i].substr(0, 3), '</a></td>',
28078                     '<td class="x-date-mp-month x-date-mp-sep"><a href="#">', this.monthNames[i+6].substr(0, 3), '</a></td>',
28079                     i == 0 ?
28080                     '<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>' :
28081                     '<td class="x-date-mp-year"><a href="#"></a></td><td class="x-date-mp-year"><a href="#"></a></td></tr>'
28082                 );
28083             }
28084             buf.push(
28085                 '<tr class="x-date-mp-btns"><td colspan="4"><button type="button" class="x-date-mp-ok">',
28086                     this.okText,
28087                     '</button><button type="button" class="x-date-mp-cancel">',
28088                     this.cancelText,
28089                     '</button></td></tr>',
28090                 '</table>'
28091             );
28092             this.monthPicker.update(buf.join(''));
28093             this.monthPicker.on('click', this.onMonthClick, this);
28094             this.monthPicker.on('dblclick', this.onMonthDblClick, this);
28095
28096             this.mpMonths = this.monthPicker.select('td.x-date-mp-month');
28097             this.mpYears = this.monthPicker.select('td.x-date-mp-year');
28098
28099             this.mpMonths.each(function(m, a, i){
28100                 i += 1;
28101                 if((i%2) == 0){
28102                     m.dom.xmonth = 5 + Math.round(i * .5);
28103                 }else{
28104                     m.dom.xmonth = Math.round((i-1) * .5);
28105                 }
28106             });
28107         }
28108     },
28109
28110     showMonthPicker : function(){
28111         this.createMonthPicker();
28112         var size = this.el.getSize();
28113         this.monthPicker.setSize(size);
28114         this.monthPicker.child('table').setSize(size);
28115
28116         this.mpSelMonth = (this.activeDate || this.value).getMonth();
28117         this.updateMPMonth(this.mpSelMonth);
28118         this.mpSelYear = (this.activeDate || this.value).getFullYear();
28119         this.updateMPYear(this.mpSelYear);
28120
28121         this.monthPicker.slideIn('t', {duration:.2});
28122     },
28123
28124     updateMPYear : function(y){
28125         this.mpyear = y;
28126         var ys = this.mpYears.elements;
28127         for(var i = 1; i <= 10; i++){
28128             var td = ys[i-1], y2;
28129             if((i%2) == 0){
28130                 y2 = y + Math.round(i * .5);
28131                 td.firstChild.innerHTML = y2;
28132                 td.xyear = y2;
28133             }else{
28134                 y2 = y - (5-Math.round(i * .5));
28135                 td.firstChild.innerHTML = y2;
28136                 td.xyear = y2;
28137             }
28138             this.mpYears.item(i-1)[y2 == this.mpSelYear ? 'addClass' : 'removeClass']('x-date-mp-sel');
28139         }
28140     },
28141
28142     updateMPMonth : function(sm){
28143         this.mpMonths.each(function(m, a, i){
28144             m[m.dom.xmonth == sm ? 'addClass' : 'removeClass']('x-date-mp-sel');
28145         });
28146     },
28147
28148     selectMPMonth: function(m){
28149         
28150     },
28151
28152     onMonthClick : function(e, t){
28153         e.stopEvent();
28154         var el = new Roo.Element(t), pn;
28155         if(el.is('button.x-date-mp-cancel')){
28156             this.hideMonthPicker();
28157         }
28158         else if(el.is('button.x-date-mp-ok')){
28159             this.update(new Date(this.mpSelYear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28160             this.hideMonthPicker();
28161         }
28162         else if(pn = el.up('td.x-date-mp-month', 2)){
28163             this.mpMonths.removeClass('x-date-mp-sel');
28164             pn.addClass('x-date-mp-sel');
28165             this.mpSelMonth = pn.dom.xmonth;
28166         }
28167         else if(pn = el.up('td.x-date-mp-year', 2)){
28168             this.mpYears.removeClass('x-date-mp-sel');
28169             pn.addClass('x-date-mp-sel');
28170             this.mpSelYear = pn.dom.xyear;
28171         }
28172         else if(el.is('a.x-date-mp-prev')){
28173             this.updateMPYear(this.mpyear-10);
28174         }
28175         else if(el.is('a.x-date-mp-next')){
28176             this.updateMPYear(this.mpyear+10);
28177         }
28178     },
28179
28180     onMonthDblClick : function(e, t){
28181         e.stopEvent();
28182         var el = new Roo.Element(t), pn;
28183         if(pn = el.up('td.x-date-mp-month', 2)){
28184             this.update(new Date(this.mpSelYear, pn.dom.xmonth, (this.activeDate || this.value).getDate()));
28185             this.hideMonthPicker();
28186         }
28187         else if(pn = el.up('td.x-date-mp-year', 2)){
28188             this.update(new Date(pn.dom.xyear, this.mpSelMonth, (this.activeDate || this.value).getDate()));
28189             this.hideMonthPicker();
28190         }
28191     },
28192
28193     hideMonthPicker : function(disableAnim){
28194         if(this.monthPicker){
28195             if(disableAnim === true){
28196                 this.monthPicker.hide();
28197             }else{
28198                 this.monthPicker.slideOut('t', {duration:.2});
28199             }
28200         }
28201     },
28202
28203     // private
28204     showPrevMonth : function(e){
28205         this.update(this.activeDate.add("mo", -1));
28206     },
28207
28208     // private
28209     showNextMonth : function(e){
28210         this.update(this.activeDate.add("mo", 1));
28211     },
28212
28213     // private
28214     showPrevYear : function(){
28215         this.update(this.activeDate.add("y", -1));
28216     },
28217
28218     // private
28219     showNextYear : function(){
28220         this.update(this.activeDate.add("y", 1));
28221     },
28222
28223     // private
28224     handleMouseWheel : function(e){
28225         var delta = e.getWheelDelta();
28226         if(delta > 0){
28227             this.showPrevMonth();
28228             e.stopEvent();
28229         } else if(delta < 0){
28230             this.showNextMonth();
28231             e.stopEvent();
28232         }
28233     },
28234
28235     // private
28236     handleDateClick : function(e, t){
28237         e.stopEvent();
28238         if(t.dateValue && !Roo.fly(t.parentNode).hasClass("x-date-disabled")){
28239             this.setValue(new Date(t.dateValue));
28240             this.fireEvent("select", this, this.value);
28241         }
28242     },
28243
28244     // private
28245     selectToday : function(){
28246         this.setValue(new Date().clearTime());
28247         this.fireEvent("select", this, this.value);
28248     },
28249
28250     // private
28251     update : function(date)
28252     {
28253         var vd = this.activeDate;
28254         this.activeDate = date;
28255         if(vd && this.el){
28256             var t = date.getTime();
28257             if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
28258                 this.cells.removeClass("x-date-selected");
28259                 this.cells.each(function(c){
28260                    if(c.dom.firstChild.dateValue == t){
28261                        c.addClass("x-date-selected");
28262                        setTimeout(function(){
28263                             try{c.dom.firstChild.focus();}catch(e){}
28264                        }, 50);
28265                        return false;
28266                    }
28267                 });
28268                 return;
28269             }
28270         }
28271         
28272         var days = date.getDaysInMonth();
28273         var firstOfMonth = date.getFirstDateOfMonth();
28274         var startingPos = firstOfMonth.getDay()-this.startDay;
28275
28276         if(startingPos <= this.startDay){
28277             startingPos += 7;
28278         }
28279
28280         var pm = date.add("mo", -1);
28281         var prevStart = pm.getDaysInMonth()-startingPos;
28282
28283         var cells = this.cells.elements;
28284         var textEls = this.textNodes;
28285         days += startingPos;
28286
28287         // convert everything to numbers so it's fast
28288         var day = 86400000;
28289         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
28290         var today = new Date().clearTime().getTime();
28291         var sel = date.clearTime().getTime();
28292         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
28293         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
28294         var ddMatch = this.disabledDatesRE;
28295         var ddText = this.disabledDatesText;
28296         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
28297         var ddaysText = this.disabledDaysText;
28298         var format = this.format;
28299
28300         var setCellClass = function(cal, cell){
28301             cell.title = "";
28302             var t = d.getTime();
28303             cell.firstChild.dateValue = t;
28304             if(t == today){
28305                 cell.className += " x-date-today";
28306                 cell.title = cal.todayText;
28307             }
28308             if(t == sel){
28309                 cell.className += " x-date-selected";
28310                 setTimeout(function(){
28311                     try{cell.firstChild.focus();}catch(e){}
28312                 }, 50);
28313             }
28314             // disabling
28315             if(t < min) {
28316                 cell.className = " x-date-disabled";
28317                 cell.title = cal.minText;
28318                 return;
28319             }
28320             if(t > max) {
28321                 cell.className = " x-date-disabled";
28322                 cell.title = cal.maxText;
28323                 return;
28324             }
28325             if(ddays){
28326                 if(ddays.indexOf(d.getDay()) != -1){
28327                     cell.title = ddaysText;
28328                     cell.className = " x-date-disabled";
28329                 }
28330             }
28331             if(ddMatch && format){
28332                 var fvalue = d.dateFormat(format);
28333                 if(ddMatch.test(fvalue)){
28334                     cell.title = ddText.replace("%0", fvalue);
28335                     cell.className = " x-date-disabled";
28336                 }
28337             }
28338         };
28339
28340         var i = 0;
28341         for(; i < startingPos; i++) {
28342             textEls[i].innerHTML = (++prevStart);
28343             d.setDate(d.getDate()+1);
28344             cells[i].className = "x-date-prevday";
28345             setCellClass(this, cells[i]);
28346         }
28347         for(; i < days; i++){
28348             intDay = i - startingPos + 1;
28349             textEls[i].innerHTML = (intDay);
28350             d.setDate(d.getDate()+1);
28351             cells[i].className = "x-date-active";
28352             setCellClass(this, cells[i]);
28353         }
28354         var extraDays = 0;
28355         for(; i < 42; i++) {
28356              textEls[i].innerHTML = (++extraDays);
28357              d.setDate(d.getDate()+1);
28358              cells[i].className = "x-date-nextday";
28359              setCellClass(this, cells[i]);
28360         }
28361
28362         this.mbtn.setText(this.monthNames[date.getMonth()] + " " + date.getFullYear());
28363         this.fireEvent('monthchange', this, date);
28364         
28365         if(!this.internalRender){
28366             var main = this.el.dom.firstChild;
28367             var w = main.offsetWidth;
28368             this.el.setWidth(w + this.el.getBorderWidth("lr"));
28369             Roo.fly(main).setWidth(w);
28370             this.internalRender = true;
28371             // opera does not respect the auto grow header center column
28372             // then, after it gets a width opera refuses to recalculate
28373             // without a second pass
28374             if(Roo.isOpera && !this.secondPass){
28375                 main.rows[0].cells[1].style.width = (w - (main.rows[0].cells[0].offsetWidth+main.rows[0].cells[2].offsetWidth)) + "px";
28376                 this.secondPass = true;
28377                 this.update.defer(10, this, [date]);
28378             }
28379         }
28380         
28381         
28382     }
28383 });        /*
28384  * Based on:
28385  * Ext JS Library 1.1.1
28386  * Copyright(c) 2006-2007, Ext JS, LLC.
28387  *
28388  * Originally Released Under LGPL - original licence link has changed is not relivant.
28389  *
28390  * Fork - LGPL
28391  * <script type="text/javascript">
28392  */
28393 /**
28394  * @class Roo.TabPanel
28395  * @extends Roo.util.Observable
28396  * A lightweight tab container.
28397  * <br><br>
28398  * Usage:
28399  * <pre><code>
28400 // basic tabs 1, built from existing content
28401 var tabs = new Roo.TabPanel("tabs1");
28402 tabs.addTab("script", "View Script");
28403 tabs.addTab("markup", "View Markup");
28404 tabs.activate("script");
28405
28406 // more advanced tabs, built from javascript
28407 var jtabs = new Roo.TabPanel("jtabs");
28408 jtabs.addTab("jtabs-1", "Normal Tab", "My content was added during construction.");
28409
28410 // set up the UpdateManager
28411 var tab2 = jtabs.addTab("jtabs-2", "Ajax Tab 1");
28412 var updater = tab2.getUpdateManager();
28413 updater.setDefaultUrl("ajax1.htm");
28414 tab2.on('activate', updater.refresh, updater, true);
28415
28416 // Use setUrl for Ajax loading
28417 var tab3 = jtabs.addTab("jtabs-3", "Ajax Tab 2");
28418 tab3.setUrl("ajax2.htm", null, true);
28419
28420 // Disabled tab
28421 var tab4 = jtabs.addTab("tabs1-5", "Disabled Tab", "Can't see me cause I'm disabled");
28422 tab4.disable();
28423
28424 jtabs.activate("jtabs-1");
28425  * </code></pre>
28426  * @constructor
28427  * Create a new TabPanel.
28428  * @param {String/HTMLElement/Roo.Element} container The id, DOM element or Roo.Element container where this TabPanel is to be rendered.
28429  * @param {Object/Boolean} config Config object to set any properties for this TabPanel, or true to render the tabs on the bottom.
28430  */
28431 Roo.TabPanel = function(container, config){
28432     /**
28433     * The container element for this TabPanel.
28434     * @type Roo.Element
28435     */
28436     this.el = Roo.get(container, true);
28437     if(config){
28438         if(typeof config == "boolean"){
28439             this.tabPosition = config ? "bottom" : "top";
28440         }else{
28441             Roo.apply(this, config);
28442         }
28443     }
28444     if(this.tabPosition == "bottom"){
28445         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28446         this.el.addClass("x-tabs-bottom");
28447     }
28448     this.stripWrap = Roo.get(this.createStrip(this.el.dom), true);
28449     this.stripEl = Roo.get(this.createStripList(this.stripWrap.dom), true);
28450     this.stripBody = Roo.get(this.stripWrap.dom.firstChild.firstChild, true);
28451     if(Roo.isIE){
28452         Roo.fly(this.stripWrap.dom.firstChild).setStyle("overflow-x", "hidden");
28453     }
28454     if(this.tabPosition != "bottom"){
28455         /** The body element that contains {@link Roo.TabPanelItem} bodies. +
28456          * @type Roo.Element
28457          */
28458         this.bodyEl = Roo.get(this.createBody(this.el.dom));
28459         this.el.addClass("x-tabs-top");
28460     }
28461     this.items = [];
28462
28463     this.bodyEl.setStyle("position", "relative");
28464
28465     this.active = null;
28466     this.activateDelegate = this.activate.createDelegate(this);
28467
28468     this.addEvents({
28469         /**
28470          * @event tabchange
28471          * Fires when the active tab changes
28472          * @param {Roo.TabPanel} this
28473          * @param {Roo.TabPanelItem} activePanel The new active tab
28474          */
28475         "tabchange": true,
28476         /**
28477          * @event beforetabchange
28478          * Fires before the active tab changes, set cancel to true on the "e" parameter to cancel the change
28479          * @param {Roo.TabPanel} this
28480          * @param {Object} e Set cancel to true on this object to cancel the tab change
28481          * @param {Roo.TabPanelItem} tab The tab being changed to
28482          */
28483         "beforetabchange" : true
28484     });
28485
28486     Roo.EventManager.onWindowResize(this.onResize, this);
28487     this.cpad = this.el.getPadding("lr");
28488     this.hiddenCount = 0;
28489
28490
28491     // toolbar on the tabbar support...
28492     if (this.toolbar) {
28493         var tcfg = this.toolbar;
28494         tcfg.container = this.stripEl.child('td.x-tab-strip-toolbar');  
28495         this.toolbar = new Roo.Toolbar(tcfg);
28496         if (Roo.isSafari) {
28497             var tbl = tcfg.container.child('table', true);
28498             tbl.setAttribute('width', '100%');
28499         }
28500         
28501     }
28502    
28503
28504
28505     Roo.TabPanel.superclass.constructor.call(this);
28506 };
28507
28508 Roo.extend(Roo.TabPanel, Roo.util.Observable, {
28509     /*
28510      *@cfg {String} tabPosition "top" or "bottom" (defaults to "top")
28511      */
28512     tabPosition : "top",
28513     /*
28514      *@cfg {Number} currentTabWidth The width of the current tab (defaults to 0)
28515      */
28516     currentTabWidth : 0,
28517     /*
28518      *@cfg {Number} minTabWidth The minimum width of a tab (defaults to 40) (ignored if {@link #resizeTabs} is not true)
28519      */
28520     minTabWidth : 40,
28521     /*
28522      *@cfg {Number} maxTabWidth The maximum width of a tab (defaults to 250) (ignored if {@link #resizeTabs} is not true)
28523      */
28524     maxTabWidth : 250,
28525     /*
28526      *@cfg {Number} preferredTabWidth The preferred (default) width of a tab (defaults to 175) (ignored if {@link #resizeTabs} is not true)
28527      */
28528     preferredTabWidth : 175,
28529     /*
28530      *@cfg {Boolean} resizeTabs True to enable dynamic tab resizing (defaults to false)
28531      */
28532     resizeTabs : false,
28533     /*
28534      *@cfg {Boolean} monitorResize Set this to true to turn on window resize monitoring (ignored if {@link #resizeTabs} is not true) (defaults to true)
28535      */
28536     monitorResize : true,
28537     /*
28538      *@cfg {Object} toolbar xtype description of toolbar to show at the right of the tab bar. 
28539      */
28540     toolbar : false,
28541
28542     /**
28543      * Creates a new {@link Roo.TabPanelItem} by looking for an existing element with the provided id -- if it's not found it creates one.
28544      * @param {String} id The id of the div to use <b>or create</b>
28545      * @param {String} text The text for the tab
28546      * @param {String} content (optional) Content to put in the TabPanelItem body
28547      * @param {Boolean} closable (optional) True to create a close icon on the tab
28548      * @return {Roo.TabPanelItem} The created TabPanelItem
28549      */
28550     addTab : function(id, text, content, closable){
28551         var item = new Roo.TabPanelItem(this, id, text, closable);
28552         this.addTabItem(item);
28553         if(content){
28554             item.setContent(content);
28555         }
28556         return item;
28557     },
28558
28559     /**
28560      * Returns the {@link Roo.TabPanelItem} with the specified id/index
28561      * @param {String/Number} id The id or index of the TabPanelItem to fetch.
28562      * @return {Roo.TabPanelItem}
28563      */
28564     getTab : function(id){
28565         return this.items[id];
28566     },
28567
28568     /**
28569      * Hides the {@link Roo.TabPanelItem} with the specified id/index
28570      * @param {String/Number} id The id or index of the TabPanelItem to hide.
28571      */
28572     hideTab : function(id){
28573         var t = this.items[id];
28574         if(!t.isHidden()){
28575            t.setHidden(true);
28576            this.hiddenCount++;
28577            this.autoSizeTabs();
28578         }
28579     },
28580
28581     /**
28582      * "Unhides" the {@link Roo.TabPanelItem} with the specified id/index.
28583      * @param {String/Number} id The id or index of the TabPanelItem to unhide.
28584      */
28585     unhideTab : function(id){
28586         var t = this.items[id];
28587         if(t.isHidden()){
28588            t.setHidden(false);
28589            this.hiddenCount--;
28590            this.autoSizeTabs();
28591         }
28592     },
28593
28594     /**
28595      * Adds an existing {@link Roo.TabPanelItem}.
28596      * @param {Roo.TabPanelItem} item The TabPanelItem to add
28597      */
28598     addTabItem : function(item){
28599         this.items[item.id] = item;
28600         this.items.push(item);
28601         if(this.resizeTabs){
28602            item.setWidth(this.currentTabWidth || this.preferredTabWidth);
28603            this.autoSizeTabs();
28604         }else{
28605             item.autoSize();
28606         }
28607     },
28608
28609     /**
28610      * Removes a {@link Roo.TabPanelItem}.
28611      * @param {String/Number} id The id or index of the TabPanelItem to remove.
28612      */
28613     removeTab : function(id){
28614         var items = this.items;
28615         var tab = items[id];
28616         if(!tab) { return; }
28617         var index = items.indexOf(tab);
28618         if(this.active == tab && items.length > 1){
28619             var newTab = this.getNextAvailable(index);
28620             if(newTab) {
28621                 newTab.activate();
28622             }
28623         }
28624         this.stripEl.dom.removeChild(tab.pnode.dom);
28625         if(tab.bodyEl.dom.parentNode == this.bodyEl.dom){ // if it was moved already prevent error
28626             this.bodyEl.dom.removeChild(tab.bodyEl.dom);
28627         }
28628         items.splice(index, 1);
28629         delete this.items[tab.id];
28630         tab.fireEvent("close", tab);
28631         tab.purgeListeners();
28632         this.autoSizeTabs();
28633     },
28634
28635     getNextAvailable : function(start){
28636         var items = this.items;
28637         var index = start;
28638         // look for a next tab that will slide over to
28639         // replace the one being removed
28640         while(index < items.length){
28641             var item = items[++index];
28642             if(item && !item.isHidden()){
28643                 return item;
28644             }
28645         }
28646         // if one isn't found select the previous tab (on the left)
28647         index = start;
28648         while(index >= 0){
28649             var item = items[--index];
28650             if(item && !item.isHidden()){
28651                 return item;
28652             }
28653         }
28654         return null;
28655     },
28656
28657     /**
28658      * Disables a {@link Roo.TabPanelItem}. It cannot be the active tab, if it is this call is ignored.
28659      * @param {String/Number} id The id or index of the TabPanelItem to disable.
28660      */
28661     disableTab : function(id){
28662         var tab = this.items[id];
28663         if(tab && this.active != tab){
28664             tab.disable();
28665         }
28666     },
28667
28668     /**
28669      * Enables a {@link Roo.TabPanelItem} that is disabled.
28670      * @param {String/Number} id The id or index of the TabPanelItem to enable.
28671      */
28672     enableTab : function(id){
28673         var tab = this.items[id];
28674         tab.enable();
28675     },
28676
28677     /**
28678      * Activates a {@link Roo.TabPanelItem}. The currently active one will be deactivated.
28679      * @param {String/Number} id The id or index of the TabPanelItem to activate.
28680      * @return {Roo.TabPanelItem} The TabPanelItem.
28681      */
28682     activate : function(id){
28683         var tab = this.items[id];
28684         if(!tab){
28685             return null;
28686         }
28687         if(tab == this.active || tab.disabled){
28688             return tab;
28689         }
28690         var e = {};
28691         this.fireEvent("beforetabchange", this, e, tab);
28692         if(e.cancel !== true && !tab.disabled){
28693             if(this.active){
28694                 this.active.hide();
28695             }
28696             this.active = this.items[id];
28697             this.active.show();
28698             this.fireEvent("tabchange", this, this.active);
28699         }
28700         return tab;
28701     },
28702
28703     /**
28704      * Gets the active {@link Roo.TabPanelItem}.
28705      * @return {Roo.TabPanelItem} The active TabPanelItem or null if none are active.
28706      */
28707     getActiveTab : function(){
28708         return this.active;
28709     },
28710
28711     /**
28712      * Updates the tab body element to fit the height of the container element
28713      * for overflow scrolling
28714      * @param {Number} targetHeight (optional) Override the starting height from the elements height
28715      */
28716     syncHeight : function(targetHeight){
28717         var height = (targetHeight || this.el.getHeight())-this.el.getBorderWidth("tb")-this.el.getPadding("tb");
28718         var bm = this.bodyEl.getMargins();
28719         var newHeight = height-(this.stripWrap.getHeight()||0)-(bm.top+bm.bottom);
28720         this.bodyEl.setHeight(newHeight);
28721         return newHeight;
28722     },
28723
28724     onResize : function(){
28725         if(this.monitorResize){
28726             this.autoSizeTabs();
28727         }
28728     },
28729
28730     /**
28731      * Disables tab resizing while tabs are being added (if {@link #resizeTabs} is false this does nothing)
28732      */
28733     beginUpdate : function(){
28734         this.updating = true;
28735     },
28736
28737     /**
28738      * Stops an update and resizes the tabs (if {@link #resizeTabs} is false this does nothing)
28739      */
28740     endUpdate : function(){
28741         this.updating = false;
28742         this.autoSizeTabs();
28743     },
28744
28745     /**
28746      * Manual call to resize the tabs (if {@link #resizeTabs} is false this does nothing)
28747      */
28748     autoSizeTabs : function(){
28749         var count = this.items.length;
28750         var vcount = count - this.hiddenCount;
28751         if(!this.resizeTabs || count < 1 || vcount < 1 || this.updating) {
28752             return;
28753         }
28754         var w = Math.max(this.el.getWidth() - this.cpad, 10);
28755         var availWidth = Math.floor(w / vcount);
28756         var b = this.stripBody;
28757         if(b.getWidth() > w){
28758             var tabs = this.items;
28759             this.setTabWidth(Math.max(availWidth, this.minTabWidth)-2);
28760             if(availWidth < this.minTabWidth){
28761                 /*if(!this.sleft){    // incomplete scrolling code
28762                     this.createScrollButtons();
28763                 }
28764                 this.showScroll();
28765                 this.stripClip.setWidth(w - (this.sleft.getWidth()+this.sright.getWidth()));*/
28766             }
28767         }else{
28768             if(this.currentTabWidth < this.preferredTabWidth){
28769                 this.setTabWidth(Math.min(availWidth, this.preferredTabWidth)-2);
28770             }
28771         }
28772     },
28773
28774     /**
28775      * Returns the number of tabs in this TabPanel.
28776      * @return {Number}
28777      */
28778      getCount : function(){
28779          return this.items.length;
28780      },
28781
28782     /**
28783      * Resizes all the tabs to the passed width
28784      * @param {Number} The new width
28785      */
28786     setTabWidth : function(width){
28787         this.currentTabWidth = width;
28788         for(var i = 0, len = this.items.length; i < len; i++) {
28789                 if(!this.items[i].isHidden()) {
28790                 this.items[i].setWidth(width);
28791             }
28792         }
28793     },
28794
28795     /**
28796      * Destroys this TabPanel
28797      * @param {Boolean} removeEl (optional) True to remove the element from the DOM as well (defaults to undefined)
28798      */
28799     destroy : function(removeEl){
28800         Roo.EventManager.removeResizeListener(this.onResize, this);
28801         for(var i = 0, len = this.items.length; i < len; i++){
28802             this.items[i].purgeListeners();
28803         }
28804         if(removeEl === true){
28805             this.el.update("");
28806             this.el.remove();
28807         }
28808     }
28809 });
28810
28811 /**
28812  * @class Roo.TabPanelItem
28813  * @extends Roo.util.Observable
28814  * Represents an individual item (tab plus body) in a TabPanel.
28815  * @param {Roo.TabPanel} tabPanel The {@link Roo.TabPanel} this TabPanelItem belongs to
28816  * @param {String} id The id of this TabPanelItem
28817  * @param {String} text The text for the tab of this TabPanelItem
28818  * @param {Boolean} closable True to allow this TabPanelItem to be closable (defaults to false)
28819  */
28820 Roo.TabPanelItem = function(tabPanel, id, text, closable){
28821     /**
28822      * The {@link Roo.TabPanel} this TabPanelItem belongs to
28823      * @type Roo.TabPanel
28824      */
28825     this.tabPanel = tabPanel;
28826     /**
28827      * The id for this TabPanelItem
28828      * @type String
28829      */
28830     this.id = id;
28831     /** @private */
28832     this.disabled = false;
28833     /** @private */
28834     this.text = text;
28835     /** @private */
28836     this.loaded = false;
28837     this.closable = closable;
28838
28839     /**
28840      * The body element for this TabPanelItem.
28841      * @type Roo.Element
28842      */
28843     this.bodyEl = Roo.get(tabPanel.createItemBody(tabPanel.bodyEl.dom, id));
28844     this.bodyEl.setVisibilityMode(Roo.Element.VISIBILITY);
28845     this.bodyEl.setStyle("display", "block");
28846     this.bodyEl.setStyle("zoom", "1");
28847     this.hideAction();
28848
28849     var els = tabPanel.createStripElements(tabPanel.stripEl.dom, text, closable);
28850     /** @private */
28851     this.el = Roo.get(els.el, true);
28852     this.inner = Roo.get(els.inner, true);
28853     this.textEl = Roo.get(this.el.dom.firstChild.firstChild.firstChild, true);
28854     this.pnode = Roo.get(els.el.parentNode, true);
28855     this.el.on("mousedown", this.onTabMouseDown, this);
28856     this.el.on("click", this.onTabClick, this);
28857     /** @private */
28858     if(closable){
28859         var c = Roo.get(els.close, true);
28860         c.dom.title = this.closeText;
28861         c.addClassOnOver("close-over");
28862         c.on("click", this.closeClick, this);
28863      }
28864
28865     this.addEvents({
28866          /**
28867          * @event activate
28868          * Fires when this tab becomes the active tab.
28869          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28870          * @param {Roo.TabPanelItem} this
28871          */
28872         "activate": true,
28873         /**
28874          * @event beforeclose
28875          * Fires before this tab is closed. To cancel the close, set cancel to true on e (e.cancel = true).
28876          * @param {Roo.TabPanelItem} this
28877          * @param {Object} e Set cancel to true on this object to cancel the close.
28878          */
28879         "beforeclose": true,
28880         /**
28881          * @event close
28882          * Fires when this tab is closed.
28883          * @param {Roo.TabPanelItem} this
28884          */
28885          "close": true,
28886         /**
28887          * @event deactivate
28888          * Fires when this tab is no longer the active tab.
28889          * @param {Roo.TabPanel} tabPanel The parent TabPanel
28890          * @param {Roo.TabPanelItem} this
28891          */
28892          "deactivate" : true
28893     });
28894     this.hidden = false;
28895
28896     Roo.TabPanelItem.superclass.constructor.call(this);
28897 };
28898
28899 Roo.extend(Roo.TabPanelItem, Roo.util.Observable, {
28900     purgeListeners : function(){
28901        Roo.util.Observable.prototype.purgeListeners.call(this);
28902        this.el.removeAllListeners();
28903     },
28904     /**
28905      * Shows this TabPanelItem -- this <b>does not</b> deactivate the currently active TabPanelItem.
28906      */
28907     show : function(){
28908         this.pnode.addClass("on");
28909         this.showAction();
28910         if(Roo.isOpera){
28911             this.tabPanel.stripWrap.repaint();
28912         }
28913         this.fireEvent("activate", this.tabPanel, this);
28914     },
28915
28916     /**
28917      * Returns true if this tab is the active tab.
28918      * @return {Boolean}
28919      */
28920     isActive : function(){
28921         return this.tabPanel.getActiveTab() == this;
28922     },
28923
28924     /**
28925      * Hides this TabPanelItem -- if you don't activate another TabPanelItem this could look odd.
28926      */
28927     hide : function(){
28928         this.pnode.removeClass("on");
28929         this.hideAction();
28930         this.fireEvent("deactivate", this.tabPanel, this);
28931     },
28932
28933     hideAction : function(){
28934         this.bodyEl.hide();
28935         this.bodyEl.setStyle("position", "absolute");
28936         this.bodyEl.setLeft("-20000px");
28937         this.bodyEl.setTop("-20000px");
28938     },
28939
28940     showAction : function(){
28941         this.bodyEl.setStyle("position", "relative");
28942         this.bodyEl.setTop("");
28943         this.bodyEl.setLeft("");
28944         this.bodyEl.show();
28945     },
28946
28947     /**
28948      * Set the tooltip for the tab.
28949      * @param {String} tooltip The tab's tooltip
28950      */
28951     setTooltip : function(text){
28952         if(Roo.QuickTips && Roo.QuickTips.isEnabled()){
28953             this.textEl.dom.qtip = text;
28954             this.textEl.dom.removeAttribute('title');
28955         }else{
28956             this.textEl.dom.title = text;
28957         }
28958     },
28959
28960     onTabClick : function(e){
28961         e.preventDefault();
28962         this.tabPanel.activate(this.id);
28963     },
28964
28965     onTabMouseDown : function(e){
28966         e.preventDefault();
28967         this.tabPanel.activate(this.id);
28968     },
28969
28970     getWidth : function(){
28971         return this.inner.getWidth();
28972     },
28973
28974     setWidth : function(width){
28975         var iwidth = width - this.pnode.getPadding("lr");
28976         this.inner.setWidth(iwidth);
28977         this.textEl.setWidth(iwidth-this.inner.getPadding("lr"));
28978         this.pnode.setWidth(width);
28979     },
28980
28981     /**
28982      * Show or hide the tab
28983      * @param {Boolean} hidden True to hide or false to show.
28984      */
28985     setHidden : function(hidden){
28986         this.hidden = hidden;
28987         this.pnode.setStyle("display", hidden ? "none" : "");
28988     },
28989
28990     /**
28991      * Returns true if this tab is "hidden"
28992      * @return {Boolean}
28993      */
28994     isHidden : function(){
28995         return this.hidden;
28996     },
28997
28998     /**
28999      * Returns the text for this tab
29000      * @return {String}
29001      */
29002     getText : function(){
29003         return this.text;
29004     },
29005
29006     autoSize : function(){
29007         //this.el.beginMeasure();
29008         this.textEl.setWidth(1);
29009         /*
29010          *  #2804 [new] Tabs in Roojs
29011          *  increase the width by 2-4 pixels to prevent the ellipssis showing in chrome
29012          */
29013         this.setWidth(this.textEl.dom.scrollWidth+this.pnode.getPadding("lr")+this.inner.getPadding("lr") + 2);
29014         //this.el.endMeasure();
29015     },
29016
29017     /**
29018      * Sets the text for the tab (Note: this also sets the tooltip text)
29019      * @param {String} text The tab's text and tooltip
29020      */
29021     setText : function(text){
29022         this.text = text;
29023         this.textEl.update(text);
29024         this.setTooltip(text);
29025         if(!this.tabPanel.resizeTabs){
29026             this.autoSize();
29027         }
29028     },
29029     /**
29030      * Activates this TabPanelItem -- this <b>does</b> deactivate the currently active TabPanelItem.
29031      */
29032     activate : function(){
29033         this.tabPanel.activate(this.id);
29034     },
29035
29036     /**
29037      * Disables this TabPanelItem -- this does nothing if this is the active TabPanelItem.
29038      */
29039     disable : function(){
29040         if(this.tabPanel.active != this){
29041             this.disabled = true;
29042             this.pnode.addClass("disabled");
29043         }
29044     },
29045
29046     /**
29047      * Enables this TabPanelItem if it was previously disabled.
29048      */
29049     enable : function(){
29050         this.disabled = false;
29051         this.pnode.removeClass("disabled");
29052     },
29053
29054     /**
29055      * Sets the content for this TabPanelItem.
29056      * @param {String} content The content
29057      * @param {Boolean} loadScripts true to look for and load scripts
29058      */
29059     setContent : function(content, loadScripts){
29060         this.bodyEl.update(content, loadScripts);
29061     },
29062
29063     /**
29064      * Gets the {@link Roo.UpdateManager} for the body of this TabPanelItem. Enables you to perform Ajax updates.
29065      * @return {Roo.UpdateManager} The UpdateManager
29066      */
29067     getUpdateManager : function(){
29068         return this.bodyEl.getUpdateManager();
29069     },
29070
29071     /**
29072      * Set a URL to be used to load the content for this TabPanelItem.
29073      * @param {String/Function} url The URL to load the content from, or a function to call to get the URL
29074      * @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)
29075      * @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)
29076      * @return {Roo.UpdateManager} The UpdateManager
29077      */
29078     setUrl : function(url, params, loadOnce){
29079         if(this.refreshDelegate){
29080             this.un('activate', this.refreshDelegate);
29081         }
29082         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
29083         this.on("activate", this.refreshDelegate);
29084         return this.bodyEl.getUpdateManager();
29085     },
29086
29087     /** @private */
29088     _handleRefresh : function(url, params, loadOnce){
29089         if(!loadOnce || !this.loaded){
29090             var updater = this.bodyEl.getUpdateManager();
29091             updater.update(url, params, this._setLoaded.createDelegate(this));
29092         }
29093     },
29094
29095     /**
29096      *   Forces a content refresh from the URL specified in the {@link #setUrl} method.
29097      *   Will fail silently if the setUrl method has not been called.
29098      *   This does not activate the panel, just updates its content.
29099      */
29100     refresh : function(){
29101         if(this.refreshDelegate){
29102            this.loaded = false;
29103            this.refreshDelegate();
29104         }
29105     },
29106
29107     /** @private */
29108     _setLoaded : function(){
29109         this.loaded = true;
29110     },
29111
29112     /** @private */
29113     closeClick : function(e){
29114         var o = {};
29115         e.stopEvent();
29116         this.fireEvent("beforeclose", this, o);
29117         if(o.cancel !== true){
29118             this.tabPanel.removeTab(this.id);
29119         }
29120     },
29121     /**
29122      * The text displayed in the tooltip for the close icon.
29123      * @type String
29124      */
29125     closeText : "Close this tab"
29126 });
29127
29128 /** @private */
29129 Roo.TabPanel.prototype.createStrip = function(container){
29130     var strip = document.createElement("div");
29131     strip.className = "x-tabs-wrap";
29132     container.appendChild(strip);
29133     return strip;
29134 };
29135 /** @private */
29136 Roo.TabPanel.prototype.createStripList = function(strip){
29137     // div wrapper for retard IE
29138     // returns the "tr" element.
29139     strip.innerHTML = '<div class="x-tabs-strip-wrap">'+
29140         '<table class="x-tabs-strip" cellspacing="0" cellpadding="0" border="0"><tbody><tr>'+
29141         '<td class="x-tab-strip-toolbar"></td></tr></tbody></table></div>';
29142     return strip.firstChild.firstChild.firstChild.firstChild;
29143 };
29144 /** @private */
29145 Roo.TabPanel.prototype.createBody = function(container){
29146     var body = document.createElement("div");
29147     Roo.id(body, "tab-body");
29148     Roo.fly(body).addClass("x-tabs-body");
29149     container.appendChild(body);
29150     return body;
29151 };
29152 /** @private */
29153 Roo.TabPanel.prototype.createItemBody = function(bodyEl, id){
29154     var body = Roo.getDom(id);
29155     if(!body){
29156         body = document.createElement("div");
29157         body.id = id;
29158     }
29159     Roo.fly(body).addClass("x-tabs-item-body");
29160     bodyEl.insertBefore(body, bodyEl.firstChild);
29161     return body;
29162 };
29163 /** @private */
29164 Roo.TabPanel.prototype.createStripElements = function(stripEl, text, closable){
29165     var td = document.createElement("td");
29166     stripEl.insertBefore(td, stripEl.childNodes[stripEl.childNodes.length-1]);
29167     //stripEl.appendChild(td);
29168     if(closable){
29169         td.className = "x-tabs-closable";
29170         if(!this.closeTpl){
29171             this.closeTpl = new Roo.Template(
29172                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29173                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span>' +
29174                '<div unselectable="on" class="close-icon">&#160;</div></em></span></a>'
29175             );
29176         }
29177         var el = this.closeTpl.overwrite(td, {"text": text});
29178         var close = el.getElementsByTagName("div")[0];
29179         var inner = el.getElementsByTagName("em")[0];
29180         return {"el": el, "close": close, "inner": inner};
29181     } else {
29182         if(!this.tabTpl){
29183             this.tabTpl = new Roo.Template(
29184                '<a href="#" class="x-tabs-right"><span class="x-tabs-left"><em class="x-tabs-inner">' +
29185                '<span unselectable="on"' + (this.disableTooltips ? '' : ' title="{text}"') +' class="x-tabs-text">{text}</span></em></span></a>'
29186             );
29187         }
29188         var el = this.tabTpl.overwrite(td, {"text": text});
29189         var inner = el.getElementsByTagName("em")[0];
29190         return {"el": el, "inner": inner};
29191     }
29192 };/*
29193  * Based on:
29194  * Ext JS Library 1.1.1
29195  * Copyright(c) 2006-2007, Ext JS, LLC.
29196  *
29197  * Originally Released Under LGPL - original licence link has changed is not relivant.
29198  *
29199  * Fork - LGPL
29200  * <script type="text/javascript">
29201  */
29202
29203 /**
29204  * @class Roo.Button
29205  * @extends Roo.util.Observable
29206  * Simple Button class
29207  * @cfg {String} text The button text
29208  * @cfg {String} icon The path to an image to display in the button (the image will be set as the background-image
29209  * CSS property of the button by default, so if you want a mixed icon/text button, set cls:"x-btn-text-icon")
29210  * @cfg {Function} handler A function called when the button is clicked (can be used instead of click event)
29211  * @cfg {Object} scope The scope of the handler
29212  * @cfg {Number} minWidth The minimum width for this button (used to give a set of buttons a common width)
29213  * @cfg {String/Object} tooltip The tooltip for the button - can be a string or QuickTips config object
29214  * @cfg {Boolean} hidden True to start hidden (defaults to false)
29215  * @cfg {Boolean} disabled True to start disabled (defaults to false)
29216  * @cfg {Boolean} pressed True to start pressed (only if enableToggle = true)
29217  * @cfg {String} toggleGroup The group this toggle button is a member of (only 1 per group can be pressed, only
29218    applies if enableToggle = true)
29219  * @cfg {String/HTMLElement/Element} renderTo The element to append the button to
29220  * @cfg {Boolean/Object} repeat True to repeat fire the click event while the mouse is down. This can also be
29221   an {@link Roo.util.ClickRepeater} config object (defaults to false).
29222  * @constructor
29223  * Create a new button
29224  * @param {Object} config The config object
29225  */
29226 Roo.Button = function(renderTo, config)
29227 {
29228     if (!config) {
29229         config = renderTo;
29230         renderTo = config.renderTo || false;
29231     }
29232     
29233     Roo.apply(this, config);
29234     this.addEvents({
29235         /**
29236              * @event click
29237              * Fires when this button is clicked
29238              * @param {Button} this
29239              * @param {EventObject} e The click event
29240              */
29241             "click" : true,
29242         /**
29243              * @event toggle
29244              * Fires when the "pressed" state of this button changes (only if enableToggle = true)
29245              * @param {Button} this
29246              * @param {Boolean} pressed
29247              */
29248             "toggle" : true,
29249         /**
29250              * @event mouseover
29251              * Fires when the mouse hovers over the button
29252              * @param {Button} this
29253              * @param {Event} e The event object
29254              */
29255         'mouseover' : true,
29256         /**
29257              * @event mouseout
29258              * Fires when the mouse exits the button
29259              * @param {Button} this
29260              * @param {Event} e The event object
29261              */
29262         'mouseout': true,
29263          /**
29264              * @event render
29265              * Fires when the button is rendered
29266              * @param {Button} this
29267              */
29268         'render': true
29269     });
29270     if(this.menu){
29271         this.menu = Roo.menu.MenuMgr.get(this.menu);
29272     }
29273     // register listeners first!!  - so render can be captured..
29274     Roo.util.Observable.call(this);
29275     if(renderTo){
29276         this.render(renderTo);
29277     }
29278     
29279   
29280 };
29281
29282 Roo.extend(Roo.Button, Roo.util.Observable, {
29283     /**
29284      * 
29285      */
29286     
29287     /**
29288      * Read-only. True if this button is hidden
29289      * @type Boolean
29290      */
29291     hidden : false,
29292     /**
29293      * Read-only. True if this button is disabled
29294      * @type Boolean
29295      */
29296     disabled : false,
29297     /**
29298      * Read-only. True if this button is pressed (only if enableToggle = true)
29299      * @type Boolean
29300      */
29301     pressed : false,
29302
29303     /**
29304      * @cfg {Number} tabIndex 
29305      * The DOM tabIndex for this button (defaults to undefined)
29306      */
29307     tabIndex : undefined,
29308
29309     /**
29310      * @cfg {Boolean} enableToggle
29311      * True to enable pressed/not pressed toggling (defaults to false)
29312      */
29313     enableToggle: false,
29314     /**
29315      * @cfg {Mixed} menu
29316      * Standard menu attribute consisting of a reference to a menu object, a menu id or a menu config blob (defaults to undefined).
29317      */
29318     menu : undefined,
29319     /**
29320      * @cfg {String} menuAlign
29321      * The position to align the menu to (see {@link Roo.Element#alignTo} for more details, defaults to 'tl-bl?').
29322      */
29323     menuAlign : "tl-bl?",
29324
29325     /**
29326      * @cfg {String} iconCls
29327      * A css class which sets a background image to be used as the icon for this button (defaults to undefined).
29328      */
29329     iconCls : undefined,
29330     /**
29331      * @cfg {String} type
29332      * The button's type, corresponding to the DOM input element type attribute.  Either "submit," "reset" or "button" (default).
29333      */
29334     type : 'button',
29335
29336     // private
29337     menuClassTarget: 'tr',
29338
29339     /**
29340      * @cfg {String} clickEvent
29341      * The type of event to map to the button's event handler (defaults to 'click')
29342      */
29343     clickEvent : 'click',
29344
29345     /**
29346      * @cfg {Boolean} handleMouseEvents
29347      * False to disable visual cues on mouseover, mouseout and mousedown (defaults to true)
29348      */
29349     handleMouseEvents : true,
29350
29351     /**
29352      * @cfg {String} tooltipType
29353      * The type of tooltip to use. Either "qtip" (default) for QuickTips or "title" for title attribute.
29354      */
29355     tooltipType : 'qtip',
29356
29357     /**
29358      * @cfg {String} cls
29359      * A CSS class to apply to the button's main element.
29360      */
29361     
29362     /**
29363      * @cfg {Roo.Template} template (Optional)
29364      * An {@link Roo.Template} with which to create the Button's main element. This Template must
29365      * contain numeric substitution parameter 0 if it is to display the tRoo property. Changing the template could
29366      * require code modifications if required elements (e.g. a button) aren't present.
29367      */
29368
29369     // private
29370     render : function(renderTo){
29371         var btn;
29372         if(this.hideParent){
29373             this.parentEl = Roo.get(renderTo);
29374         }
29375         if(!this.dhconfig){
29376             if(!this.template){
29377                 if(!Roo.Button.buttonTemplate){
29378                     // hideous table template
29379                     Roo.Button.buttonTemplate = new Roo.Template(
29380                         '<table border="0" cellpadding="0" cellspacing="0" class="x-btn-wrap"><tbody><tr>',
29381                         '<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>',
29382                         "</tr></tbody></table>");
29383                 }
29384                 this.template = Roo.Button.buttonTemplate;
29385             }
29386             btn = this.template.append(renderTo, [this.text || '&#160;', this.type], true);
29387             var btnEl = btn.child("button:first");
29388             btnEl.on('focus', this.onFocus, this);
29389             btnEl.on('blur', this.onBlur, this);
29390             if(this.cls){
29391                 btn.addClass(this.cls);
29392             }
29393             if(this.icon){
29394                 btnEl.setStyle('background-image', 'url(' +this.icon +')');
29395             }
29396             if(this.iconCls){
29397                 btnEl.addClass(this.iconCls);
29398                 if(!this.cls){
29399                     btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29400                 }
29401             }
29402             if(this.tabIndex !== undefined){
29403                 btnEl.dom.tabIndex = this.tabIndex;
29404             }
29405             if(this.tooltip){
29406                 if(typeof this.tooltip == 'object'){
29407                     Roo.QuickTips.tips(Roo.apply({
29408                           target: btnEl.id
29409                     }, this.tooltip));
29410                 } else {
29411                     btnEl.dom[this.tooltipType] = this.tooltip;
29412                 }
29413             }
29414         }else{
29415             btn = Roo.DomHelper.append(Roo.get(renderTo).dom, this.dhconfig, true);
29416         }
29417         this.el = btn;
29418         if(this.id){
29419             this.el.dom.id = this.el.id = this.id;
29420         }
29421         if(this.menu){
29422             this.el.child(this.menuClassTarget).addClass("x-btn-with-menu");
29423             this.menu.on("show", this.onMenuShow, this);
29424             this.menu.on("hide", this.onMenuHide, this);
29425         }
29426         btn.addClass("x-btn");
29427         if(Roo.isIE && !Roo.isIE7){
29428             this.autoWidth.defer(1, this);
29429         }else{
29430             this.autoWidth();
29431         }
29432         if(this.handleMouseEvents){
29433             btn.on("mouseover", this.onMouseOver, this);
29434             btn.on("mouseout", this.onMouseOut, this);
29435             btn.on("mousedown", this.onMouseDown, this);
29436         }
29437         btn.on(this.clickEvent, this.onClick, this);
29438         //btn.on("mouseup", this.onMouseUp, this);
29439         if(this.hidden){
29440             this.hide();
29441         }
29442         if(this.disabled){
29443             this.disable();
29444         }
29445         Roo.ButtonToggleMgr.register(this);
29446         if(this.pressed){
29447             this.el.addClass("x-btn-pressed");
29448         }
29449         if(this.repeat){
29450             var repeater = new Roo.util.ClickRepeater(btn,
29451                 typeof this.repeat == "object" ? this.repeat : {}
29452             );
29453             repeater.on("click", this.onClick,  this);
29454         }
29455         
29456         this.fireEvent('render', this);
29457         
29458     },
29459     /**
29460      * Returns the button's underlying element
29461      * @return {Roo.Element} The element
29462      */
29463     getEl : function(){
29464         return this.el;  
29465     },
29466     
29467     /**
29468      * Destroys this Button and removes any listeners.
29469      */
29470     destroy : function(){
29471         Roo.ButtonToggleMgr.unregister(this);
29472         this.el.removeAllListeners();
29473         this.purgeListeners();
29474         this.el.remove();
29475     },
29476
29477     // private
29478     autoWidth : function(){
29479         if(this.el){
29480             this.el.setWidth("auto");
29481             if(Roo.isIE7 && Roo.isStrict){
29482                 var ib = this.el.child('button');
29483                 if(ib && ib.getWidth() > 20){
29484                     ib.clip();
29485                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29486                 }
29487             }
29488             if(this.minWidth){
29489                 if(this.hidden){
29490                     this.el.beginMeasure();
29491                 }
29492                 if(this.el.getWidth() < this.minWidth){
29493                     this.el.setWidth(this.minWidth);
29494                 }
29495                 if(this.hidden){
29496                     this.el.endMeasure();
29497                 }
29498             }
29499         }
29500     },
29501
29502     /**
29503      * Assigns this button's click handler
29504      * @param {Function} handler The function to call when the button is clicked
29505      * @param {Object} scope (optional) Scope for the function passed in
29506      */
29507     setHandler : function(handler, scope){
29508         this.handler = handler;
29509         this.scope = scope;  
29510     },
29511     
29512     /**
29513      * Sets this button's text
29514      * @param {String} text The button text
29515      */
29516     setText : function(text){
29517         this.text = text;
29518         if(this.el){
29519             this.el.child("td.x-btn-center button.x-btn-text").update(text);
29520         }
29521         this.autoWidth();
29522     },
29523     
29524     /**
29525      * Gets the text for this button
29526      * @return {String} The button text
29527      */
29528     getText : function(){
29529         return this.text;  
29530     },
29531     
29532     /**
29533      * Show this button
29534      */
29535     show: function(){
29536         this.hidden = false;
29537         if(this.el){
29538             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "");
29539         }
29540     },
29541     
29542     /**
29543      * Hide this button
29544      */
29545     hide: function(){
29546         this.hidden = true;
29547         if(this.el){
29548             this[this.hideParent? 'parentEl' : 'el'].setStyle("display", "none");
29549         }
29550     },
29551     
29552     /**
29553      * Convenience function for boolean show/hide
29554      * @param {Boolean} visible True to show, false to hide
29555      */
29556     setVisible: function(visible){
29557         if(visible) {
29558             this.show();
29559         }else{
29560             this.hide();
29561         }
29562     },
29563     
29564     /**
29565      * If a state it passed, it becomes the pressed state otherwise the current state is toggled.
29566      * @param {Boolean} state (optional) Force a particular state
29567      */
29568     toggle : function(state){
29569         state = state === undefined ? !this.pressed : state;
29570         if(state != this.pressed){
29571             if(state){
29572                 this.el.addClass("x-btn-pressed");
29573                 this.pressed = true;
29574                 this.fireEvent("toggle", this, true);
29575             }else{
29576                 this.el.removeClass("x-btn-pressed");
29577                 this.pressed = false;
29578                 this.fireEvent("toggle", this, false);
29579             }
29580             if(this.toggleHandler){
29581                 this.toggleHandler.call(this.scope || this, this, state);
29582             }
29583         }
29584     },
29585     
29586     /**
29587      * Focus the button
29588      */
29589     focus : function(){
29590         this.el.child('button:first').focus();
29591     },
29592     
29593     /**
29594      * Disable this button
29595      */
29596     disable : function(){
29597         if(this.el){
29598             this.el.addClass("x-btn-disabled");
29599         }
29600         this.disabled = true;
29601     },
29602     
29603     /**
29604      * Enable this button
29605      */
29606     enable : function(){
29607         if(this.el){
29608             this.el.removeClass("x-btn-disabled");
29609         }
29610         this.disabled = false;
29611     },
29612
29613     /**
29614      * Convenience function for boolean enable/disable
29615      * @param {Boolean} enabled True to enable, false to disable
29616      */
29617     setDisabled : function(v){
29618         this[v !== true ? "enable" : "disable"]();
29619     },
29620
29621     // private
29622     onClick : function(e)
29623     {
29624         if(e){
29625             e.preventDefault();
29626         }
29627         if(e.button != 0){
29628             return;
29629         }
29630         if(!this.disabled){
29631             if(this.enableToggle){
29632                 this.toggle();
29633             }
29634             if(this.menu && !this.menu.isVisible()){
29635                 this.menu.show(this.el, this.menuAlign);
29636             }
29637             this.fireEvent("click", this, e);
29638             if(this.handler){
29639                 this.el.removeClass("x-btn-over");
29640                 this.handler.call(this.scope || this, this, e);
29641             }
29642         }
29643     },
29644     // private
29645     onMouseOver : function(e){
29646         if(!this.disabled){
29647             this.el.addClass("x-btn-over");
29648             this.fireEvent('mouseover', this, e);
29649         }
29650     },
29651     // private
29652     onMouseOut : function(e){
29653         if(!e.within(this.el,  true)){
29654             this.el.removeClass("x-btn-over");
29655             this.fireEvent('mouseout', this, e);
29656         }
29657     },
29658     // private
29659     onFocus : function(e){
29660         if(!this.disabled){
29661             this.el.addClass("x-btn-focus");
29662         }
29663     },
29664     // private
29665     onBlur : function(e){
29666         this.el.removeClass("x-btn-focus");
29667     },
29668     // private
29669     onMouseDown : function(e){
29670         if(!this.disabled && e.button == 0){
29671             this.el.addClass("x-btn-click");
29672             Roo.get(document).on('mouseup', this.onMouseUp, this);
29673         }
29674     },
29675     // private
29676     onMouseUp : function(e){
29677         if(e.button == 0){
29678             this.el.removeClass("x-btn-click");
29679             Roo.get(document).un('mouseup', this.onMouseUp, this);
29680         }
29681     },
29682     // private
29683     onMenuShow : function(e){
29684         this.el.addClass("x-btn-menu-active");
29685     },
29686     // private
29687     onMenuHide : function(e){
29688         this.el.removeClass("x-btn-menu-active");
29689     }   
29690 });
29691
29692 // Private utility class used by Button
29693 Roo.ButtonToggleMgr = function(){
29694    var groups = {};
29695    
29696    function toggleGroup(btn, state){
29697        if(state){
29698            var g = groups[btn.toggleGroup];
29699            for(var i = 0, l = g.length; i < l; i++){
29700                if(g[i] != btn){
29701                    g[i].toggle(false);
29702                }
29703            }
29704        }
29705    }
29706    
29707    return {
29708        register : function(btn){
29709            if(!btn.toggleGroup){
29710                return;
29711            }
29712            var g = groups[btn.toggleGroup];
29713            if(!g){
29714                g = groups[btn.toggleGroup] = [];
29715            }
29716            g.push(btn);
29717            btn.on("toggle", toggleGroup);
29718        },
29719        
29720        unregister : function(btn){
29721            if(!btn.toggleGroup){
29722                return;
29723            }
29724            var g = groups[btn.toggleGroup];
29725            if(g){
29726                g.remove(btn);
29727                btn.un("toggle", toggleGroup);
29728            }
29729        }
29730    };
29731 }();/*
29732  * Based on:
29733  * Ext JS Library 1.1.1
29734  * Copyright(c) 2006-2007, Ext JS, LLC.
29735  *
29736  * Originally Released Under LGPL - original licence link has changed is not relivant.
29737  *
29738  * Fork - LGPL
29739  * <script type="text/javascript">
29740  */
29741  
29742 /**
29743  * @class Roo.SplitButton
29744  * @extends Roo.Button
29745  * A split button that provides a built-in dropdown arrow that can fire an event separately from the default
29746  * click event of the button.  Typically this would be used to display a dropdown menu that provides additional
29747  * options to the primary button action, but any custom handler can provide the arrowclick implementation.
29748  * @cfg {Function} arrowHandler A function called when the arrow button is clicked (can be used instead of click event)
29749  * @cfg {String} arrowTooltip The title attribute of the arrow
29750  * @constructor
29751  * Create a new menu button
29752  * @param {String/HTMLElement/Element} renderTo The element to append the button to
29753  * @param {Object} config The config object
29754  */
29755 Roo.SplitButton = function(renderTo, config){
29756     Roo.SplitButton.superclass.constructor.call(this, renderTo, config);
29757     /**
29758      * @event arrowclick
29759      * Fires when this button's arrow is clicked
29760      * @param {SplitButton} this
29761      * @param {EventObject} e The click event
29762      */
29763     this.addEvents({"arrowclick":true});
29764 };
29765
29766 Roo.extend(Roo.SplitButton, Roo.Button, {
29767     render : function(renderTo){
29768         // this is one sweet looking template!
29769         var tpl = new Roo.Template(
29770             '<table cellspacing="0" class="x-btn-menu-wrap x-btn"><tr><td>',
29771             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-text-wrap"><tbody>',
29772             '<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>',
29773             "</tbody></table></td><td>",
29774             '<table cellspacing="0" class="x-btn-wrap x-btn-menu-arrow-wrap"><tbody>',
29775             '<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>',
29776             "</tbody></table></td></tr></table>"
29777         );
29778         var btn = tpl.append(renderTo, [this.text, this.type], true);
29779         var btnEl = btn.child("button");
29780         if(this.cls){
29781             btn.addClass(this.cls);
29782         }
29783         if(this.icon){
29784             btnEl.setStyle('background-image', 'url(' +this.icon +')');
29785         }
29786         if(this.iconCls){
29787             btnEl.addClass(this.iconCls);
29788             if(!this.cls){
29789                 btn.addClass(this.text ? 'x-btn-text-icon' : 'x-btn-icon');
29790             }
29791         }
29792         this.el = btn;
29793         if(this.handleMouseEvents){
29794             btn.on("mouseover", this.onMouseOver, this);
29795             btn.on("mouseout", this.onMouseOut, this);
29796             btn.on("mousedown", this.onMouseDown, this);
29797             btn.on("mouseup", this.onMouseUp, this);
29798         }
29799         btn.on(this.clickEvent, this.onClick, this);
29800         if(this.tooltip){
29801             if(typeof this.tooltip == 'object'){
29802                 Roo.QuickTips.tips(Roo.apply({
29803                       target: btnEl.id
29804                 }, this.tooltip));
29805             } else {
29806                 btnEl.dom[this.tooltipType] = this.tooltip;
29807             }
29808         }
29809         if(this.arrowTooltip){
29810             btn.child("button:nth(2)").dom[this.tooltipType] = this.arrowTooltip;
29811         }
29812         if(this.hidden){
29813             this.hide();
29814         }
29815         if(this.disabled){
29816             this.disable();
29817         }
29818         if(this.pressed){
29819             this.el.addClass("x-btn-pressed");
29820         }
29821         if(Roo.isIE && !Roo.isIE7){
29822             this.autoWidth.defer(1, this);
29823         }else{
29824             this.autoWidth();
29825         }
29826         if(this.menu){
29827             this.menu.on("show", this.onMenuShow, this);
29828             this.menu.on("hide", this.onMenuHide, this);
29829         }
29830         this.fireEvent('render', this);
29831     },
29832
29833     // private
29834     autoWidth : function(){
29835         if(this.el){
29836             var tbl = this.el.child("table:first");
29837             var tbl2 = this.el.child("table:last");
29838             this.el.setWidth("auto");
29839             tbl.setWidth("auto");
29840             if(Roo.isIE7 && Roo.isStrict){
29841                 var ib = this.el.child('button:first');
29842                 if(ib && ib.getWidth() > 20){
29843                     ib.clip();
29844                     ib.setWidth(Roo.util.TextMetrics.measure(ib, this.text).width+ib.getFrameWidth('lr'));
29845                 }
29846             }
29847             if(this.minWidth){
29848                 if(this.hidden){
29849                     this.el.beginMeasure();
29850                 }
29851                 if((tbl.getWidth()+tbl2.getWidth()) < this.minWidth){
29852                     tbl.setWidth(this.minWidth-tbl2.getWidth());
29853                 }
29854                 if(this.hidden){
29855                     this.el.endMeasure();
29856                 }
29857             }
29858             this.el.setWidth(tbl.getWidth()+tbl2.getWidth());
29859         } 
29860     },
29861     /**
29862      * Sets this button's click handler
29863      * @param {Function} handler The function to call when the button is clicked
29864      * @param {Object} scope (optional) Scope for the function passed above
29865      */
29866     setHandler : function(handler, scope){
29867         this.handler = handler;
29868         this.scope = scope;  
29869     },
29870     
29871     /**
29872      * Sets this button's arrow click handler
29873      * @param {Function} handler The function to call when the arrow is clicked
29874      * @param {Object} scope (optional) Scope for the function passed above
29875      */
29876     setArrowHandler : function(handler, scope){
29877         this.arrowHandler = handler;
29878         this.scope = scope;  
29879     },
29880     
29881     /**
29882      * Focus the button
29883      */
29884     focus : function(){
29885         if(this.el){
29886             this.el.child("button:first").focus();
29887         }
29888     },
29889
29890     // private
29891     onClick : function(e){
29892         e.preventDefault();
29893         if(!this.disabled){
29894             if(e.getTarget(".x-btn-menu-arrow-wrap")){
29895                 if(this.menu && !this.menu.isVisible()){
29896                     this.menu.show(this.el, this.menuAlign);
29897                 }
29898                 this.fireEvent("arrowclick", this, e);
29899                 if(this.arrowHandler){
29900                     this.arrowHandler.call(this.scope || this, this, e);
29901                 }
29902             }else{
29903                 this.fireEvent("click", this, e);
29904                 if(this.handler){
29905                     this.handler.call(this.scope || this, this, e);
29906                 }
29907             }
29908         }
29909     },
29910     // private
29911     onMouseDown : function(e){
29912         if(!this.disabled){
29913             Roo.fly(e.getTarget("table")).addClass("x-btn-click");
29914         }
29915     },
29916     // private
29917     onMouseUp : function(e){
29918         Roo.fly(e.getTarget("table")).removeClass("x-btn-click");
29919     }   
29920 });
29921
29922
29923 // backwards compat
29924 Roo.MenuButton = Roo.SplitButton;/*
29925  * Based on:
29926  * Ext JS Library 1.1.1
29927  * Copyright(c) 2006-2007, Ext JS, LLC.
29928  *
29929  * Originally Released Under LGPL - original licence link has changed is not relivant.
29930  *
29931  * Fork - LGPL
29932  * <script type="text/javascript">
29933  */
29934
29935 /**
29936  * @class Roo.Toolbar
29937  * Basic Toolbar class.
29938  * @constructor
29939  * Creates a new Toolbar
29940  * @param {Object} container The config object
29941  */ 
29942 Roo.Toolbar = function(container, buttons, config)
29943 {
29944     /// old consturctor format still supported..
29945     if(container instanceof Array){ // omit the container for later rendering
29946         buttons = container;
29947         config = buttons;
29948         container = null;
29949     }
29950     if (typeof(container) == 'object' && container.xtype) {
29951         config = container;
29952         container = config.container;
29953         buttons = config.buttons || []; // not really - use items!!
29954     }
29955     var xitems = [];
29956     if (config && config.items) {
29957         xitems = config.items;
29958         delete config.items;
29959     }
29960     Roo.apply(this, config);
29961     this.buttons = buttons;
29962     
29963     if(container){
29964         this.render(container);
29965     }
29966     this.xitems = xitems;
29967     Roo.each(xitems, function(b) {
29968         this.add(b);
29969     }, this);
29970     
29971 };
29972
29973 Roo.Toolbar.prototype = {
29974     /**
29975      * @cfg {Array} items
29976      * array of button configs or elements to add (will be converted to a MixedCollection)
29977      */
29978     
29979     /**
29980      * @cfg {String/HTMLElement/Element} container
29981      * The id or element that will contain the toolbar
29982      */
29983     // private
29984     render : function(ct){
29985         this.el = Roo.get(ct);
29986         if(this.cls){
29987             this.el.addClass(this.cls);
29988         }
29989         // using a table allows for vertical alignment
29990         // 100% width is needed by Safari...
29991         this.el.update('<div class="x-toolbar x-small-editor"><table cellspacing="0"><tr></tr></table></div>');
29992         this.tr = this.el.child("tr", true);
29993         var autoId = 0;
29994         this.items = new Roo.util.MixedCollection(false, function(o){
29995             return o.id || ("item" + (++autoId));
29996         });
29997         if(this.buttons){
29998             this.add.apply(this, this.buttons);
29999             delete this.buttons;
30000         }
30001     },
30002
30003     /**
30004      * Adds element(s) to the toolbar -- this function takes a variable number of 
30005      * arguments of mixed type and adds them to the toolbar.
30006      * @param {Mixed} arg1 The following types of arguments are all valid:<br />
30007      * <ul>
30008      * <li>{@link Roo.Toolbar.Button} config: A valid button config object (equivalent to {@link #addButton})</li>
30009      * <li>HtmlElement: Any standard HTML element (equivalent to {@link #addElement})</li>
30010      * <li>Field: Any form field (equivalent to {@link #addField})</li>
30011      * <li>Item: Any subclass of {@link Roo.Toolbar.Item} (equivalent to {@link #addItem})</li>
30012      * <li>String: Any generic string (gets wrapped in a {@link Roo.Toolbar.TextItem}, equivalent to {@link #addText}).
30013      * Note that there are a few special strings that are treated differently as explained nRoo.</li>
30014      * <li>'separator' or '-': Creates a separator element (equivalent to {@link #addSeparator})</li>
30015      * <li>' ': Creates a spacer element (equivalent to {@link #addSpacer})</li>
30016      * <li>'->': Creates a fill element (equivalent to {@link #addFill})</li>
30017      * </ul>
30018      * @param {Mixed} arg2
30019      * @param {Mixed} etc.
30020      */
30021     add : function(){
30022         var a = arguments, l = a.length;
30023         for(var i = 0; i < l; i++){
30024             this._add(a[i]);
30025         }
30026     },
30027     // private..
30028     _add : function(el) {
30029         
30030         if (el.xtype) {
30031             el = Roo.factory(el, typeof(Roo.Toolbar[el.xtype]) == 'undefined' ? Roo.form : Roo.Toolbar);
30032         }
30033         
30034         if (el.applyTo){ // some kind of form field
30035             return this.addField(el);
30036         } 
30037         if (el.render){ // some kind of Toolbar.Item
30038             return this.addItem(el);
30039         }
30040         if (typeof el == "string"){ // string
30041             if(el == "separator" || el == "-"){
30042                 return this.addSeparator();
30043             }
30044             if (el == " "){
30045                 return this.addSpacer();
30046             }
30047             if(el == "->"){
30048                 return this.addFill();
30049             }
30050             return this.addText(el);
30051             
30052         }
30053         if(el.tagName){ // element
30054             return this.addElement(el);
30055         }
30056         if(typeof el == "object"){ // must be button config?
30057             return this.addButton(el);
30058         }
30059         // and now what?!?!
30060         return false;
30061         
30062     },
30063     
30064     /**
30065      * Add an Xtype element
30066      * @param {Object} xtype Xtype Object
30067      * @return {Object} created Object
30068      */
30069     addxtype : function(e){
30070         return this.add(e);  
30071     },
30072     
30073     /**
30074      * Returns the Element for this toolbar.
30075      * @return {Roo.Element}
30076      */
30077     getEl : function(){
30078         return this.el;  
30079     },
30080     
30081     /**
30082      * Adds a separator
30083      * @return {Roo.Toolbar.Item} The separator item
30084      */
30085     addSeparator : function(){
30086         return this.addItem(new Roo.Toolbar.Separator());
30087     },
30088
30089     /**
30090      * Adds a spacer element
30091      * @return {Roo.Toolbar.Spacer} The spacer item
30092      */
30093     addSpacer : function(){
30094         return this.addItem(new Roo.Toolbar.Spacer());
30095     },
30096
30097     /**
30098      * Adds a fill element that forces subsequent additions to the right side of the toolbar
30099      * @return {Roo.Toolbar.Fill} The fill item
30100      */
30101     addFill : function(){
30102         return this.addItem(new Roo.Toolbar.Fill());
30103     },
30104
30105     /**
30106      * Adds any standard HTML element to the toolbar
30107      * @param {String/HTMLElement/Element} el The element or id of the element to add
30108      * @return {Roo.Toolbar.Item} The element's item
30109      */
30110     addElement : function(el){
30111         return this.addItem(new Roo.Toolbar.Item(el));
30112     },
30113     /**
30114      * Collection of items on the toolbar.. (only Toolbar Items, so use fields to retrieve fields)
30115      * @type Roo.util.MixedCollection  
30116      */
30117     items : false,
30118      
30119     /**
30120      * Adds any Toolbar.Item or subclass
30121      * @param {Roo.Toolbar.Item} item
30122      * @return {Roo.Toolbar.Item} The item
30123      */
30124     addItem : function(item){
30125         var td = this.nextBlock();
30126         item.render(td);
30127         this.items.add(item);
30128         return item;
30129     },
30130     
30131     /**
30132      * Adds a button (or buttons). See {@link Roo.Toolbar.Button} for more info on the config.
30133      * @param {Object/Array} config A button config or array of configs
30134      * @return {Roo.Toolbar.Button/Array}
30135      */
30136     addButton : function(config){
30137         if(config instanceof Array){
30138             var buttons = [];
30139             for(var i = 0, len = config.length; i < len; i++) {
30140                 buttons.push(this.addButton(config[i]));
30141             }
30142             return buttons;
30143         }
30144         var b = config;
30145         if(!(config instanceof Roo.Toolbar.Button)){
30146             b = config.split ?
30147                 new Roo.Toolbar.SplitButton(config) :
30148                 new Roo.Toolbar.Button(config);
30149         }
30150         var td = this.nextBlock();
30151         b.render(td);
30152         this.items.add(b);
30153         return b;
30154     },
30155     
30156     /**
30157      * Adds text to the toolbar
30158      * @param {String} text The text to add
30159      * @return {Roo.Toolbar.Item} The element's item
30160      */
30161     addText : function(text){
30162         return this.addItem(new Roo.Toolbar.TextItem(text));
30163     },
30164     
30165     /**
30166      * Inserts any {@link Roo.Toolbar.Item}/{@link Roo.Toolbar.Button} at the specified index.
30167      * @param {Number} index The index where the item is to be inserted
30168      * @param {Object/Roo.Toolbar.Item/Roo.Toolbar.Button (may be Array)} item The button, or button config object to be inserted.
30169      * @return {Roo.Toolbar.Button/Item}
30170      */
30171     insertButton : function(index, item){
30172         if(item instanceof Array){
30173             var buttons = [];
30174             for(var i = 0, len = item.length; i < len; i++) {
30175                buttons.push(this.insertButton(index + i, item[i]));
30176             }
30177             return buttons;
30178         }
30179         if (!(item instanceof Roo.Toolbar.Button)){
30180            item = new Roo.Toolbar.Button(item);
30181         }
30182         var td = document.createElement("td");
30183         this.tr.insertBefore(td, this.tr.childNodes[index]);
30184         item.render(td);
30185         this.items.insert(index, item);
30186         return item;
30187     },
30188     
30189     /**
30190      * Adds a new element to the toolbar from the passed {@link Roo.DomHelper} config.
30191      * @param {Object} config
30192      * @return {Roo.Toolbar.Item} The element's item
30193      */
30194     addDom : function(config, returnEl){
30195         var td = this.nextBlock();
30196         Roo.DomHelper.overwrite(td, config);
30197         var ti = new Roo.Toolbar.Item(td.firstChild);
30198         ti.render(td);
30199         this.items.add(ti);
30200         return ti;
30201     },
30202
30203     /**
30204      * Collection of fields on the toolbar.. usefull for quering (value is false if there are no fields)
30205      * @type Roo.util.MixedCollection  
30206      */
30207     fields : false,
30208     
30209     /**
30210      * Adds a dynamically rendered Roo.form field (TextField, ComboBox, etc).
30211      * Note: the field should not have been rendered yet. For a field that has already been
30212      * rendered, use {@link #addElement}.
30213      * @param {Roo.form.Field} field
30214      * @return {Roo.ToolbarItem}
30215      */
30216      
30217       
30218     addField : function(field) {
30219         if (!this.fields) {
30220             var autoId = 0;
30221             this.fields = new Roo.util.MixedCollection(false, function(o){
30222                 return o.id || ("item" + (++autoId));
30223             });
30224
30225         }
30226         
30227         var td = this.nextBlock();
30228         field.render(td);
30229         var ti = new Roo.Toolbar.Item(td.firstChild);
30230         ti.render(td);
30231         this.items.add(ti);
30232         this.fields.add(field);
30233         return ti;
30234     },
30235     /**
30236      * Hide the toolbar
30237      * @method hide
30238      */
30239      
30240       
30241     hide : function()
30242     {
30243         this.el.child('div').setVisibilityMode(Roo.Element.DISPLAY);
30244         this.el.child('div').hide();
30245     },
30246     /**
30247      * Show the toolbar
30248      * @method show
30249      */
30250     show : function()
30251     {
30252         this.el.child('div').show();
30253     },
30254       
30255     // private
30256     nextBlock : function(){
30257         var td = document.createElement("td");
30258         this.tr.appendChild(td);
30259         return td;
30260     },
30261
30262     // private
30263     destroy : function(){
30264         if(this.items){ // rendered?
30265             Roo.destroy.apply(Roo, this.items.items);
30266         }
30267         if(this.fields){ // rendered?
30268             Roo.destroy.apply(Roo, this.fields.items);
30269         }
30270         Roo.Element.uncache(this.el, this.tr);
30271     }
30272 };
30273
30274 /**
30275  * @class Roo.Toolbar.Item
30276  * The base class that other classes should extend in order to get some basic common toolbar item functionality.
30277  * @constructor
30278  * Creates a new Item
30279  * @param {HTMLElement} el 
30280  */
30281 Roo.Toolbar.Item = function(el){
30282     var cfg = {};
30283     if (typeof (el.xtype) != 'undefined') {
30284         cfg = el;
30285         el = cfg.el;
30286     }
30287     
30288     this.el = Roo.getDom(el);
30289     this.id = Roo.id(this.el);
30290     this.hidden = false;
30291     
30292     this.addEvents({
30293          /**
30294              * @event render
30295              * Fires when the button is rendered
30296              * @param {Button} this
30297              */
30298         'render': true
30299     });
30300     Roo.Toolbar.Item.superclass.constructor.call(this,cfg);
30301 };
30302 Roo.extend(Roo.Toolbar.Item, Roo.util.Observable, {
30303 //Roo.Toolbar.Item.prototype = {
30304     
30305     /**
30306      * Get this item's HTML Element
30307      * @return {HTMLElement}
30308      */
30309     getEl : function(){
30310        return this.el;  
30311     },
30312
30313     // private
30314     render : function(td){
30315         
30316          this.td = td;
30317         td.appendChild(this.el);
30318         
30319         this.fireEvent('render', this);
30320     },
30321     
30322     /**
30323      * Removes and destroys this item.
30324      */
30325     destroy : function(){
30326         this.td.parentNode.removeChild(this.td);
30327     },
30328     
30329     /**
30330      * Shows this item.
30331      */
30332     show: function(){
30333         this.hidden = false;
30334         this.td.style.display = "";
30335     },
30336     
30337     /**
30338      * Hides this item.
30339      */
30340     hide: function(){
30341         this.hidden = true;
30342         this.td.style.display = "none";
30343     },
30344     
30345     /**
30346      * Convenience function for boolean show/hide.
30347      * @param {Boolean} visible true to show/false to hide
30348      */
30349     setVisible: function(visible){
30350         if(visible) {
30351             this.show();
30352         }else{
30353             this.hide();
30354         }
30355     },
30356     
30357     /**
30358      * Try to focus this item.
30359      */
30360     focus : function(){
30361         Roo.fly(this.el).focus();
30362     },
30363     
30364     /**
30365      * Disables this item.
30366      */
30367     disable : function(){
30368         Roo.fly(this.td).addClass("x-item-disabled");
30369         this.disabled = true;
30370         this.el.disabled = true;
30371     },
30372     
30373     /**
30374      * Enables this item.
30375      */
30376     enable : function(){
30377         Roo.fly(this.td).removeClass("x-item-disabled");
30378         this.disabled = false;
30379         this.el.disabled = false;
30380     }
30381 });
30382
30383
30384 /**
30385  * @class Roo.Toolbar.Separator
30386  * @extends Roo.Toolbar.Item
30387  * A simple toolbar separator class
30388  * @constructor
30389  * Creates a new Separator
30390  */
30391 Roo.Toolbar.Separator = function(cfg){
30392     
30393     var s = document.createElement("span");
30394     s.className = "ytb-sep";
30395     if (cfg) {
30396         cfg.el = s;
30397     }
30398     
30399     Roo.Toolbar.Separator.superclass.constructor.call(this, cfg || s);
30400 };
30401 Roo.extend(Roo.Toolbar.Separator, Roo.Toolbar.Item, {
30402     enable:Roo.emptyFn,
30403     disable:Roo.emptyFn,
30404     focus:Roo.emptyFn
30405 });
30406
30407 /**
30408  * @class Roo.Toolbar.Spacer
30409  * @extends Roo.Toolbar.Item
30410  * A simple element that adds extra horizontal space to a toolbar.
30411  * @constructor
30412  * Creates a new Spacer
30413  */
30414 Roo.Toolbar.Spacer = function(cfg){
30415     var s = document.createElement("div");
30416     s.className = "ytb-spacer";
30417     if (cfg) {
30418         cfg.el = s;
30419     }
30420     Roo.Toolbar.Spacer.superclass.constructor.call(this, cfg || s);
30421 };
30422 Roo.extend(Roo.Toolbar.Spacer, Roo.Toolbar.Item, {
30423     enable:Roo.emptyFn,
30424     disable:Roo.emptyFn,
30425     focus:Roo.emptyFn
30426 });
30427
30428 /**
30429  * @class Roo.Toolbar.Fill
30430  * @extends Roo.Toolbar.Spacer
30431  * A simple element that adds a greedy (100% width) horizontal space to a toolbar.
30432  * @constructor
30433  * Creates a new Spacer
30434  */
30435 Roo.Toolbar.Fill = Roo.extend(Roo.Toolbar.Spacer, {
30436     // private
30437     render : function(td){
30438         td.style.width = '100%';
30439         Roo.Toolbar.Fill.superclass.render.call(this, td);
30440     }
30441 });
30442
30443 /**
30444  * @class Roo.Toolbar.TextItem
30445  * @extends Roo.Toolbar.Item
30446  * A simple class that renders text directly into a toolbar.
30447  * @constructor
30448  * Creates a new TextItem
30449  * @param {String} text
30450  */
30451 Roo.Toolbar.TextItem = function(cfg){
30452     var  text = cfg || "";
30453     if (typeof(cfg) == 'object') {
30454         text = cfg.text || "";
30455     }  else {
30456         cfg = null;
30457     }
30458     var s = document.createElement("span");
30459     s.className = "ytb-text";
30460     s.innerHTML = text;
30461     if (cfg) {
30462         cfg.el  = s;
30463     }
30464     
30465     Roo.Toolbar.TextItem.superclass.constructor.call(this, cfg ||  s);
30466 };
30467 Roo.extend(Roo.Toolbar.TextItem, Roo.Toolbar.Item, {
30468     
30469      
30470     enable:Roo.emptyFn,
30471     disable:Roo.emptyFn,
30472     focus:Roo.emptyFn
30473 });
30474
30475 /**
30476  * @class Roo.Toolbar.Button
30477  * @extends Roo.Button
30478  * A button that renders into a toolbar.
30479  * @constructor
30480  * Creates a new Button
30481  * @param {Object} config A standard {@link Roo.Button} config object
30482  */
30483 Roo.Toolbar.Button = function(config){
30484     Roo.Toolbar.Button.superclass.constructor.call(this, null, config);
30485 };
30486 Roo.extend(Roo.Toolbar.Button, Roo.Button, {
30487     render : function(td){
30488         this.td = td;
30489         Roo.Toolbar.Button.superclass.render.call(this, td);
30490     },
30491     
30492     /**
30493      * Removes and destroys this button
30494      */
30495     destroy : function(){
30496         Roo.Toolbar.Button.superclass.destroy.call(this);
30497         this.td.parentNode.removeChild(this.td);
30498     },
30499     
30500     /**
30501      * Shows this button
30502      */
30503     show: function(){
30504         this.hidden = false;
30505         this.td.style.display = "";
30506     },
30507     
30508     /**
30509      * Hides this button
30510      */
30511     hide: function(){
30512         this.hidden = true;
30513         this.td.style.display = "none";
30514     },
30515
30516     /**
30517      * Disables this item
30518      */
30519     disable : function(){
30520         Roo.fly(this.td).addClass("x-item-disabled");
30521         this.disabled = true;
30522     },
30523
30524     /**
30525      * Enables this item
30526      */
30527     enable : function(){
30528         Roo.fly(this.td).removeClass("x-item-disabled");
30529         this.disabled = false;
30530     }
30531 });
30532 // backwards compat
30533 Roo.ToolbarButton = Roo.Toolbar.Button;
30534
30535 /**
30536  * @class Roo.Toolbar.SplitButton
30537  * @extends Roo.SplitButton
30538  * A menu button that renders into a toolbar.
30539  * @constructor
30540  * Creates a new SplitButton
30541  * @param {Object} config A standard {@link Roo.SplitButton} config object
30542  */
30543 Roo.Toolbar.SplitButton = function(config){
30544     Roo.Toolbar.SplitButton.superclass.constructor.call(this, null, config);
30545 };
30546 Roo.extend(Roo.Toolbar.SplitButton, Roo.SplitButton, {
30547     render : function(td){
30548         this.td = td;
30549         Roo.Toolbar.SplitButton.superclass.render.call(this, td);
30550     },
30551     
30552     /**
30553      * Removes and destroys this button
30554      */
30555     destroy : function(){
30556         Roo.Toolbar.SplitButton.superclass.destroy.call(this);
30557         this.td.parentNode.removeChild(this.td);
30558     },
30559     
30560     /**
30561      * Shows this button
30562      */
30563     show: function(){
30564         this.hidden = false;
30565         this.td.style.display = "";
30566     },
30567     
30568     /**
30569      * Hides this button
30570      */
30571     hide: function(){
30572         this.hidden = true;
30573         this.td.style.display = "none";
30574     }
30575 });
30576
30577 // backwards compat
30578 Roo.Toolbar.MenuButton = Roo.Toolbar.SplitButton;/*
30579  * Based on:
30580  * Ext JS Library 1.1.1
30581  * Copyright(c) 2006-2007, Ext JS, LLC.
30582  *
30583  * Originally Released Under LGPL - original licence link has changed is not relivant.
30584  *
30585  * Fork - LGPL
30586  * <script type="text/javascript">
30587  */
30588  
30589 /**
30590  * @class Roo.PagingToolbar
30591  * @extends Roo.Toolbar
30592  * A specialized toolbar that is bound to a {@link Roo.data.Store} and provides automatic paging controls.
30593  * @constructor
30594  * Create a new PagingToolbar
30595  * @param {Object} config The config object
30596  */
30597 Roo.PagingToolbar = function(el, ds, config)
30598 {
30599     // old args format still supported... - xtype is prefered..
30600     if (typeof(el) == 'object' && el.xtype) {
30601         // created from xtype...
30602         config = el;
30603         ds = el.dataSource;
30604         el = config.container;
30605     }
30606     var items = [];
30607     if (config.items) {
30608         items = config.items;
30609         config.items = [];
30610     }
30611     
30612     Roo.PagingToolbar.superclass.constructor.call(this, el, null, config);
30613     this.ds = ds;
30614     this.cursor = 0;
30615     this.renderButtons(this.el);
30616     this.bind(ds);
30617     
30618     // supprot items array.
30619    
30620     Roo.each(items, function(e) {
30621         this.add(Roo.factory(e));
30622     },this);
30623     
30624 };
30625
30626 Roo.extend(Roo.PagingToolbar, Roo.Toolbar, {
30627     /**
30628      * @cfg {Roo.data.Store} dataSource
30629      * The underlying data store providing the paged data
30630      */
30631     /**
30632      * @cfg {String/HTMLElement/Element} container
30633      * container The id or element that will contain the toolbar
30634      */
30635     /**
30636      * @cfg {Boolean} displayInfo
30637      * True to display the displayMsg (defaults to false)
30638      */
30639     /**
30640      * @cfg {Number} pageSize
30641      * The number of records to display per page (defaults to 20)
30642      */
30643     pageSize: 20,
30644     /**
30645      * @cfg {String} displayMsg
30646      * The paging status message to display (defaults to "Displaying {start} - {end} of {total}")
30647      */
30648     displayMsg : 'Displaying {0} - {1} of {2}',
30649     /**
30650      * @cfg {String} emptyMsg
30651      * The message to display when no records are found (defaults to "No data to display")
30652      */
30653     emptyMsg : 'No data to display',
30654     /**
30655      * Customizable piece of the default paging text (defaults to "Page")
30656      * @type String
30657      */
30658     beforePageText : "Page",
30659     /**
30660      * Customizable piece of the default paging text (defaults to "of %0")
30661      * @type String
30662      */
30663     afterPageText : "of {0}",
30664     /**
30665      * Customizable piece of the default paging text (defaults to "First Page")
30666      * @type String
30667      */
30668     firstText : "First Page",
30669     /**
30670      * Customizable piece of the default paging text (defaults to "Previous Page")
30671      * @type String
30672      */
30673     prevText : "Previous Page",
30674     /**
30675      * Customizable piece of the default paging text (defaults to "Next Page")
30676      * @type String
30677      */
30678     nextText : "Next Page",
30679     /**
30680      * Customizable piece of the default paging text (defaults to "Last Page")
30681      * @type String
30682      */
30683     lastText : "Last Page",
30684     /**
30685      * Customizable piece of the default paging text (defaults to "Refresh")
30686      * @type String
30687      */
30688     refreshText : "Refresh",
30689
30690     // private
30691     renderButtons : function(el){
30692         Roo.PagingToolbar.superclass.render.call(this, el);
30693         this.first = this.addButton({
30694             tooltip: this.firstText,
30695             cls: "x-btn-icon x-grid-page-first",
30696             disabled: true,
30697             handler: this.onClick.createDelegate(this, ["first"])
30698         });
30699         this.prev = this.addButton({
30700             tooltip: this.prevText,
30701             cls: "x-btn-icon x-grid-page-prev",
30702             disabled: true,
30703             handler: this.onClick.createDelegate(this, ["prev"])
30704         });
30705         //this.addSeparator();
30706         this.add(this.beforePageText);
30707         this.field = Roo.get(this.addDom({
30708            tag: "input",
30709            type: "text",
30710            size: "3",
30711            value: "1",
30712            cls: "x-grid-page-number"
30713         }).el);
30714         this.field.on("keydown", this.onPagingKeydown, this);
30715         this.field.on("focus", function(){this.dom.select();});
30716         this.afterTextEl = this.addText(String.format(this.afterPageText, 1));
30717         this.field.setHeight(18);
30718         //this.addSeparator();
30719         this.next = this.addButton({
30720             tooltip: this.nextText,
30721             cls: "x-btn-icon x-grid-page-next",
30722             disabled: true,
30723             handler: this.onClick.createDelegate(this, ["next"])
30724         });
30725         this.last = this.addButton({
30726             tooltip: this.lastText,
30727             cls: "x-btn-icon x-grid-page-last",
30728             disabled: true,
30729             handler: this.onClick.createDelegate(this, ["last"])
30730         });
30731         //this.addSeparator();
30732         this.loading = this.addButton({
30733             tooltip: this.refreshText,
30734             cls: "x-btn-icon x-grid-loading",
30735             handler: this.onClick.createDelegate(this, ["refresh"])
30736         });
30737
30738         if(this.displayInfo){
30739             this.displayEl = Roo.fly(this.el.dom.firstChild).createChild({cls:'x-paging-info'});
30740         }
30741     },
30742
30743     // private
30744     updateInfo : function(){
30745         if(this.displayEl){
30746             var count = this.ds.getCount();
30747             var msg = count == 0 ?
30748                 this.emptyMsg :
30749                 String.format(
30750                     this.displayMsg,
30751                     this.cursor+1, this.cursor+count, this.ds.getTotalCount()    
30752                 );
30753             this.displayEl.update(msg);
30754         }
30755     },
30756
30757     // private
30758     onLoad : function(ds, r, o){
30759        this.cursor = o.params ? o.params.start : 0;
30760        var d = this.getPageData(), ap = d.activePage, ps = d.pages;
30761
30762        this.afterTextEl.el.innerHTML = String.format(this.afterPageText, d.pages);
30763        this.field.dom.value = ap;
30764        this.first.setDisabled(ap == 1);
30765        this.prev.setDisabled(ap == 1);
30766        this.next.setDisabled(ap == ps);
30767        this.last.setDisabled(ap == ps);
30768        this.loading.enable();
30769        this.updateInfo();
30770     },
30771
30772     // private
30773     getPageData : function(){
30774         var total = this.ds.getTotalCount();
30775         return {
30776             total : total,
30777             activePage : Math.ceil((this.cursor+this.pageSize)/this.pageSize),
30778             pages :  total < this.pageSize ? 1 : Math.ceil(total/this.pageSize)
30779         };
30780     },
30781
30782     // private
30783     onLoadError : function(){
30784         this.loading.enable();
30785     },
30786
30787     // private
30788     onPagingKeydown : function(e){
30789         var k = e.getKey();
30790         var d = this.getPageData();
30791         if(k == e.RETURN){
30792             var v = this.field.dom.value, pageNum;
30793             if(!v || isNaN(pageNum = parseInt(v, 10))){
30794                 this.field.dom.value = d.activePage;
30795                 return;
30796             }
30797             pageNum = Math.min(Math.max(1, pageNum), d.pages) - 1;
30798             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30799             e.stopEvent();
30800         }
30801         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))
30802         {
30803           var pageNum = (k == e.HOME || (k == e.DOWN && e.ctrlKey) || (k == e.LEFT && e.ctrlKey) || (k == e.PAGEDOWN && e.ctrlKey)) ? 1 : d.pages;
30804           this.field.dom.value = pageNum;
30805           this.ds.load({params:{start: (pageNum - 1) * this.pageSize, limit: this.pageSize}});
30806           e.stopEvent();
30807         }
30808         else if(k == e.UP || k == e.RIGHT || k == e.PAGEUP || k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN)
30809         {
30810           var v = this.field.dom.value, pageNum; 
30811           var increment = (e.shiftKey) ? 10 : 1;
30812           if(k == e.DOWN || k == e.LEFT || k == e.PAGEDOWN) {
30813             increment *= -1;
30814           }
30815           if(!v || isNaN(pageNum = parseInt(v, 10))) {
30816             this.field.dom.value = d.activePage;
30817             return;
30818           }
30819           else if(parseInt(v, 10) + increment >= 1 & parseInt(v, 10) + increment <= d.pages)
30820           {
30821             this.field.dom.value = parseInt(v, 10) + increment;
30822             pageNum = Math.min(Math.max(1, pageNum + increment), d.pages) - 1;
30823             this.ds.load({params:{start: pageNum * this.pageSize, limit: this.pageSize}});
30824           }
30825           e.stopEvent();
30826         }
30827     },
30828
30829     // private
30830     beforeLoad : function(){
30831         if(this.loading){
30832             this.loading.disable();
30833         }
30834     },
30835
30836     // private
30837     onClick : function(which){
30838         var ds = this.ds;
30839         switch(which){
30840             case "first":
30841                 ds.load({params:{start: 0, limit: this.pageSize}});
30842             break;
30843             case "prev":
30844                 ds.load({params:{start: Math.max(0, this.cursor-this.pageSize), limit: this.pageSize}});
30845             break;
30846             case "next":
30847                 ds.load({params:{start: this.cursor+this.pageSize, limit: this.pageSize}});
30848             break;
30849             case "last":
30850                 var total = ds.getTotalCount();
30851                 var extra = total % this.pageSize;
30852                 var lastStart = extra ? (total - extra) : total-this.pageSize;
30853                 ds.load({params:{start: lastStart, limit: this.pageSize}});
30854             break;
30855             case "refresh":
30856                 ds.load({params:{start: this.cursor, limit: this.pageSize}});
30857             break;
30858         }
30859     },
30860
30861     /**
30862      * Unbinds the paging toolbar from the specified {@link Roo.data.Store}
30863      * @param {Roo.data.Store} store The data store to unbind
30864      */
30865     unbind : function(ds){
30866         ds.un("beforeload", this.beforeLoad, this);
30867         ds.un("load", this.onLoad, this);
30868         ds.un("loadexception", this.onLoadError, this);
30869         ds.un("remove", this.updateInfo, this);
30870         ds.un("add", this.updateInfo, this);
30871         this.ds = undefined;
30872     },
30873
30874     /**
30875      * Binds the paging toolbar to the specified {@link Roo.data.Store}
30876      * @param {Roo.data.Store} store The data store to bind
30877      */
30878     bind : function(ds){
30879         ds.on("beforeload", this.beforeLoad, this);
30880         ds.on("load", this.onLoad, this);
30881         ds.on("loadexception", this.onLoadError, this);
30882         ds.on("remove", this.updateInfo, this);
30883         ds.on("add", this.updateInfo, this);
30884         this.ds = ds;
30885     }
30886 });/*
30887  * Based on:
30888  * Ext JS Library 1.1.1
30889  * Copyright(c) 2006-2007, Ext JS, LLC.
30890  *
30891  * Originally Released Under LGPL - original licence link has changed is not relivant.
30892  *
30893  * Fork - LGPL
30894  * <script type="text/javascript">
30895  */
30896
30897 /**
30898  * @class Roo.Resizable
30899  * @extends Roo.util.Observable
30900  * <p>Applies drag handles to an element to make it resizable. The drag handles are inserted into the element
30901  * and positioned absolute. Some elements, such as a textarea or image, don't support this. To overcome that, you can wrap
30902  * 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
30903  * the element will be wrapped for you automatically.</p>
30904  * <p>Here is the list of valid resize handles:</p>
30905  * <pre>
30906 Value   Description
30907 ------  -------------------
30908  'n'     north
30909  's'     south
30910  'e'     east
30911  'w'     west
30912  'nw'    northwest
30913  'sw'    southwest
30914  'se'    southeast
30915  'ne'    northeast
30916  'hd'    horizontal drag
30917  'all'   all
30918 </pre>
30919  * <p>Here's an example showing the creation of a typical Resizable:</p>
30920  * <pre><code>
30921 var resizer = new Roo.Resizable("element-id", {
30922     handles: 'all',
30923     minWidth: 200,
30924     minHeight: 100,
30925     maxWidth: 500,
30926     maxHeight: 400,
30927     pinned: true
30928 });
30929 resizer.on("resize", myHandler);
30930 </code></pre>
30931  * <p>To hide a particular handle, set its display to none in CSS, or through script:<br>
30932  * resizer.east.setDisplayed(false);</p>
30933  * @cfg {Boolean/String/Element} resizeChild True to resize the first child, or id/element to resize (defaults to false)
30934  * @cfg {Array/String} adjustments String "auto" or an array [width, height] with values to be <b>added</b> to the
30935  * resize operation's new size (defaults to [0, 0])
30936  * @cfg {Number} minWidth The minimum width for the element (defaults to 5)
30937  * @cfg {Number} minHeight The minimum height for the element (defaults to 5)
30938  * @cfg {Number} maxWidth The maximum width for the element (defaults to 10000)
30939  * @cfg {Number} maxHeight The maximum height for the element (defaults to 10000)
30940  * @cfg {Boolean} enabled False to disable resizing (defaults to true)
30941  * @cfg {Boolean} wrap True to wrap an element with a div if needed (required for textareas and images, defaults to false)
30942  * @cfg {Number} width The width of the element in pixels (defaults to null)
30943  * @cfg {Number} height The height of the element in pixels (defaults to null)
30944  * @cfg {Boolean} animate True to animate the resize (not compatible with dynamic sizing, defaults to false)
30945  * @cfg {Number} duration Animation duration if animate = true (defaults to .35)
30946  * @cfg {Boolean} dynamic True to resize the element while dragging instead of using a proxy (defaults to false)
30947  * @cfg {String} handles String consisting of the resize handles to display (defaults to undefined)
30948  * @cfg {Boolean} multiDirectional <b>Deprecated</b>.  The old style of adding multi-direction resize handles, deprecated
30949  * in favor of the handles config option (defaults to false)
30950  * @cfg {Boolean} disableTrackOver True to disable mouse tracking. This is only applied at config time. (defaults to false)
30951  * @cfg {String} easing Animation easing if animate = true (defaults to 'easingOutStrong')
30952  * @cfg {Number} widthIncrement The increment to snap the width resize in pixels (dynamic must be true, defaults to 0)
30953  * @cfg {Number} heightIncrement The increment to snap the height resize in pixels (dynamic must be true, defaults to 0)
30954  * @cfg {Boolean} pinned True to ensure that the resize handles are always visible, false to display them only when the
30955  * user mouses over the resizable borders. This is only applied at config time. (defaults to false)
30956  * @cfg {Boolean} preserveRatio True to preserve the original ratio between height and width during resize (defaults to false)
30957  * @cfg {Boolean} transparent True for transparent handles. This is only applied at config time. (defaults to false)
30958  * @cfg {Number} minX The minimum allowed page X for the element (only used for west resizing, defaults to 0)
30959  * @cfg {Number} minY The minimum allowed page Y for the element (only used for north resizing, defaults to 0)
30960  * @cfg {Boolean} draggable Convenience to initialize drag drop (defaults to false)
30961  * @constructor
30962  * Create a new resizable component
30963  * @param {String/HTMLElement/Roo.Element} el The id or element to resize
30964  * @param {Object} config configuration options
30965   */
30966 Roo.Resizable = function(el, config)
30967 {
30968     this.el = Roo.get(el);
30969
30970     if(config && config.wrap){
30971         config.resizeChild = this.el;
30972         this.el = this.el.wrap(typeof config.wrap == "object" ? config.wrap : {cls:"xresizable-wrap"});
30973         this.el.id = this.el.dom.id = config.resizeChild.id + "-rzwrap";
30974         this.el.setStyle("overflow", "hidden");
30975         this.el.setPositioning(config.resizeChild.getPositioning());
30976         config.resizeChild.clearPositioning();
30977         if(!config.width || !config.height){
30978             var csize = config.resizeChild.getSize();
30979             this.el.setSize(csize.width, csize.height);
30980         }
30981         if(config.pinned && !config.adjustments){
30982             config.adjustments = "auto";
30983         }
30984     }
30985
30986     this.proxy = this.el.createProxy({tag: "div", cls: "x-resizable-proxy", id: this.el.id + "-rzproxy"});
30987     this.proxy.unselectable();
30988     this.proxy.enableDisplayMode('block');
30989
30990     Roo.apply(this, config);
30991
30992     if(this.pinned){
30993         this.disableTrackOver = true;
30994         this.el.addClass("x-resizable-pinned");
30995     }
30996     // if the element isn't positioned, make it relative
30997     var position = this.el.getStyle("position");
30998     if(position != "absolute" && position != "fixed"){
30999         this.el.setStyle("position", "relative");
31000     }
31001     if(!this.handles){ // no handles passed, must be legacy style
31002         this.handles = 's,e,se';
31003         if(this.multiDirectional){
31004             this.handles += ',n,w';
31005         }
31006     }
31007     if(this.handles == "all"){
31008         this.handles = "n s e w ne nw se sw";
31009     }
31010     var hs = this.handles.split(/\s*?[,;]\s*?| /);
31011     var ps = Roo.Resizable.positions;
31012     for(var i = 0, len = hs.length; i < len; i++){
31013         if(hs[i] && ps[hs[i]]){
31014             var pos = ps[hs[i]];
31015             this[pos] = new Roo.Resizable.Handle(this, pos, this.disableTrackOver, this.transparent);
31016         }
31017     }
31018     // legacy
31019     this.corner = this.southeast;
31020     
31021     // updateBox = the box can move..
31022     if(this.handles.indexOf("n") != -1 || this.handles.indexOf("w") != -1 || this.handles.indexOf("hd") != -1) {
31023         this.updateBox = true;
31024     }
31025
31026     this.activeHandle = null;
31027
31028     if(this.resizeChild){
31029         if(typeof this.resizeChild == "boolean"){
31030             this.resizeChild = Roo.get(this.el.dom.firstChild, true);
31031         }else{
31032             this.resizeChild = Roo.get(this.resizeChild, true);
31033         }
31034     }
31035     
31036     if(this.adjustments == "auto"){
31037         var rc = this.resizeChild;
31038         var hw = this.west, he = this.east, hn = this.north, hs = this.south;
31039         if(rc && (hw || hn)){
31040             rc.position("relative");
31041             rc.setLeft(hw ? hw.el.getWidth() : 0);
31042             rc.setTop(hn ? hn.el.getHeight() : 0);
31043         }
31044         this.adjustments = [
31045             (he ? -he.el.getWidth() : 0) + (hw ? -hw.el.getWidth() : 0),
31046             (hn ? -hn.el.getHeight() : 0) + (hs ? -hs.el.getHeight() : 0) -1
31047         ];
31048     }
31049
31050     if(this.draggable){
31051         this.dd = this.dynamic ?
31052             this.el.initDD(null) : this.el.initDDProxy(null, {dragElId: this.proxy.id});
31053         this.dd.setHandleElId(this.resizeChild ? this.resizeChild.id : this.el.id);
31054     }
31055
31056     // public events
31057     this.addEvents({
31058         /**
31059          * @event beforeresize
31060          * Fired before resize is allowed. Set enabled to false to cancel resize.
31061          * @param {Roo.Resizable} this
31062          * @param {Roo.EventObject} e The mousedown event
31063          */
31064         "beforeresize" : true,
31065         /**
31066          * @event resizing
31067          * Fired a resizing.
31068          * @param {Roo.Resizable} this
31069          * @param {Number} x The new x position
31070          * @param {Number} y The new y position
31071          * @param {Number} w The new w width
31072          * @param {Number} h The new h hight
31073          * @param {Roo.EventObject} e The mouseup event
31074          */
31075         "resizing" : true,
31076         /**
31077          * @event resize
31078          * Fired after a resize.
31079          * @param {Roo.Resizable} this
31080          * @param {Number} width The new width
31081          * @param {Number} height The new height
31082          * @param {Roo.EventObject} e The mouseup event
31083          */
31084         "resize" : true
31085     });
31086
31087     if(this.width !== null && this.height !== null){
31088         this.resizeTo(this.width, this.height);
31089     }else{
31090         this.updateChildSize();
31091     }
31092     if(Roo.isIE){
31093         this.el.dom.style.zoom = 1;
31094     }
31095     Roo.Resizable.superclass.constructor.call(this);
31096 };
31097
31098 Roo.extend(Roo.Resizable, Roo.util.Observable, {
31099         resizeChild : false,
31100         adjustments : [0, 0],
31101         minWidth : 5,
31102         minHeight : 5,
31103         maxWidth : 10000,
31104         maxHeight : 10000,
31105         enabled : true,
31106         animate : false,
31107         duration : .35,
31108         dynamic : false,
31109         handles : false,
31110         multiDirectional : false,
31111         disableTrackOver : false,
31112         easing : 'easeOutStrong',
31113         widthIncrement : 0,
31114         heightIncrement : 0,
31115         pinned : false,
31116         width : null,
31117         height : null,
31118         preserveRatio : false,
31119         transparent: false,
31120         minX: 0,
31121         minY: 0,
31122         draggable: false,
31123
31124         /**
31125          * @cfg {String/HTMLElement/Element} constrainTo Constrain the resize to a particular element
31126          */
31127         constrainTo: undefined,
31128         /**
31129          * @cfg {Roo.lib.Region} resizeRegion Constrain the resize to a particular region
31130          */
31131         resizeRegion: undefined,
31132
31133
31134     /**
31135      * Perform a manual resize
31136      * @param {Number} width
31137      * @param {Number} height
31138      */
31139     resizeTo : function(width, height){
31140         this.el.setSize(width, height);
31141         this.updateChildSize();
31142         this.fireEvent("resize", this, width, height, null);
31143     },
31144
31145     // private
31146     startSizing : function(e, handle){
31147         this.fireEvent("beforeresize", this, e);
31148         if(this.enabled){ // 2nd enabled check in case disabled before beforeresize handler
31149
31150             if(!this.overlay){
31151                 this.overlay = this.el.createProxy({tag: "div", cls: "x-resizable-overlay", html: "&#160;"});
31152                 this.overlay.unselectable();
31153                 this.overlay.enableDisplayMode("block");
31154                 this.overlay.on("mousemove", this.onMouseMove, this);
31155                 this.overlay.on("mouseup", this.onMouseUp, this);
31156             }
31157             this.overlay.setStyle("cursor", handle.el.getStyle("cursor"));
31158
31159             this.resizing = true;
31160             this.startBox = this.el.getBox();
31161             this.startPoint = e.getXY();
31162             this.offsets = [(this.startBox.x + this.startBox.width) - this.startPoint[0],
31163                             (this.startBox.y + this.startBox.height) - this.startPoint[1]];
31164
31165             this.overlay.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
31166             this.overlay.show();
31167
31168             if(this.constrainTo) {
31169                 var ct = Roo.get(this.constrainTo);
31170                 this.resizeRegion = ct.getRegion().adjust(
31171                     ct.getFrameWidth('t'),
31172                     ct.getFrameWidth('l'),
31173                     -ct.getFrameWidth('b'),
31174                     -ct.getFrameWidth('r')
31175                 );
31176             }
31177
31178             this.proxy.setStyle('visibility', 'hidden'); // workaround display none
31179             this.proxy.show();
31180             this.proxy.setBox(this.startBox);
31181             if(!this.dynamic){
31182                 this.proxy.setStyle('visibility', 'visible');
31183             }
31184         }
31185     },
31186
31187     // private
31188     onMouseDown : function(handle, e){
31189         if(this.enabled){
31190             e.stopEvent();
31191             this.activeHandle = handle;
31192             this.startSizing(e, handle);
31193         }
31194     },
31195
31196     // private
31197     onMouseUp : function(e){
31198         var size = this.resizeElement();
31199         this.resizing = false;
31200         this.handleOut();
31201         this.overlay.hide();
31202         this.proxy.hide();
31203         this.fireEvent("resize", this, size.width, size.height, e);
31204     },
31205
31206     // private
31207     updateChildSize : function(){
31208         
31209         if(this.resizeChild){
31210             var el = this.el;
31211             var child = this.resizeChild;
31212             var adj = this.adjustments;
31213             if(el.dom.offsetWidth){
31214                 var b = el.getSize(true);
31215                 child.setSize(b.width+adj[0], b.height+adj[1]);
31216             }
31217             // Second call here for IE
31218             // The first call enables instant resizing and
31219             // the second call corrects scroll bars if they
31220             // exist
31221             if(Roo.isIE){
31222                 setTimeout(function(){
31223                     if(el.dom.offsetWidth){
31224                         var b = el.getSize(true);
31225                         child.setSize(b.width+adj[0], b.height+adj[1]);
31226                     }
31227                 }, 10);
31228             }
31229         }
31230     },
31231
31232     // private
31233     snap : function(value, inc, min){
31234         if(!inc || !value) {
31235             return value;
31236         }
31237         var newValue = value;
31238         var m = value % inc;
31239         if(m > 0){
31240             if(m > (inc/2)){
31241                 newValue = value + (inc-m);
31242             }else{
31243                 newValue = value - m;
31244             }
31245         }
31246         return Math.max(min, newValue);
31247     },
31248
31249     // private
31250     resizeElement : function(){
31251         var box = this.proxy.getBox();
31252         if(this.updateBox){
31253             this.el.setBox(box, false, this.animate, this.duration, null, this.easing);
31254         }else{
31255             this.el.setSize(box.width, box.height, this.animate, this.duration, null, this.easing);
31256         }
31257         this.updateChildSize();
31258         if(!this.dynamic){
31259             this.proxy.hide();
31260         }
31261         return box;
31262     },
31263
31264     // private
31265     constrain : function(v, diff, m, mx){
31266         if(v - diff < m){
31267             diff = v - m;
31268         }else if(v - diff > mx){
31269             diff = mx - v;
31270         }
31271         return diff;
31272     },
31273
31274     // private
31275     onMouseMove : function(e){
31276         
31277         if(this.enabled){
31278             try{// try catch so if something goes wrong the user doesn't get hung
31279
31280             if(this.resizeRegion && !this.resizeRegion.contains(e.getPoint())) {
31281                 return;
31282             }
31283
31284             //var curXY = this.startPoint;
31285             var curSize = this.curSize || this.startBox;
31286             var x = this.startBox.x, y = this.startBox.y;
31287             var ox = x, oy = y;
31288             var w = curSize.width, h = curSize.height;
31289             var ow = w, oh = h;
31290             var mw = this.minWidth, mh = this.minHeight;
31291             var mxw = this.maxWidth, mxh = this.maxHeight;
31292             var wi = this.widthIncrement;
31293             var hi = this.heightIncrement;
31294
31295             var eventXY = e.getXY();
31296             var diffX = -(this.startPoint[0] - Math.max(this.minX, eventXY[0]));
31297             var diffY = -(this.startPoint[1] - Math.max(this.minY, eventXY[1]));
31298
31299             var pos = this.activeHandle.position;
31300
31301             switch(pos){
31302                 case "east":
31303                     w += diffX;
31304                     w = Math.min(Math.max(mw, w), mxw);
31305                     break;
31306              
31307                 case "south":
31308                     h += diffY;
31309                     h = Math.min(Math.max(mh, h), mxh);
31310                     break;
31311                 case "southeast":
31312                     w += diffX;
31313                     h += diffY;
31314                     w = Math.min(Math.max(mw, w), mxw);
31315                     h = Math.min(Math.max(mh, h), mxh);
31316                     break;
31317                 case "north":
31318                     diffY = this.constrain(h, diffY, mh, mxh);
31319                     y += diffY;
31320                     h -= diffY;
31321                     break;
31322                 case "hdrag":
31323                     
31324                     if (wi) {
31325                         var adiffX = Math.abs(diffX);
31326                         var sub = (adiffX % wi); // how much 
31327                         if (sub > (wi/2)) { // far enough to snap
31328                             diffX = (diffX > 0) ? diffX-sub + wi : diffX+sub - wi;
31329                         } else {
31330                             // remove difference.. 
31331                             diffX = (diffX > 0) ? diffX-sub : diffX+sub;
31332                         }
31333                     }
31334                     x += diffX;
31335                     x = Math.max(this.minX, x);
31336                     break;
31337                 case "west":
31338                     diffX = this.constrain(w, diffX, mw, mxw);
31339                     x += diffX;
31340                     w -= diffX;
31341                     break;
31342                 case "northeast":
31343                     w += diffX;
31344                     w = Math.min(Math.max(mw, w), mxw);
31345                     diffY = this.constrain(h, diffY, mh, mxh);
31346                     y += diffY;
31347                     h -= diffY;
31348                     break;
31349                 case "northwest":
31350                     diffX = this.constrain(w, diffX, mw, mxw);
31351                     diffY = this.constrain(h, diffY, mh, mxh);
31352                     y += diffY;
31353                     h -= diffY;
31354                     x += diffX;
31355                     w -= diffX;
31356                     break;
31357                case "southwest":
31358                     diffX = this.constrain(w, diffX, mw, mxw);
31359                     h += diffY;
31360                     h = Math.min(Math.max(mh, h), mxh);
31361                     x += diffX;
31362                     w -= diffX;
31363                     break;
31364             }
31365
31366             var sw = this.snap(w, wi, mw);
31367             var sh = this.snap(h, hi, mh);
31368             if(sw != w || sh != h){
31369                 switch(pos){
31370                     case "northeast":
31371                         y -= sh - h;
31372                     break;
31373                     case "north":
31374                         y -= sh - h;
31375                         break;
31376                     case "southwest":
31377                         x -= sw - w;
31378                     break;
31379                     case "west":
31380                         x -= sw - w;
31381                         break;
31382                     case "northwest":
31383                         x -= sw - w;
31384                         y -= sh - h;
31385                     break;
31386                 }
31387                 w = sw;
31388                 h = sh;
31389             }
31390
31391             if(this.preserveRatio){
31392                 switch(pos){
31393                     case "southeast":
31394                     case "east":
31395                         h = oh * (w/ow);
31396                         h = Math.min(Math.max(mh, h), mxh);
31397                         w = ow * (h/oh);
31398                        break;
31399                     case "south":
31400                         w = ow * (h/oh);
31401                         w = Math.min(Math.max(mw, w), mxw);
31402                         h = oh * (w/ow);
31403                         break;
31404                     case "northeast":
31405                         w = ow * (h/oh);
31406                         w = Math.min(Math.max(mw, w), mxw);
31407                         h = oh * (w/ow);
31408                     break;
31409                     case "north":
31410                         var tw = w;
31411                         w = ow * (h/oh);
31412                         w = Math.min(Math.max(mw, w), mxw);
31413                         h = oh * (w/ow);
31414                         x += (tw - w) / 2;
31415                         break;
31416                     case "southwest":
31417                         h = oh * (w/ow);
31418                         h = Math.min(Math.max(mh, h), mxh);
31419                         var tw = w;
31420                         w = ow * (h/oh);
31421                         x += tw - w;
31422                         break;
31423                     case "west":
31424                         var th = h;
31425                         h = oh * (w/ow);
31426                         h = Math.min(Math.max(mh, h), mxh);
31427                         y += (th - h) / 2;
31428                         var tw = w;
31429                         w = ow * (h/oh);
31430                         x += tw - w;
31431                        break;
31432                     case "northwest":
31433                         var tw = w;
31434                         var th = h;
31435                         h = oh * (w/ow);
31436                         h = Math.min(Math.max(mh, h), mxh);
31437                         w = ow * (h/oh);
31438                         y += th - h;
31439                         x += tw - w;
31440                        break;
31441
31442                 }
31443             }
31444             if (pos == 'hdrag') {
31445                 w = ow;
31446             }
31447             this.proxy.setBounds(x, y, w, h);
31448             if(this.dynamic){
31449                 this.resizeElement();
31450             }
31451             }catch(e){}
31452         }
31453         this.fireEvent("resizing", this, x, y, w, h, e);
31454     },
31455
31456     // private
31457     handleOver : function(){
31458         if(this.enabled){
31459             this.el.addClass("x-resizable-over");
31460         }
31461     },
31462
31463     // private
31464     handleOut : function(){
31465         if(!this.resizing){
31466             this.el.removeClass("x-resizable-over");
31467         }
31468     },
31469
31470     /**
31471      * Returns the element this component is bound to.
31472      * @return {Roo.Element}
31473      */
31474     getEl : function(){
31475         return this.el;
31476     },
31477
31478     /**
31479      * Returns the resizeChild element (or null).
31480      * @return {Roo.Element}
31481      */
31482     getResizeChild : function(){
31483         return this.resizeChild;
31484     },
31485     groupHandler : function()
31486     {
31487         
31488     },
31489     /**
31490      * Destroys this resizable. If the element was wrapped and
31491      * removeEl is not true then the element remains.
31492      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
31493      */
31494     destroy : function(removeEl){
31495         this.proxy.remove();
31496         if(this.overlay){
31497             this.overlay.removeAllListeners();
31498             this.overlay.remove();
31499         }
31500         var ps = Roo.Resizable.positions;
31501         for(var k in ps){
31502             if(typeof ps[k] != "function" && this[ps[k]]){
31503                 var h = this[ps[k]];
31504                 h.el.removeAllListeners();
31505                 h.el.remove();
31506             }
31507         }
31508         if(removeEl){
31509             this.el.update("");
31510             this.el.remove();
31511         }
31512     }
31513 });
31514
31515 // private
31516 // hash to map config positions to true positions
31517 Roo.Resizable.positions = {
31518     n: "north", s: "south", e: "east", w: "west", se: "southeast", sw: "southwest", nw: "northwest", ne: "northeast", 
31519     hd: "hdrag"
31520 };
31521
31522 // private
31523 Roo.Resizable.Handle = function(rz, pos, disableTrackOver, transparent){
31524     if(!this.tpl){
31525         // only initialize the template if resizable is used
31526         var tpl = Roo.DomHelper.createTemplate(
31527             {tag: "div", cls: "x-resizable-handle x-resizable-handle-{0}"}
31528         );
31529         tpl.compile();
31530         Roo.Resizable.Handle.prototype.tpl = tpl;
31531     }
31532     this.position = pos;
31533     this.rz = rz;
31534     // show north drag fro topdra
31535     var handlepos = pos == 'hdrag' ? 'north' : pos;
31536     
31537     this.el = this.tpl.append(rz.el.dom, [handlepos], true);
31538     if (pos == 'hdrag') {
31539         this.el.setStyle('cursor', 'pointer');
31540     }
31541     this.el.unselectable();
31542     if(transparent){
31543         this.el.setOpacity(0);
31544     }
31545     this.el.on("mousedown", this.onMouseDown, this);
31546     if(!disableTrackOver){
31547         this.el.on("mouseover", this.onMouseOver, this);
31548         this.el.on("mouseout", this.onMouseOut, this);
31549     }
31550 };
31551
31552 // private
31553 Roo.Resizable.Handle.prototype = {
31554     afterResize : function(rz){
31555         Roo.log('after?');
31556         // do nothing
31557     },
31558     // private
31559     onMouseDown : function(e){
31560         this.rz.onMouseDown(this, e);
31561     },
31562     // private
31563     onMouseOver : function(e){
31564         this.rz.handleOver(this, e);
31565     },
31566     // private
31567     onMouseOut : function(e){
31568         this.rz.handleOut(this, e);
31569     }
31570 };/*
31571  * Based on:
31572  * Ext JS Library 1.1.1
31573  * Copyright(c) 2006-2007, Ext JS, LLC.
31574  *
31575  * Originally Released Under LGPL - original licence link has changed is not relivant.
31576  *
31577  * Fork - LGPL
31578  * <script type="text/javascript">
31579  */
31580
31581 /**
31582  * @class Roo.Editor
31583  * @extends Roo.Component
31584  * A base editor field that handles displaying/hiding on demand and has some built-in sizing and event handling logic.
31585  * @constructor
31586  * Create a new Editor
31587  * @param {Roo.form.Field} field The Field object (or descendant)
31588  * @param {Object} config The config object
31589  */
31590 Roo.Editor = function(field, config){
31591     Roo.Editor.superclass.constructor.call(this, config);
31592     this.field = field;
31593     this.addEvents({
31594         /**
31595              * @event beforestartedit
31596              * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
31597              * false from the handler of this event.
31598              * @param {Editor} this
31599              * @param {Roo.Element} boundEl The underlying element bound to this editor
31600              * @param {Mixed} value The field value being set
31601              */
31602         "beforestartedit" : true,
31603         /**
31604              * @event startedit
31605              * Fires when this editor is displayed
31606              * @param {Roo.Element} boundEl The underlying element bound to this editor
31607              * @param {Mixed} value The starting field value
31608              */
31609         "startedit" : true,
31610         /**
31611              * @event beforecomplete
31612              * Fires after a change has been made to the field, but before the change is reflected in the underlying
31613              * field.  Saving the change to the field can be canceled by returning false from the handler of this event.
31614              * Note that if the value has not changed and ignoreNoChange = true, the editing will still end but this
31615              * event will not fire since no edit actually occurred.
31616              * @param {Editor} this
31617              * @param {Mixed} value The current field value
31618              * @param {Mixed} startValue The original field value
31619              */
31620         "beforecomplete" : true,
31621         /**
31622              * @event complete
31623              * Fires after editing is complete and any changed value has been written to the underlying field.
31624              * @param {Editor} this
31625              * @param {Mixed} value The current field value
31626              * @param {Mixed} startValue The original field value
31627              */
31628         "complete" : true,
31629         /**
31630          * @event specialkey
31631          * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
31632          * {@link Roo.EventObject#getKey} to determine which key was pressed.
31633          * @param {Roo.form.Field} this
31634          * @param {Roo.EventObject} e The event object
31635          */
31636         "specialkey" : true
31637     });
31638 };
31639
31640 Roo.extend(Roo.Editor, Roo.Component, {
31641     /**
31642      * @cfg {Boolean/String} autosize
31643      * True for the editor to automatically adopt the size of the underlying field, "width" to adopt the width only,
31644      * or "height" to adopt the height only (defaults to false)
31645      */
31646     /**
31647      * @cfg {Boolean} revertInvalid
31648      * True to automatically revert the field value and cancel the edit when the user completes an edit and the field
31649      * validation fails (defaults to true)
31650      */
31651     /**
31652      * @cfg {Boolean} ignoreNoChange
31653      * True to skip the the edit completion process (no save, no events fired) if the user completes an edit and
31654      * the value has not changed (defaults to false).  Applies only to string values - edits for other data types
31655      * will never be ignored.
31656      */
31657     /**
31658      * @cfg {Boolean} hideEl
31659      * False to keep the bound element visible while the editor is displayed (defaults to true)
31660      */
31661     /**
31662      * @cfg {Mixed} value
31663      * The data value of the underlying field (defaults to "")
31664      */
31665     value : "",
31666     /**
31667      * @cfg {String} alignment
31668      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "c-c?").
31669      */
31670     alignment: "c-c?",
31671     /**
31672      * @cfg {Boolean/String} shadow "sides" for sides/bottom only, "frame" for 4-way shadow, and "drop"
31673      * for bottom-right shadow (defaults to "frame")
31674      */
31675     shadow : "frame",
31676     /**
31677      * @cfg {Boolean} constrain True to constrain the editor to the viewport
31678      */
31679     constrain : false,
31680     /**
31681      * @cfg {Boolean} completeOnEnter True to complete the edit when the enter key is pressed (defaults to false)
31682      */
31683     completeOnEnter : false,
31684     /**
31685      * @cfg {Boolean} cancelOnEsc True to cancel the edit when the escape key is pressed (defaults to false)
31686      */
31687     cancelOnEsc : false,
31688     /**
31689      * @cfg {Boolean} updateEl True to update the innerHTML of the bound element when the update completes (defaults to false)
31690      */
31691     updateEl : false,
31692
31693     // private
31694     onRender : function(ct, position){
31695         this.el = new Roo.Layer({
31696             shadow: this.shadow,
31697             cls: "x-editor",
31698             parentEl : ct,
31699             shim : this.shim,
31700             shadowOffset:4,
31701             id: this.id,
31702             constrain: this.constrain
31703         });
31704         this.el.setStyle("overflow", Roo.isGecko ? "auto" : "hidden");
31705         if(this.field.msgTarget != 'title'){
31706             this.field.msgTarget = 'qtip';
31707         }
31708         this.field.render(this.el);
31709         if(Roo.isGecko){
31710             this.field.el.dom.setAttribute('autocomplete', 'off');
31711         }
31712         this.field.on("specialkey", this.onSpecialKey, this);
31713         if(this.swallowKeys){
31714             this.field.el.swallowEvent(['keydown','keypress']);
31715         }
31716         this.field.show();
31717         this.field.on("blur", this.onBlur, this);
31718         if(this.field.grow){
31719             this.field.on("autosize", this.el.sync,  this.el, {delay:1});
31720         }
31721     },
31722
31723     onSpecialKey : function(field, e)
31724     {
31725         //Roo.log('editor onSpecialKey');
31726         if(this.completeOnEnter && e.getKey() == e.ENTER){
31727             e.stopEvent();
31728             this.completeEdit();
31729             return;
31730         }
31731         // do not fire special key otherwise it might hide close the editor...
31732         if(e.getKey() == e.ENTER){    
31733             return;
31734         }
31735         if(this.cancelOnEsc && e.getKey() == e.ESC){
31736             this.cancelEdit();
31737             return;
31738         } 
31739         this.fireEvent('specialkey', field, e);
31740     
31741     },
31742
31743     /**
31744      * Starts the editing process and shows the editor.
31745      * @param {String/HTMLElement/Element} el The element to edit
31746      * @param {String} value (optional) A value to initialize the editor with. If a value is not provided, it defaults
31747       * to the innerHTML of el.
31748      */
31749     startEdit : function(el, value){
31750         if(this.editing){
31751             this.completeEdit();
31752         }
31753         this.boundEl = Roo.get(el);
31754         var v = value !== undefined ? value : this.boundEl.dom.innerHTML;
31755         if(!this.rendered){
31756             this.render(this.parentEl || document.body);
31757         }
31758         if(this.fireEvent("beforestartedit", this, this.boundEl, v) === false){
31759             return;
31760         }
31761         this.startValue = v;
31762         this.field.setValue(v);
31763         if(this.autoSize){
31764             var sz = this.boundEl.getSize();
31765             switch(this.autoSize){
31766                 case "width":
31767                 this.setSize(sz.width,  "");
31768                 break;
31769                 case "height":
31770                 this.setSize("",  sz.height);
31771                 break;
31772                 default:
31773                 this.setSize(sz.width,  sz.height);
31774             }
31775         }
31776         this.el.alignTo(this.boundEl, this.alignment);
31777         this.editing = true;
31778         if(Roo.QuickTips){
31779             Roo.QuickTips.disable();
31780         }
31781         this.show();
31782     },
31783
31784     /**
31785      * Sets the height and width of this editor.
31786      * @param {Number} width The new width
31787      * @param {Number} height The new height
31788      */
31789     setSize : function(w, h){
31790         this.field.setSize(w, h);
31791         if(this.el){
31792             this.el.sync();
31793         }
31794     },
31795
31796     /**
31797      * Realigns the editor to the bound field based on the current alignment config value.
31798      */
31799     realign : function(){
31800         this.el.alignTo(this.boundEl, this.alignment);
31801     },
31802
31803     /**
31804      * Ends the editing process, persists the changed value to the underlying field, and hides the editor.
31805      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after edit (defaults to false)
31806      */
31807     completeEdit : function(remainVisible){
31808         if(!this.editing){
31809             return;
31810         }
31811         var v = this.getValue();
31812         if(this.revertInvalid !== false && !this.field.isValid()){
31813             v = this.startValue;
31814             this.cancelEdit(true);
31815         }
31816         if(String(v) === String(this.startValue) && this.ignoreNoChange){
31817             this.editing = false;
31818             this.hide();
31819             return;
31820         }
31821         if(this.fireEvent("beforecomplete", this, v, this.startValue) !== false){
31822             this.editing = false;
31823             if(this.updateEl && this.boundEl){
31824                 this.boundEl.update(v);
31825             }
31826             if(remainVisible !== true){
31827                 this.hide();
31828             }
31829             this.fireEvent("complete", this, v, this.startValue);
31830         }
31831     },
31832
31833     // private
31834     onShow : function(){
31835         this.el.show();
31836         if(this.hideEl !== false){
31837             this.boundEl.hide();
31838         }
31839         this.field.show();
31840         if(Roo.isIE && !this.fixIEFocus){ // IE has problems with focusing the first time
31841             this.fixIEFocus = true;
31842             this.deferredFocus.defer(50, this);
31843         }else{
31844             this.field.focus();
31845         }
31846         this.fireEvent("startedit", this.boundEl, this.startValue);
31847     },
31848
31849     deferredFocus : function(){
31850         if(this.editing){
31851             this.field.focus();
31852         }
31853     },
31854
31855     /**
31856      * Cancels the editing process and hides the editor without persisting any changes.  The field value will be
31857      * reverted to the original starting value.
31858      * @param {Boolean} remainVisible Override the default behavior and keep the editor visible after
31859      * cancel (defaults to false)
31860      */
31861     cancelEdit : function(remainVisible){
31862         if(this.editing){
31863             this.setValue(this.startValue);
31864             if(remainVisible !== true){
31865                 this.hide();
31866             }
31867         }
31868     },
31869
31870     // private
31871     onBlur : function(){
31872         if(this.allowBlur !== true && this.editing){
31873             this.completeEdit();
31874         }
31875     },
31876
31877     // private
31878     onHide : function(){
31879         if(this.editing){
31880             this.completeEdit();
31881             return;
31882         }
31883         this.field.blur();
31884         if(this.field.collapse){
31885             this.field.collapse();
31886         }
31887         this.el.hide();
31888         if(this.hideEl !== false){
31889             this.boundEl.show();
31890         }
31891         if(Roo.QuickTips){
31892             Roo.QuickTips.enable();
31893         }
31894     },
31895
31896     /**
31897      * Sets the data value of the editor
31898      * @param {Mixed} value Any valid value supported by the underlying field
31899      */
31900     setValue : function(v){
31901         this.field.setValue(v);
31902     },
31903
31904     /**
31905      * Gets the data value of the editor
31906      * @return {Mixed} The data value
31907      */
31908     getValue : function(){
31909         return this.field.getValue();
31910     }
31911 });/*
31912  * Based on:
31913  * Ext JS Library 1.1.1
31914  * Copyright(c) 2006-2007, Ext JS, LLC.
31915  *
31916  * Originally Released Under LGPL - original licence link has changed is not relivant.
31917  *
31918  * Fork - LGPL
31919  * <script type="text/javascript">
31920  */
31921  
31922 /**
31923  * @class Roo.BasicDialog
31924  * @extends Roo.util.Observable
31925  * Lightweight Dialog Class.  The code below shows the creation of a typical dialog using existing HTML markup:
31926  * <pre><code>
31927 var dlg = new Roo.BasicDialog("my-dlg", {
31928     height: 200,
31929     width: 300,
31930     minHeight: 100,
31931     minWidth: 150,
31932     modal: true,
31933     proxyDrag: true,
31934     shadow: true
31935 });
31936 dlg.addKeyListener(27, dlg.hide, dlg); // ESC can also close the dialog
31937 dlg.addButton('OK', dlg.hide, dlg);    // Could call a save function instead of hiding
31938 dlg.addButton('Cancel', dlg.hide, dlg);
31939 dlg.show();
31940 </code></pre>
31941   <b>A Dialog should always be a direct child of the body element.</b>
31942  * @cfg {Boolean/DomHelper} autoCreate True to auto create from scratch, or using a DomHelper Object (defaults to false)
31943  * @cfg {String} title Default text to display in the title bar (defaults to null)
31944  * @cfg {Number} width Width of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31945  * @cfg {Number} height Height of the dialog in pixels (can also be set via CSS).  Determined by browser if unspecified.
31946  * @cfg {Number} x The default left page coordinate of the dialog (defaults to center screen)
31947  * @cfg {Number} y The default top page coordinate of the dialog (defaults to center screen)
31948  * @cfg {String/Element} animateTarget Id or element from which the dialog should animate while opening
31949  * (defaults to null with no animation)
31950  * @cfg {Boolean} resizable False to disable manual dialog resizing (defaults to true)
31951  * @cfg {String} resizeHandles Which resize handles to display - see the {@link Roo.Resizable} handles config
31952  * property for valid values (defaults to 'all')
31953  * @cfg {Number} minHeight The minimum allowable height for a resizable dialog (defaults to 80)
31954  * @cfg {Number} minWidth The minimum allowable width for a resizable dialog (defaults to 200)
31955  * @cfg {Boolean} modal True to show the dialog modally, preventing user interaction with the rest of the page (defaults to false)
31956  * @cfg {Boolean} autoScroll True to allow the dialog body contents to overflow and display scrollbars (defaults to false)
31957  * @cfg {Boolean} closable False to remove the built-in top-right corner close button (defaults to true)
31958  * @cfg {Boolean} collapsible False to remove the built-in top-right corner collapse button (defaults to true)
31959  * @cfg {Boolean} constraintoviewport True to keep the dialog constrained within the visible viewport boundaries (defaults to true)
31960  * @cfg {Boolean} syncHeightBeforeShow True to cause the dimensions to be recalculated before the dialog is shown (defaults to false)
31961  * @cfg {Boolean} draggable False to disable dragging of the dialog within the viewport (defaults to true)
31962  * @cfg {Boolean} autoTabs If true, all elements with class 'x-dlg-tab' will get automatically converted to tabs (defaults to false)
31963  * @cfg {String} tabTag The tag name of tab elements, used when autoTabs = true (defaults to 'div')
31964  * @cfg {Boolean} proxyDrag True to drag a lightweight proxy element rather than the dialog itself, used when
31965  * draggable = true (defaults to false)
31966  * @cfg {Boolean} fixedcenter True to ensure that anytime the dialog is shown or resized it gets centered (defaults to false)
31967  * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
31968  * shadow (defaults to false)
31969  * @cfg {Number} shadowOffset The number of pixels to offset the shadow if displayed (defaults to 5)
31970  * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "right")
31971  * @cfg {Number} minButtonWidth Minimum width of all dialog buttons (defaults to 75)
31972  * @cfg {Array} buttons Array of buttons
31973  * @cfg {Boolean} shim True to create an iframe shim that prevents selects from showing through (defaults to false)
31974  * @constructor
31975  * Create a new BasicDialog.
31976  * @param {String/HTMLElement/Roo.Element} el The container element or DOM node, or its id
31977  * @param {Object} config Configuration options
31978  */
31979 Roo.BasicDialog = function(el, config){
31980     this.el = Roo.get(el);
31981     var dh = Roo.DomHelper;
31982     if(!this.el && config && config.autoCreate){
31983         if(typeof config.autoCreate == "object"){
31984             if(!config.autoCreate.id){
31985                 config.autoCreate.id = el;
31986             }
31987             this.el = dh.append(document.body,
31988                         config.autoCreate, true);
31989         }else{
31990             this.el = dh.append(document.body,
31991                         {tag: "div", id: el, style:'visibility:hidden;'}, true);
31992         }
31993     }
31994     el = this.el;
31995     el.setDisplayed(true);
31996     el.hide = this.hideAction;
31997     this.id = el.id;
31998     el.addClass("x-dlg");
31999
32000     Roo.apply(this, config);
32001
32002     this.proxy = el.createProxy("x-dlg-proxy");
32003     this.proxy.hide = this.hideAction;
32004     this.proxy.setOpacity(.5);
32005     this.proxy.hide();
32006
32007     if(config.width){
32008         el.setWidth(config.width);
32009     }
32010     if(config.height){
32011         el.setHeight(config.height);
32012     }
32013     this.size = el.getSize();
32014     if(typeof config.x != "undefined" && typeof config.y != "undefined"){
32015         this.xy = [config.x,config.y];
32016     }else{
32017         this.xy = el.getCenterXY(true);
32018     }
32019     /** The header element @type Roo.Element */
32020     this.header = el.child("> .x-dlg-hd");
32021     /** The body element @type Roo.Element */
32022     this.body = el.child("> .x-dlg-bd");
32023     /** The footer element @type Roo.Element */
32024     this.footer = el.child("> .x-dlg-ft");
32025
32026     if(!this.header){
32027         this.header = el.createChild({tag: "div", cls:"x-dlg-hd", html: "&#160;"}, this.body ? this.body.dom : null);
32028     }
32029     if(!this.body){
32030         this.body = el.createChild({tag: "div", cls:"x-dlg-bd"});
32031     }
32032
32033     this.header.unselectable();
32034     if(this.title){
32035         this.header.update(this.title);
32036     }
32037     // this element allows the dialog to be focused for keyboard event
32038     this.focusEl = el.createChild({tag: "a", href:"#", cls:"x-dlg-focus", tabIndex:"-1"});
32039     this.focusEl.swallowEvent("click", true);
32040
32041     this.header.wrap({cls:"x-dlg-hd-right"}).wrap({cls:"x-dlg-hd-left"}, true);
32042
32043     // wrap the body and footer for special rendering
32044     this.bwrap = this.body.wrap({tag: "div", cls:"x-dlg-dlg-body"});
32045     if(this.footer){
32046         this.bwrap.dom.appendChild(this.footer.dom);
32047     }
32048
32049     this.bg = this.el.createChild({
32050         tag: "div", cls:"x-dlg-bg",
32051         html: '<div class="x-dlg-bg-left"><div class="x-dlg-bg-right"><div class="x-dlg-bg-center">&#160;</div></div></div>'
32052     });
32053     this.centerBg = this.bg.child("div.x-dlg-bg-center");
32054
32055
32056     if(this.autoScroll !== false && !this.autoTabs){
32057         this.body.setStyle("overflow", "auto");
32058     }
32059
32060     this.toolbox = this.el.createChild({cls: "x-dlg-toolbox"});
32061
32062     if(this.closable !== false){
32063         this.el.addClass("x-dlg-closable");
32064         this.close = this.toolbox.createChild({cls:"x-dlg-close"});
32065         this.close.on("click", this.closeClick, this);
32066         this.close.addClassOnOver("x-dlg-close-over");
32067     }
32068     if(this.collapsible !== false){
32069         this.collapseBtn = this.toolbox.createChild({cls:"x-dlg-collapse"});
32070         this.collapseBtn.on("click", this.collapseClick, this);
32071         this.collapseBtn.addClassOnOver("x-dlg-collapse-over");
32072         this.header.on("dblclick", this.collapseClick, this);
32073     }
32074     if(this.resizable !== false){
32075         this.el.addClass("x-dlg-resizable");
32076         this.resizer = new Roo.Resizable(el, {
32077             minWidth: this.minWidth || 80,
32078             minHeight:this.minHeight || 80,
32079             handles: this.resizeHandles || "all",
32080             pinned: true
32081         });
32082         this.resizer.on("beforeresize", this.beforeResize, this);
32083         this.resizer.on("resize", this.onResize, this);
32084     }
32085     if(this.draggable !== false){
32086         el.addClass("x-dlg-draggable");
32087         if (!this.proxyDrag) {
32088             var dd = new Roo.dd.DD(el.dom.id, "WindowDrag");
32089         }
32090         else {
32091             var dd = new Roo.dd.DDProxy(el.dom.id, "WindowDrag", {dragElId: this.proxy.id});
32092         }
32093         dd.setHandleElId(this.header.id);
32094         dd.endDrag = this.endMove.createDelegate(this);
32095         dd.startDrag = this.startMove.createDelegate(this);
32096         dd.onDrag = this.onDrag.createDelegate(this);
32097         dd.scroll = false;
32098         this.dd = dd;
32099     }
32100     if(this.modal){
32101         this.mask = dh.append(document.body, {tag: "div", cls:"x-dlg-mask"}, true);
32102         this.mask.enableDisplayMode("block");
32103         this.mask.hide();
32104         this.el.addClass("x-dlg-modal");
32105     }
32106     if(this.shadow){
32107         this.shadow = new Roo.Shadow({
32108             mode : typeof this.shadow == "string" ? this.shadow : "sides",
32109             offset : this.shadowOffset
32110         });
32111     }else{
32112         this.shadowOffset = 0;
32113     }
32114     if(Roo.useShims && this.shim !== false){
32115         this.shim = this.el.createShim();
32116         this.shim.hide = this.hideAction;
32117         this.shim.hide();
32118     }else{
32119         this.shim = false;
32120     }
32121     if(this.autoTabs){
32122         this.initTabs();
32123     }
32124     if (this.buttons) { 
32125         var bts= this.buttons;
32126         this.buttons = [];
32127         Roo.each(bts, function(b) {
32128             this.addButton(b);
32129         }, this);
32130     }
32131     
32132     
32133     this.addEvents({
32134         /**
32135          * @event keydown
32136          * Fires when a key is pressed
32137          * @param {Roo.BasicDialog} this
32138          * @param {Roo.EventObject} e
32139          */
32140         "keydown" : true,
32141         /**
32142          * @event move
32143          * Fires when this dialog is moved by the user.
32144          * @param {Roo.BasicDialog} this
32145          * @param {Number} x The new page X
32146          * @param {Number} y The new page Y
32147          */
32148         "move" : true,
32149         /**
32150          * @event resize
32151          * Fires when this dialog is resized by the user.
32152          * @param {Roo.BasicDialog} this
32153          * @param {Number} width The new width
32154          * @param {Number} height The new height
32155          */
32156         "resize" : true,
32157         /**
32158          * @event beforehide
32159          * Fires before this dialog is hidden.
32160          * @param {Roo.BasicDialog} this
32161          */
32162         "beforehide" : true,
32163         /**
32164          * @event hide
32165          * Fires when this dialog is hidden.
32166          * @param {Roo.BasicDialog} this
32167          */
32168         "hide" : true,
32169         /**
32170          * @event beforeshow
32171          * Fires before this dialog is shown.
32172          * @param {Roo.BasicDialog} this
32173          */
32174         "beforeshow" : true,
32175         /**
32176          * @event show
32177          * Fires when this dialog is shown.
32178          * @param {Roo.BasicDialog} this
32179          */
32180         "show" : true
32181     });
32182     el.on("keydown", this.onKeyDown, this);
32183     el.on("mousedown", this.toFront, this);
32184     Roo.EventManager.onWindowResize(this.adjustViewport, this, true);
32185     this.el.hide();
32186     Roo.DialogManager.register(this);
32187     Roo.BasicDialog.superclass.constructor.call(this);
32188 };
32189
32190 Roo.extend(Roo.BasicDialog, Roo.util.Observable, {
32191     shadowOffset: Roo.isIE ? 6 : 5,
32192     minHeight: 80,
32193     minWidth: 200,
32194     minButtonWidth: 75,
32195     defaultButton: null,
32196     buttonAlign: "right",
32197     tabTag: 'div',
32198     firstShow: true,
32199
32200     /**
32201      * Sets the dialog title text
32202      * @param {String} text The title text to display
32203      * @return {Roo.BasicDialog} this
32204      */
32205     setTitle : function(text){
32206         this.header.update(text);
32207         return this;
32208     },
32209
32210     // private
32211     closeClick : function(){
32212         this.hide();
32213     },
32214
32215     // private
32216     collapseClick : function(){
32217         this[this.collapsed ? "expand" : "collapse"]();
32218     },
32219
32220     /**
32221      * Collapses the dialog to its minimized state (only the title bar is visible).
32222      * Equivalent to the user clicking the collapse dialog button.
32223      */
32224     collapse : function(){
32225         if(!this.collapsed){
32226             this.collapsed = true;
32227             this.el.addClass("x-dlg-collapsed");
32228             this.restoreHeight = this.el.getHeight();
32229             this.resizeTo(this.el.getWidth(), this.header.getHeight());
32230         }
32231     },
32232
32233     /**
32234      * Expands a collapsed dialog back to its normal state.  Equivalent to the user
32235      * clicking the expand dialog button.
32236      */
32237     expand : function(){
32238         if(this.collapsed){
32239             this.collapsed = false;
32240             this.el.removeClass("x-dlg-collapsed");
32241             this.resizeTo(this.el.getWidth(), this.restoreHeight);
32242         }
32243     },
32244
32245     /**
32246      * Reinitializes the tabs component, clearing out old tabs and finding new ones.
32247      * @return {Roo.TabPanel} The tabs component
32248      */
32249     initTabs : function(){
32250         var tabs = this.getTabs();
32251         while(tabs.getTab(0)){
32252             tabs.removeTab(0);
32253         }
32254         this.el.select(this.tabTag+'.x-dlg-tab').each(function(el){
32255             var dom = el.dom;
32256             tabs.addTab(Roo.id(dom), dom.title);
32257             dom.title = "";
32258         });
32259         tabs.activate(0);
32260         return tabs;
32261     },
32262
32263     // private
32264     beforeResize : function(){
32265         this.resizer.minHeight = Math.max(this.minHeight, this.getHeaderFooterHeight(true)+40);
32266     },
32267
32268     // private
32269     onResize : function(){
32270         this.refreshSize();
32271         this.syncBodyHeight();
32272         this.adjustAssets();
32273         this.focus();
32274         this.fireEvent("resize", this, this.size.width, this.size.height);
32275     },
32276
32277     // private
32278     onKeyDown : function(e){
32279         if(this.isVisible()){
32280             this.fireEvent("keydown", this, e);
32281         }
32282     },
32283
32284     /**
32285      * Resizes the dialog.
32286      * @param {Number} width
32287      * @param {Number} height
32288      * @return {Roo.BasicDialog} this
32289      */
32290     resizeTo : function(width, height){
32291         this.el.setSize(width, height);
32292         this.size = {width: width, height: height};
32293         this.syncBodyHeight();
32294         if(this.fixedcenter){
32295             this.center();
32296         }
32297         if(this.isVisible()){
32298             this.constrainXY();
32299             this.adjustAssets();
32300         }
32301         this.fireEvent("resize", this, width, height);
32302         return this;
32303     },
32304
32305
32306     /**
32307      * Resizes the dialog to fit the specified content size.
32308      * @param {Number} width
32309      * @param {Number} height
32310      * @return {Roo.BasicDialog} this
32311      */
32312     setContentSize : function(w, h){
32313         h += this.getHeaderFooterHeight() + this.body.getMargins("tb");
32314         w += this.body.getMargins("lr") + this.bwrap.getMargins("lr") + this.centerBg.getPadding("lr");
32315         //if(!this.el.isBorderBox()){
32316             h +=  this.body.getPadding("tb") + this.bwrap.getBorderWidth("tb") + this.body.getBorderWidth("tb") + this.el.getBorderWidth("tb");
32317             w += this.body.getPadding("lr") + this.bwrap.getBorderWidth("lr") + this.body.getBorderWidth("lr") + this.bwrap.getPadding("lr") + this.el.getBorderWidth("lr");
32318         //}
32319         if(this.tabs){
32320             h += this.tabs.stripWrap.getHeight() + this.tabs.bodyEl.getMargins("tb") + this.tabs.bodyEl.getPadding("tb");
32321             w += this.tabs.bodyEl.getMargins("lr") + this.tabs.bodyEl.getPadding("lr");
32322         }
32323         this.resizeTo(w, h);
32324         return this;
32325     },
32326
32327     /**
32328      * Adds a key listener for when this dialog is displayed.  This allows you to hook in a function that will be
32329      * executed in response to a particular key being pressed while the dialog is active.
32330      * @param {Number/Array/Object} key Either the numeric key code, array of key codes or an object with the following options:
32331      *                                  {key: (number or array), shift: (true/false), ctrl: (true/false), alt: (true/false)}
32332      * @param {Function} fn The function to call
32333      * @param {Object} scope (optional) The scope of the function
32334      * @return {Roo.BasicDialog} this
32335      */
32336     addKeyListener : function(key, fn, scope){
32337         var keyCode, shift, ctrl, alt;
32338         if(typeof key == "object" && !(key instanceof Array)){
32339             keyCode = key["key"];
32340             shift = key["shift"];
32341             ctrl = key["ctrl"];
32342             alt = key["alt"];
32343         }else{
32344             keyCode = key;
32345         }
32346         var handler = function(dlg, e){
32347             if((!shift || e.shiftKey) && (!ctrl || e.ctrlKey) &&  (!alt || e.altKey)){
32348                 var k = e.getKey();
32349                 if(keyCode instanceof Array){
32350                     for(var i = 0, len = keyCode.length; i < len; i++){
32351                         if(keyCode[i] == k){
32352                           fn.call(scope || window, dlg, k, e);
32353                           return;
32354                         }
32355                     }
32356                 }else{
32357                     if(k == keyCode){
32358                         fn.call(scope || window, dlg, k, e);
32359                     }
32360                 }
32361             }
32362         };
32363         this.on("keydown", handler);
32364         return this;
32365     },
32366
32367     /**
32368      * Returns the TabPanel component (creates it if it doesn't exist).
32369      * Note: If you wish to simply check for the existence of tabs without creating them,
32370      * check for a null 'tabs' property.
32371      * @return {Roo.TabPanel} The tabs component
32372      */
32373     getTabs : function(){
32374         if(!this.tabs){
32375             this.el.addClass("x-dlg-auto-tabs");
32376             this.body.addClass(this.tabPosition == "bottom" ? "x-tabs-bottom" : "x-tabs-top");
32377             this.tabs = new Roo.TabPanel(this.body.dom, this.tabPosition == "bottom");
32378         }
32379         return this.tabs;
32380     },
32381
32382     /**
32383      * Adds a button to the footer section of the dialog.
32384      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
32385      * object or a valid Roo.DomHelper element config
32386      * @param {Function} handler The function called when the button is clicked
32387      * @param {Object} scope (optional) The scope of the handler function (accepts position as a property)
32388      * @return {Roo.Button} The new button
32389      */
32390     addButton : function(config, handler, scope){
32391         var dh = Roo.DomHelper;
32392         if(!this.footer){
32393             this.footer = dh.append(this.bwrap, {tag: "div", cls:"x-dlg-ft"}, true);
32394         }
32395         if(!this.btnContainer){
32396             var tb = this.footer.createChild({
32397
32398                 cls:"x-dlg-btns x-dlg-btns-"+this.buttonAlign,
32399                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
32400             }, null, true);
32401             this.btnContainer = tb.firstChild.firstChild.firstChild;
32402         }
32403         var bconfig = {
32404             handler: handler,
32405             scope: scope,
32406             minWidth: this.minButtonWidth,
32407             hideParent:true
32408         };
32409         if(typeof config == "string"){
32410             bconfig.text = config;
32411         }else{
32412             if(config.tag){
32413                 bconfig.dhconfig = config;
32414             }else{
32415                 Roo.apply(bconfig, config);
32416             }
32417         }
32418         var fc = false;
32419         if ((typeof(bconfig.position) != 'undefined') && bconfig.position < this.btnContainer.childNodes.length-1) {
32420             bconfig.position = Math.max(0, bconfig.position);
32421             fc = this.btnContainer.childNodes[bconfig.position];
32422         }
32423          
32424         var btn = new Roo.Button(
32425             fc ? 
32426                 this.btnContainer.insertBefore(document.createElement("td"),fc)
32427                 : this.btnContainer.appendChild(document.createElement("td")),
32428             //Roo.get(this.btnContainer).createChild( { tag: 'td'},  fc ),
32429             bconfig
32430         );
32431         this.syncBodyHeight();
32432         if(!this.buttons){
32433             /**
32434              * Array of all the buttons that have been added to this dialog via addButton
32435              * @type Array
32436              */
32437             this.buttons = [];
32438         }
32439         this.buttons.push(btn);
32440         return btn;
32441     },
32442
32443     /**
32444      * Sets the default button to be focused when the dialog is displayed.
32445      * @param {Roo.BasicDialog.Button} btn The button object returned by {@link #addButton}
32446      * @return {Roo.BasicDialog} this
32447      */
32448     setDefaultButton : function(btn){
32449         this.defaultButton = btn;
32450         return this;
32451     },
32452
32453     // private
32454     getHeaderFooterHeight : function(safe){
32455         var height = 0;
32456         if(this.header){
32457            height += this.header.getHeight();
32458         }
32459         if(this.footer){
32460            var fm = this.footer.getMargins();
32461             height += (this.footer.getHeight()+fm.top+fm.bottom);
32462         }
32463         height += this.bwrap.getPadding("tb")+this.bwrap.getBorderWidth("tb");
32464         height += this.centerBg.getPadding("tb");
32465         return height;
32466     },
32467
32468     // private
32469     syncBodyHeight : function()
32470     {
32471         var bd = this.body, // the text
32472             cb = this.centerBg, // wrapper around bottom.. but does not seem to be used..
32473             bw = this.bwrap;
32474         var height = this.size.height - this.getHeaderFooterHeight(false);
32475         bd.setHeight(height-bd.getMargins("tb"));
32476         var hh = this.header.getHeight();
32477         var h = this.size.height-hh;
32478         cb.setHeight(h);
32479         
32480         bw.setLeftTop(cb.getPadding("l"), hh+cb.getPadding("t"));
32481         bw.setHeight(h-cb.getPadding("tb"));
32482         
32483         bw.setWidth(this.el.getWidth(true)-cb.getPadding("lr"));
32484         bd.setWidth(bw.getWidth(true));
32485         if(this.tabs){
32486             this.tabs.syncHeight();
32487             if(Roo.isIE){
32488                 this.tabs.el.repaint();
32489             }
32490         }
32491     },
32492
32493     /**
32494      * Restores the previous state of the dialog if Roo.state is configured.
32495      * @return {Roo.BasicDialog} this
32496      */
32497     restoreState : function(){
32498         var box = Roo.state.Manager.get(this.stateId || (this.el.id + "-state"));
32499         if(box && box.width){
32500             this.xy = [box.x, box.y];
32501             this.resizeTo(box.width, box.height);
32502         }
32503         return this;
32504     },
32505
32506     // private
32507     beforeShow : function(){
32508         this.expand();
32509         if(this.fixedcenter){
32510             this.xy = this.el.getCenterXY(true);
32511         }
32512         if(this.modal){
32513             Roo.get(document.body).addClass("x-body-masked");
32514             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32515             this.mask.show();
32516         }
32517         this.constrainXY();
32518     },
32519
32520     // private
32521     animShow : function(){
32522         var b = Roo.get(this.animateTarget).getBox();
32523         this.proxy.setSize(b.width, b.height);
32524         this.proxy.setLocation(b.x, b.y);
32525         this.proxy.show();
32526         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height,
32527                     true, .35, this.showEl.createDelegate(this));
32528     },
32529
32530     /**
32531      * Shows the dialog.
32532      * @param {String/HTMLElement/Roo.Element} animateTarget (optional) Reset the animation target
32533      * @return {Roo.BasicDialog} this
32534      */
32535     show : function(animateTarget){
32536         if (this.fireEvent("beforeshow", this) === false){
32537             return;
32538         }
32539         if(this.syncHeightBeforeShow){
32540             this.syncBodyHeight();
32541         }else if(this.firstShow){
32542             this.firstShow = false;
32543             this.syncBodyHeight(); // sync the height on the first show instead of in the constructor
32544         }
32545         this.animateTarget = animateTarget || this.animateTarget;
32546         if(!this.el.isVisible()){
32547             this.beforeShow();
32548             if(this.animateTarget && Roo.get(this.animateTarget)){
32549                 this.animShow();
32550             }else{
32551                 this.showEl();
32552             }
32553         }
32554         return this;
32555     },
32556
32557     // private
32558     showEl : function(){
32559         this.proxy.hide();
32560         this.el.setXY(this.xy);
32561         this.el.show();
32562         this.adjustAssets(true);
32563         this.toFront();
32564         this.focus();
32565         // IE peekaboo bug - fix found by Dave Fenwick
32566         if(Roo.isIE){
32567             this.el.repaint();
32568         }
32569         this.fireEvent("show", this);
32570     },
32571
32572     /**
32573      * Focuses the dialog.  If a defaultButton is set, it will receive focus, otherwise the
32574      * dialog itself will receive focus.
32575      */
32576     focus : function(){
32577         if(this.defaultButton){
32578             this.defaultButton.focus();
32579         }else{
32580             this.focusEl.focus();
32581         }
32582     },
32583
32584     // private
32585     constrainXY : function(){
32586         if(this.constraintoviewport !== false){
32587             if(!this.viewSize){
32588                 if(this.container){
32589                     var s = this.container.getSize();
32590                     this.viewSize = [s.width, s.height];
32591                 }else{
32592                     this.viewSize = [Roo.lib.Dom.getViewWidth(),Roo.lib.Dom.getViewHeight()];
32593                 }
32594             }
32595             var s = Roo.get(this.container||document).getScroll();
32596
32597             var x = this.xy[0], y = this.xy[1];
32598             var w = this.size.width, h = this.size.height;
32599             var vw = this.viewSize[0], vh = this.viewSize[1];
32600             // only move it if it needs it
32601             var moved = false;
32602             // first validate right/bottom
32603             if(x + w > vw+s.left){
32604                 x = vw - w;
32605                 moved = true;
32606             }
32607             if(y + h > vh+s.top){
32608                 y = vh - h;
32609                 moved = true;
32610             }
32611             // then make sure top/left isn't negative
32612             if(x < s.left){
32613                 x = s.left;
32614                 moved = true;
32615             }
32616             if(y < s.top){
32617                 y = s.top;
32618                 moved = true;
32619             }
32620             if(moved){
32621                 // cache xy
32622                 this.xy = [x, y];
32623                 if(this.isVisible()){
32624                     this.el.setLocation(x, y);
32625                     this.adjustAssets();
32626                 }
32627             }
32628         }
32629     },
32630
32631     // private
32632     onDrag : function(){
32633         if(!this.proxyDrag){
32634             this.xy = this.el.getXY();
32635             this.adjustAssets();
32636         }
32637     },
32638
32639     // private
32640     adjustAssets : function(doShow){
32641         var x = this.xy[0], y = this.xy[1];
32642         var w = this.size.width, h = this.size.height;
32643         if(doShow === true){
32644             if(this.shadow){
32645                 this.shadow.show(this.el);
32646             }
32647             if(this.shim){
32648                 this.shim.show();
32649             }
32650         }
32651         if(this.shadow && this.shadow.isVisible()){
32652             this.shadow.show(this.el);
32653         }
32654         if(this.shim && this.shim.isVisible()){
32655             this.shim.setBounds(x, y, w, h);
32656         }
32657     },
32658
32659     // private
32660     adjustViewport : function(w, h){
32661         if(!w || !h){
32662             w = Roo.lib.Dom.getViewWidth();
32663             h = Roo.lib.Dom.getViewHeight();
32664         }
32665         // cache the size
32666         this.viewSize = [w, h];
32667         if(this.modal && this.mask.isVisible()){
32668             this.mask.setSize(w, h); // first make sure the mask isn't causing overflow
32669             this.mask.setSize(Roo.lib.Dom.getViewWidth(true), Roo.lib.Dom.getViewHeight(true));
32670         }
32671         if(this.isVisible()){
32672             this.constrainXY();
32673         }
32674     },
32675
32676     /**
32677      * Destroys this dialog and all its supporting elements (including any tabs, shim,
32678      * shadow, proxy, mask, etc.)  Also removes all event listeners.
32679      * @param {Boolean} removeEl (optional) true to remove the element from the DOM
32680      */
32681     destroy : function(removeEl){
32682         if(this.isVisible()){
32683             this.animateTarget = null;
32684             this.hide();
32685         }
32686         Roo.EventManager.removeResizeListener(this.adjustViewport, this);
32687         if(this.tabs){
32688             this.tabs.destroy(removeEl);
32689         }
32690         Roo.destroy(
32691              this.shim,
32692              this.proxy,
32693              this.resizer,
32694              this.close,
32695              this.mask
32696         );
32697         if(this.dd){
32698             this.dd.unreg();
32699         }
32700         if(this.buttons){
32701            for(var i = 0, len = this.buttons.length; i < len; i++){
32702                this.buttons[i].destroy();
32703            }
32704         }
32705         this.el.removeAllListeners();
32706         if(removeEl === true){
32707             this.el.update("");
32708             this.el.remove();
32709         }
32710         Roo.DialogManager.unregister(this);
32711     },
32712
32713     // private
32714     startMove : function(){
32715         if(this.proxyDrag){
32716             this.proxy.show();
32717         }
32718         if(this.constraintoviewport !== false){
32719             this.dd.constrainTo(document.body, {right: this.shadowOffset, bottom: this.shadowOffset});
32720         }
32721     },
32722
32723     // private
32724     endMove : function(){
32725         if(!this.proxyDrag){
32726             Roo.dd.DD.prototype.endDrag.apply(this.dd, arguments);
32727         }else{
32728             Roo.dd.DDProxy.prototype.endDrag.apply(this.dd, arguments);
32729             this.proxy.hide();
32730         }
32731         this.refreshSize();
32732         this.adjustAssets();
32733         this.focus();
32734         this.fireEvent("move", this, this.xy[0], this.xy[1]);
32735     },
32736
32737     /**
32738      * Brings this dialog to the front of any other visible dialogs
32739      * @return {Roo.BasicDialog} this
32740      */
32741     toFront : function(){
32742         Roo.DialogManager.bringToFront(this);
32743         return this;
32744     },
32745
32746     /**
32747      * Sends this dialog to the back (under) of any other visible dialogs
32748      * @return {Roo.BasicDialog} this
32749      */
32750     toBack : function(){
32751         Roo.DialogManager.sendToBack(this);
32752         return this;
32753     },
32754
32755     /**
32756      * Centers this dialog in the viewport
32757      * @return {Roo.BasicDialog} this
32758      */
32759     center : function(){
32760         var xy = this.el.getCenterXY(true);
32761         this.moveTo(xy[0], xy[1]);
32762         return this;
32763     },
32764
32765     /**
32766      * Moves the dialog's top-left corner to the specified point
32767      * @param {Number} x
32768      * @param {Number} y
32769      * @return {Roo.BasicDialog} this
32770      */
32771     moveTo : function(x, y){
32772         this.xy = [x,y];
32773         if(this.isVisible()){
32774             this.el.setXY(this.xy);
32775             this.adjustAssets();
32776         }
32777         return this;
32778     },
32779
32780     /**
32781      * Aligns the dialog to the specified element
32782      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32783      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details).
32784      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32785      * @return {Roo.BasicDialog} this
32786      */
32787     alignTo : function(element, position, offsets){
32788         this.xy = this.el.getAlignToXY(element, position, offsets);
32789         if(this.isVisible()){
32790             this.el.setXY(this.xy);
32791             this.adjustAssets();
32792         }
32793         return this;
32794     },
32795
32796     /**
32797      * Anchors an element to another element and realigns it when the window is resized.
32798      * @param {String/HTMLElement/Roo.Element} element The element to align to.
32799      * @param {String} position The position to align to (see {@link Roo.Element#alignTo} for more details)
32800      * @param {Array} offsets (optional) Offset the positioning by [x, y]
32801      * @param {Boolean/Number} monitorScroll (optional) true to monitor body scroll and reposition. If this parameter
32802      * is a number, it is used as the buffer delay (defaults to 50ms).
32803      * @return {Roo.BasicDialog} this
32804      */
32805     anchorTo : function(el, alignment, offsets, monitorScroll){
32806         var action = function(){
32807             this.alignTo(el, alignment, offsets);
32808         };
32809         Roo.EventManager.onWindowResize(action, this);
32810         var tm = typeof monitorScroll;
32811         if(tm != 'undefined'){
32812             Roo.EventManager.on(window, 'scroll', action, this,
32813                 {buffer: tm == 'number' ? monitorScroll : 50});
32814         }
32815         action.call(this);
32816         return this;
32817     },
32818
32819     /**
32820      * Returns true if the dialog is visible
32821      * @return {Boolean}
32822      */
32823     isVisible : function(){
32824         return this.el.isVisible();
32825     },
32826
32827     // private
32828     animHide : function(callback){
32829         var b = Roo.get(this.animateTarget).getBox();
32830         this.proxy.show();
32831         this.proxy.setBounds(this.xy[0], this.xy[1], this.size.width, this.size.height);
32832         this.el.hide();
32833         this.proxy.setBounds(b.x, b.y, b.width, b.height, true, .35,
32834                     this.hideEl.createDelegate(this, [callback]));
32835     },
32836
32837     /**
32838      * Hides the dialog.
32839      * @param {Function} callback (optional) Function to call when the dialog is hidden
32840      * @return {Roo.BasicDialog} this
32841      */
32842     hide : function(callback){
32843         if (this.fireEvent("beforehide", this) === false){
32844             return;
32845         }
32846         if(this.shadow){
32847             this.shadow.hide();
32848         }
32849         if(this.shim) {
32850           this.shim.hide();
32851         }
32852         // sometimes animateTarget seems to get set.. causing problems...
32853         // this just double checks..
32854         if(this.animateTarget && Roo.get(this.animateTarget)) {
32855            this.animHide(callback);
32856         }else{
32857             this.el.hide();
32858             this.hideEl(callback);
32859         }
32860         return this;
32861     },
32862
32863     // private
32864     hideEl : function(callback){
32865         this.proxy.hide();
32866         if(this.modal){
32867             this.mask.hide();
32868             Roo.get(document.body).removeClass("x-body-masked");
32869         }
32870         this.fireEvent("hide", this);
32871         if(typeof callback == "function"){
32872             callback();
32873         }
32874     },
32875
32876     // private
32877     hideAction : function(){
32878         this.setLeft("-10000px");
32879         this.setTop("-10000px");
32880         this.setStyle("visibility", "hidden");
32881     },
32882
32883     // private
32884     refreshSize : function(){
32885         this.size = this.el.getSize();
32886         this.xy = this.el.getXY();
32887         Roo.state.Manager.set(this.stateId || this.el.id + "-state", this.el.getBox());
32888     },
32889
32890     // private
32891     // z-index is managed by the DialogManager and may be overwritten at any time
32892     setZIndex : function(index){
32893         if(this.modal){
32894             this.mask.setStyle("z-index", index);
32895         }
32896         if(this.shim){
32897             this.shim.setStyle("z-index", ++index);
32898         }
32899         if(this.shadow){
32900             this.shadow.setZIndex(++index);
32901         }
32902         this.el.setStyle("z-index", ++index);
32903         if(this.proxy){
32904             this.proxy.setStyle("z-index", ++index);
32905         }
32906         if(this.resizer){
32907             this.resizer.proxy.setStyle("z-index", ++index);
32908         }
32909
32910         this.lastZIndex = index;
32911     },
32912
32913     /**
32914      * Returns the element for this dialog
32915      * @return {Roo.Element} The underlying dialog Element
32916      */
32917     getEl : function(){
32918         return this.el;
32919     }
32920 });
32921
32922 /**
32923  * @class Roo.DialogManager
32924  * Provides global access to BasicDialogs that have been created and
32925  * support for z-indexing (layering) multiple open dialogs.
32926  */
32927 Roo.DialogManager = function(){
32928     var list = {};
32929     var accessList = [];
32930     var front = null;
32931
32932     // private
32933     var sortDialogs = function(d1, d2){
32934         return (!d1._lastAccess || d1._lastAccess < d2._lastAccess) ? -1 : 1;
32935     };
32936
32937     // private
32938     var orderDialogs = function(){
32939         accessList.sort(sortDialogs);
32940         var seed = Roo.DialogManager.zseed;
32941         for(var i = 0, len = accessList.length; i < len; i++){
32942             var dlg = accessList[i];
32943             if(dlg){
32944                 dlg.setZIndex(seed + (i*10));
32945             }
32946         }
32947     };
32948
32949     return {
32950         /**
32951          * The starting z-index for BasicDialogs (defaults to 9000)
32952          * @type Number The z-index value
32953          */
32954         zseed : 9000,
32955
32956         // private
32957         register : function(dlg){
32958             list[dlg.id] = dlg;
32959             accessList.push(dlg);
32960         },
32961
32962         // private
32963         unregister : function(dlg){
32964             delete list[dlg.id];
32965             var i=0;
32966             var len=0;
32967             if(!accessList.indexOf){
32968                 for(  i = 0, len = accessList.length; i < len; i++){
32969                     if(accessList[i] == dlg){
32970                         accessList.splice(i, 1);
32971                         return;
32972                     }
32973                 }
32974             }else{
32975                  i = accessList.indexOf(dlg);
32976                 if(i != -1){
32977                     accessList.splice(i, 1);
32978                 }
32979             }
32980         },
32981
32982         /**
32983          * Gets a registered dialog by id
32984          * @param {String/Object} id The id of the dialog or a dialog
32985          * @return {Roo.BasicDialog} this
32986          */
32987         get : function(id){
32988             return typeof id == "object" ? id : list[id];
32989         },
32990
32991         /**
32992          * Brings the specified dialog to the front
32993          * @param {String/Object} dlg The id of the dialog or a dialog
32994          * @return {Roo.BasicDialog} this
32995          */
32996         bringToFront : function(dlg){
32997             dlg = this.get(dlg);
32998             if(dlg != front){
32999                 front = dlg;
33000                 dlg._lastAccess = new Date().getTime();
33001                 orderDialogs();
33002             }
33003             return dlg;
33004         },
33005
33006         /**
33007          * Sends the specified dialog to the back
33008          * @param {String/Object} dlg The id of the dialog or a dialog
33009          * @return {Roo.BasicDialog} this
33010          */
33011         sendToBack : function(dlg){
33012             dlg = this.get(dlg);
33013             dlg._lastAccess = -(new Date().getTime());
33014             orderDialogs();
33015             return dlg;
33016         },
33017
33018         /**
33019          * Hides all dialogs
33020          */
33021         hideAll : function(){
33022             for(var id in list){
33023                 if(list[id] && typeof list[id] != "function" && list[id].isVisible()){
33024                     list[id].hide();
33025                 }
33026             }
33027         }
33028     };
33029 }();
33030
33031 /**
33032  * @class Roo.LayoutDialog
33033  * @extends Roo.BasicDialog
33034  * Dialog which provides adjustments for working with a layout in a Dialog.
33035  * Add your necessary layout config options to the dialog's config.<br>
33036  * Example usage (including a nested layout):
33037  * <pre><code>
33038 if(!dialog){
33039     dialog = new Roo.LayoutDialog("download-dlg", {
33040         modal: true,
33041         width:600,
33042         height:450,
33043         shadow:true,
33044         minWidth:500,
33045         minHeight:350,
33046         autoTabs:true,
33047         proxyDrag:true,
33048         // layout config merges with the dialog config
33049         center:{
33050             tabPosition: "top",
33051             alwaysShowTabs: true
33052         }
33053     });
33054     dialog.addKeyListener(27, dialog.hide, dialog);
33055     dialog.setDefaultButton(dialog.addButton("Close", dialog.hide, dialog));
33056     dialog.addButton("Build It!", this.getDownload, this);
33057
33058     // we can even add nested layouts
33059     var innerLayout = new Roo.BorderLayout("dl-inner", {
33060         east: {
33061             initialSize: 200,
33062             autoScroll:true,
33063             split:true
33064         },
33065         center: {
33066             autoScroll:true
33067         }
33068     });
33069     innerLayout.beginUpdate();
33070     innerLayout.add("east", new Roo.ContentPanel("dl-details"));
33071     innerLayout.add("center", new Roo.ContentPanel("selection-panel"));
33072     innerLayout.endUpdate(true);
33073
33074     var layout = dialog.getLayout();
33075     layout.beginUpdate();
33076     layout.add("center", new Roo.ContentPanel("standard-panel",
33077                         {title: "Download the Source", fitToFrame:true}));
33078     layout.add("center", new Roo.NestedLayoutPanel(innerLayout,
33079                {title: "Build your own roo.js"}));
33080     layout.getRegion("center").showPanel(sp);
33081     layout.endUpdate();
33082 }
33083 </code></pre>
33084     * @constructor
33085     * @param {String/HTMLElement/Roo.Element} el The id of or container element, or config
33086     * @param {Object} config configuration options
33087   */
33088 Roo.LayoutDialog = function(el, cfg){
33089     
33090     var config=  cfg;
33091     if (typeof(cfg) == 'undefined') {
33092         config = Roo.apply({}, el);
33093         // not sure why we use documentElement here.. - it should always be body.
33094         // IE7 borks horribly if we use documentElement.
33095         // webkit also does not like documentElement - it creates a body element...
33096         el = Roo.get( document.body || document.documentElement ).createChild();
33097         //config.autoCreate = true;
33098     }
33099     
33100     
33101     config.autoTabs = false;
33102     Roo.LayoutDialog.superclass.constructor.call(this, el, config);
33103     this.body.setStyle({overflow:"hidden", position:"relative"});
33104     this.layout = new Roo.BorderLayout(this.body.dom, config);
33105     this.layout.monitorWindowResize = false;
33106     this.el.addClass("x-dlg-auto-layout");
33107     // fix case when center region overwrites center function
33108     this.center = Roo.BasicDialog.prototype.center;
33109     this.on("show", this.layout.layout, this.layout, true);
33110     if (config.items) {
33111         var xitems = config.items;
33112         delete config.items;
33113         Roo.each(xitems, this.addxtype, this);
33114     }
33115     
33116     
33117 };
33118 Roo.extend(Roo.LayoutDialog, Roo.BasicDialog, {
33119     /**
33120      * Ends update of the layout <strike>and resets display to none</strike>. Use standard beginUpdate/endUpdate on the layout.
33121      * @deprecated
33122      */
33123     endUpdate : function(){
33124         this.layout.endUpdate();
33125     },
33126
33127     /**
33128      * Begins an update of the layout <strike>and sets display to block and visibility to hidden</strike>. Use standard beginUpdate/endUpdate on the layout.
33129      *  @deprecated
33130      */
33131     beginUpdate : function(){
33132         this.layout.beginUpdate();
33133     },
33134
33135     /**
33136      * Get the BorderLayout for this dialog
33137      * @return {Roo.BorderLayout}
33138      */
33139     getLayout : function(){
33140         return this.layout;
33141     },
33142
33143     showEl : function(){
33144         Roo.LayoutDialog.superclass.showEl.apply(this, arguments);
33145         if(Roo.isIE7){
33146             this.layout.layout();
33147         }
33148     },
33149
33150     // private
33151     // Use the syncHeightBeforeShow config option to control this automatically
33152     syncBodyHeight : function(){
33153         Roo.LayoutDialog.superclass.syncBodyHeight.call(this);
33154         if(this.layout){this.layout.layout();}
33155     },
33156     
33157       /**
33158      * Add an xtype element (actually adds to the layout.)
33159      * @return {Object} xdata xtype object data.
33160      */
33161     
33162     addxtype : function(c) {
33163         return this.layout.addxtype(c);
33164     }
33165 });/*
33166  * Based on:
33167  * Ext JS Library 1.1.1
33168  * Copyright(c) 2006-2007, Ext JS, LLC.
33169  *
33170  * Originally Released Under LGPL - original licence link has changed is not relivant.
33171  *
33172  * Fork - LGPL
33173  * <script type="text/javascript">
33174  */
33175  
33176 /**
33177  * @class Roo.MessageBox
33178  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
33179  * Example usage:
33180  *<pre><code>
33181 // Basic alert:
33182 Roo.Msg.alert('Status', 'Changes saved successfully.');
33183
33184 // Prompt for user data:
33185 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
33186     if (btn == 'ok'){
33187         // process text value...
33188     }
33189 });
33190
33191 // Show a dialog using config options:
33192 Roo.Msg.show({
33193    title:'Save Changes?',
33194    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
33195    buttons: Roo.Msg.YESNOCANCEL,
33196    fn: processResult,
33197    animEl: 'elId'
33198 });
33199 </code></pre>
33200  * @singleton
33201  */
33202 Roo.MessageBox = function(){
33203     var dlg, opt, mask, waitTimer;
33204     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
33205     var buttons, activeTextEl, bwidth;
33206
33207     // private
33208     var handleButton = function(button){
33209         dlg.hide();
33210         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
33211     };
33212
33213     // private
33214     var handleHide = function(){
33215         if(opt && opt.cls){
33216             dlg.el.removeClass(opt.cls);
33217         }
33218         if(waitTimer){
33219             Roo.TaskMgr.stop(waitTimer);
33220             waitTimer = null;
33221         }
33222     };
33223
33224     // private
33225     var updateButtons = function(b){
33226         var width = 0;
33227         if(!b){
33228             buttons["ok"].hide();
33229             buttons["cancel"].hide();
33230             buttons["yes"].hide();
33231             buttons["no"].hide();
33232             dlg.footer.dom.style.display = 'none';
33233             return width;
33234         }
33235         dlg.footer.dom.style.display = '';
33236         for(var k in buttons){
33237             if(typeof buttons[k] != "function"){
33238                 if(b[k]){
33239                     buttons[k].show();
33240                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
33241                     width += buttons[k].el.getWidth()+15;
33242                 }else{
33243                     buttons[k].hide();
33244                 }
33245             }
33246         }
33247         return width;
33248     };
33249
33250     // private
33251     var handleEsc = function(d, k, e){
33252         if(opt && opt.closable !== false){
33253             dlg.hide();
33254         }
33255         if(e){
33256             e.stopEvent();
33257         }
33258     };
33259
33260     return {
33261         /**
33262          * Returns a reference to the underlying {@link Roo.BasicDialog} element
33263          * @return {Roo.BasicDialog} The BasicDialog element
33264          */
33265         getDialog : function(){
33266            if(!dlg){
33267                 dlg = new Roo.BasicDialog("x-msg-box", {
33268                     autoCreate : true,
33269                     shadow: true,
33270                     draggable: true,
33271                     resizable:false,
33272                     constraintoviewport:false,
33273                     fixedcenter:true,
33274                     collapsible : false,
33275                     shim:true,
33276                     modal: true,
33277                     width:400, height:100,
33278                     buttonAlign:"center",
33279                     closeClick : function(){
33280                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
33281                             handleButton("no");
33282                         }else{
33283                             handleButton("cancel");
33284                         }
33285                     }
33286                 });
33287                 dlg.on("hide", handleHide);
33288                 mask = dlg.mask;
33289                 dlg.addKeyListener(27, handleEsc);
33290                 buttons = {};
33291                 var bt = this.buttonText;
33292                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
33293                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
33294                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
33295                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
33296                 bodyEl = dlg.body.createChild({
33297
33298                     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>'
33299                 });
33300                 msgEl = bodyEl.dom.firstChild;
33301                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
33302                 textboxEl.enableDisplayMode();
33303                 textboxEl.addKeyListener([10,13], function(){
33304                     if(dlg.isVisible() && opt && opt.buttons){
33305                         if(opt.buttons.ok){
33306                             handleButton("ok");
33307                         }else if(opt.buttons.yes){
33308                             handleButton("yes");
33309                         }
33310                     }
33311                 });
33312                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
33313                 textareaEl.enableDisplayMode();
33314                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
33315                 progressEl.enableDisplayMode();
33316                 var pf = progressEl.dom.firstChild;
33317                 if (pf) {
33318                     pp = Roo.get(pf.firstChild);
33319                     pp.setHeight(pf.offsetHeight);
33320                 }
33321                 
33322             }
33323             return dlg;
33324         },
33325
33326         /**
33327          * Updates the message box body text
33328          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
33329          * the XHTML-compliant non-breaking space character '&amp;#160;')
33330          * @return {Roo.MessageBox} This message box
33331          */
33332         updateText : function(text){
33333             if(!dlg.isVisible() && !opt.width){
33334                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
33335             }
33336             msgEl.innerHTML = text || '&#160;';
33337       
33338             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
33339             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
33340             var w = Math.max(
33341                     Math.min(opt.width || cw , this.maxWidth), 
33342                     Math.max(opt.minWidth || this.minWidth, bwidth)
33343             );
33344             if(opt.prompt){
33345                 activeTextEl.setWidth(w);
33346             }
33347             if(dlg.isVisible()){
33348                 dlg.fixedcenter = false;
33349             }
33350             // to big, make it scroll. = But as usual stupid IE does not support
33351             // !important..
33352             
33353             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
33354                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
33355                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
33356             } else {
33357                 bodyEl.dom.style.height = '';
33358                 bodyEl.dom.style.overflowY = '';
33359             }
33360             if (cw > w) {
33361                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
33362             } else {
33363                 bodyEl.dom.style.overflowX = '';
33364             }
33365             
33366             dlg.setContentSize(w, bodyEl.getHeight());
33367             if(dlg.isVisible()){
33368                 dlg.fixedcenter = true;
33369             }
33370             return this;
33371         },
33372
33373         /**
33374          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
33375          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
33376          * @param {Number} value Any number between 0 and 1 (e.g., .5)
33377          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
33378          * @return {Roo.MessageBox} This message box
33379          */
33380         updateProgress : function(value, text){
33381             if(text){
33382                 this.updateText(text);
33383             }
33384             if (pp) { // weird bug on my firefox - for some reason this is not defined
33385                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
33386             }
33387             return this;
33388         },        
33389
33390         /**
33391          * Returns true if the message box is currently displayed
33392          * @return {Boolean} True if the message box is visible, else false
33393          */
33394         isVisible : function(){
33395             return dlg && dlg.isVisible();  
33396         },
33397
33398         /**
33399          * Hides the message box if it is displayed
33400          */
33401         hide : function(){
33402             if(this.isVisible()){
33403                 dlg.hide();
33404             }  
33405         },
33406
33407         /**
33408          * Displays a new message box, or reinitializes an existing message box, based on the config options
33409          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
33410          * The following config object properties are supported:
33411          * <pre>
33412 Property    Type             Description
33413 ----------  ---------------  ------------------------------------------------------------------------------------
33414 animEl            String/Element   An id or Element from which the message box should animate as it opens and
33415                                    closes (defaults to undefined)
33416 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
33417                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
33418 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
33419                                    progress and wait dialogs will ignore this property and always hide the
33420                                    close button as they can only be closed programmatically.
33421 cls               String           A custom CSS class to apply to the message box element
33422 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
33423                                    displayed (defaults to 75)
33424 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
33425                                    function will be btn (the name of the button that was clicked, if applicable,
33426                                    e.g. "ok"), and text (the value of the active text field, if applicable).
33427                                    Progress and wait dialogs will ignore this option since they do not respond to
33428                                    user actions and can only be closed programmatically, so any required function
33429                                    should be called by the same code after it closes the dialog.
33430 icon              String           A CSS class that provides a background image to be used as an icon for
33431                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
33432 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
33433 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
33434 modal             Boolean          False to allow user interaction with the page while the message box is
33435                                    displayed (defaults to true)
33436 msg               String           A string that will replace the existing message box body text (defaults
33437                                    to the XHTML-compliant non-breaking space character '&#160;')
33438 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
33439 progress          Boolean          True to display a progress bar (defaults to false)
33440 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
33441 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
33442 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
33443 title             String           The title text
33444 value             String           The string value to set into the active textbox element if displayed
33445 wait              Boolean          True to display a progress bar (defaults to false)
33446 width             Number           The width of the dialog in pixels
33447 </pre>
33448          *
33449          * Example usage:
33450          * <pre><code>
33451 Roo.Msg.show({
33452    title: 'Address',
33453    msg: 'Please enter your address:',
33454    width: 300,
33455    buttons: Roo.MessageBox.OKCANCEL,
33456    multiline: true,
33457    fn: saveAddress,
33458    animEl: 'addAddressBtn'
33459 });
33460 </code></pre>
33461          * @param {Object} config Configuration options
33462          * @return {Roo.MessageBox} This message box
33463          */
33464         show : function(options)
33465         {
33466             
33467             // this causes nightmares if you show one dialog after another
33468             // especially on callbacks..
33469              
33470             if(this.isVisible()){
33471                 
33472                 this.hide();
33473                 Roo.log("[Roo.Messagebox] Show called while message displayed:" );
33474                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML );
33475                 Roo.log("New Dialog Message:" +  options.msg )
33476                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
33477                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
33478                 
33479             }
33480             var d = this.getDialog();
33481             opt = options;
33482             d.setTitle(opt.title || "&#160;");
33483             d.close.setDisplayed(opt.closable !== false);
33484             activeTextEl = textboxEl;
33485             opt.prompt = opt.prompt || (opt.multiline ? true : false);
33486             if(opt.prompt){
33487                 if(opt.multiline){
33488                     textboxEl.hide();
33489                     textareaEl.show();
33490                     textareaEl.setHeight(typeof opt.multiline == "number" ?
33491                         opt.multiline : this.defaultTextHeight);
33492                     activeTextEl = textareaEl;
33493                 }else{
33494                     textboxEl.show();
33495                     textareaEl.hide();
33496                 }
33497             }else{
33498                 textboxEl.hide();
33499                 textareaEl.hide();
33500             }
33501             progressEl.setDisplayed(opt.progress === true);
33502             this.updateProgress(0);
33503             activeTextEl.dom.value = opt.value || "";
33504             if(opt.prompt){
33505                 dlg.setDefaultButton(activeTextEl);
33506             }else{
33507                 var bs = opt.buttons;
33508                 var db = null;
33509                 if(bs && bs.ok){
33510                     db = buttons["ok"];
33511                 }else if(bs && bs.yes){
33512                     db = buttons["yes"];
33513                 }
33514                 dlg.setDefaultButton(db);
33515             }
33516             bwidth = updateButtons(opt.buttons);
33517             this.updateText(opt.msg);
33518             if(opt.cls){
33519                 d.el.addClass(opt.cls);
33520             }
33521             d.proxyDrag = opt.proxyDrag === true;
33522             d.modal = opt.modal !== false;
33523             d.mask = opt.modal !== false ? mask : false;
33524             if(!d.isVisible()){
33525                 // force it to the end of the z-index stack so it gets a cursor in FF
33526                 document.body.appendChild(dlg.el.dom);
33527                 d.animateTarget = null;
33528                 d.show(options.animEl);
33529             }
33530             return this;
33531         },
33532
33533         /**
33534          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
33535          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
33536          * and closing the message box when the process is complete.
33537          * @param {String} title The title bar text
33538          * @param {String} msg The message box body text
33539          * @return {Roo.MessageBox} This message box
33540          */
33541         progress : function(title, msg){
33542             this.show({
33543                 title : title,
33544                 msg : msg,
33545                 buttons: false,
33546                 progress:true,
33547                 closable:false,
33548                 minWidth: this.minProgressWidth,
33549                 modal : true
33550             });
33551             return this;
33552         },
33553
33554         /**
33555          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
33556          * If a callback function is passed it will be called after the user clicks the button, and the
33557          * id of the button that was clicked will be passed as the only parameter to the callback
33558          * (could also be the top-right close button).
33559          * @param {String} title The title bar text
33560          * @param {String} msg The message box body text
33561          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33562          * @param {Object} scope (optional) The scope of the callback function
33563          * @return {Roo.MessageBox} This message box
33564          */
33565         alert : function(title, msg, fn, scope){
33566             this.show({
33567                 title : title,
33568                 msg : msg,
33569                 buttons: this.OK,
33570                 fn: fn,
33571                 scope : scope,
33572                 modal : true
33573             });
33574             return this;
33575         },
33576
33577         /**
33578          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
33579          * interaction while waiting for a long-running process to complete that does not have defined intervals.
33580          * You are responsible for closing the message box when the process is complete.
33581          * @param {String} msg The message box body text
33582          * @param {String} title (optional) The title bar text
33583          * @return {Roo.MessageBox} This message box
33584          */
33585         wait : function(msg, title){
33586             this.show({
33587                 title : title,
33588                 msg : msg,
33589                 buttons: false,
33590                 closable:false,
33591                 progress:true,
33592                 modal:true,
33593                 width:300,
33594                 wait:true
33595             });
33596             waitTimer = Roo.TaskMgr.start({
33597                 run: function(i){
33598                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
33599                 },
33600                 interval: 1000
33601             });
33602             return this;
33603         },
33604
33605         /**
33606          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
33607          * If a callback function is passed it will be called after the user clicks either button, and the id of the
33608          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
33609          * @param {String} title The title bar text
33610          * @param {String} msg The message box body text
33611          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33612          * @param {Object} scope (optional) The scope of the callback function
33613          * @return {Roo.MessageBox} This message box
33614          */
33615         confirm : function(title, msg, fn, scope){
33616             this.show({
33617                 title : title,
33618                 msg : msg,
33619                 buttons: this.YESNO,
33620                 fn: fn,
33621                 scope : scope,
33622                 modal : true
33623             });
33624             return this;
33625         },
33626
33627         /**
33628          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
33629          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
33630          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
33631          * (could also be the top-right close button) and the text that was entered will be passed as the two
33632          * parameters to the callback.
33633          * @param {String} title The title bar text
33634          * @param {String} msg The message box body text
33635          * @param {Function} fn (optional) The callback function invoked after the message box is closed
33636          * @param {Object} scope (optional) The scope of the callback function
33637          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
33638          * property, or the height in pixels to create the textbox (defaults to false / single-line)
33639          * @return {Roo.MessageBox} This message box
33640          */
33641         prompt : function(title, msg, fn, scope, multiline){
33642             this.show({
33643                 title : title,
33644                 msg : msg,
33645                 buttons: this.OKCANCEL,
33646                 fn: fn,
33647                 minWidth:250,
33648                 scope : scope,
33649                 prompt:true,
33650                 multiline: multiline,
33651                 modal : true
33652             });
33653             return this;
33654         },
33655
33656         /**
33657          * Button config that displays a single OK button
33658          * @type Object
33659          */
33660         OK : {ok:true},
33661         /**
33662          * Button config that displays Yes and No buttons
33663          * @type Object
33664          */
33665         YESNO : {yes:true, no:true},
33666         /**
33667          * Button config that displays OK and Cancel buttons
33668          * @type Object
33669          */
33670         OKCANCEL : {ok:true, cancel:true},
33671         /**
33672          * Button config that displays Yes, No and Cancel buttons
33673          * @type Object
33674          */
33675         YESNOCANCEL : {yes:true, no:true, cancel:true},
33676
33677         /**
33678          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
33679          * @type Number
33680          */
33681         defaultTextHeight : 75,
33682         /**
33683          * The maximum width in pixels of the message box (defaults to 600)
33684          * @type Number
33685          */
33686         maxWidth : 600,
33687         /**
33688          * The minimum width in pixels of the message box (defaults to 100)
33689          * @type Number
33690          */
33691         minWidth : 100,
33692         /**
33693          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
33694          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
33695          * @type Number
33696          */
33697         minProgressWidth : 250,
33698         /**
33699          * An object containing the default button text strings that can be overriden for localized language support.
33700          * Supported properties are: ok, cancel, yes and no.
33701          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
33702          * @type Object
33703          */
33704         buttonText : {
33705             ok : "OK",
33706             cancel : "Cancel",
33707             yes : "Yes",
33708             no : "No"
33709         }
33710     };
33711 }();
33712
33713 /**
33714  * Shorthand for {@link Roo.MessageBox}
33715  */
33716 Roo.Msg = Roo.MessageBox;/*
33717  * Based on:
33718  * Ext JS Library 1.1.1
33719  * Copyright(c) 2006-2007, Ext JS, LLC.
33720  *
33721  * Originally Released Under LGPL - original licence link has changed is not relivant.
33722  *
33723  * Fork - LGPL
33724  * <script type="text/javascript">
33725  */
33726 /**
33727  * @class Roo.QuickTips
33728  * Provides attractive and customizable tooltips for any element.
33729  * @singleton
33730  */
33731 Roo.QuickTips = function(){
33732     var el, tipBody, tipBodyText, tipTitle, tm, cfg, close, tagEls = {}, esc, removeCls = null, bdLeft, bdRight;
33733     var ce, bd, xy, dd;
33734     var visible = false, disabled = true, inited = false;
33735     var showProc = 1, hideProc = 1, dismissProc = 1, locks = [];
33736     
33737     var onOver = function(e){
33738         if(disabled){
33739             return;
33740         }
33741         var t = e.getTarget();
33742         if(!t || t.nodeType !== 1 || t == document || t == document.body){
33743             return;
33744         }
33745         if(ce && t == ce.el){
33746             clearTimeout(hideProc);
33747             return;
33748         }
33749         if(t && tagEls[t.id]){
33750             tagEls[t.id].el = t;
33751             showProc = show.defer(tm.showDelay, tm, [tagEls[t.id]]);
33752             return;
33753         }
33754         var ttp, et = Roo.fly(t);
33755         var ns = cfg.namespace;
33756         if(tm.interceptTitles && t.title){
33757             ttp = t.title;
33758             t.qtip = ttp;
33759             t.removeAttribute("title");
33760             e.preventDefault();
33761         }else{
33762             ttp = t.qtip || et.getAttributeNS(ns, cfg.attribute) || et.getAttributeNS(cfg.alt_namespace, cfg.attribute) ;
33763         }
33764         if(ttp){
33765             showProc = show.defer(tm.showDelay, tm, [{
33766                 el: t, 
33767                 text: ttp.replace(/\\n/g,'<br/>'),
33768                 width: et.getAttributeNS(ns, cfg.width),
33769                 autoHide: et.getAttributeNS(ns, cfg.hide) != "user",
33770                 title: et.getAttributeNS(ns, cfg.title),
33771                     cls: et.getAttributeNS(ns, cfg.cls)
33772             }]);
33773         }
33774     };
33775     
33776     var onOut = function(e){
33777         clearTimeout(showProc);
33778         var t = e.getTarget();
33779         if(t && ce && ce.el == t && (tm.autoHide && ce.autoHide !== false)){
33780             hideProc = setTimeout(hide, tm.hideDelay);
33781         }
33782     };
33783     
33784     var onMove = function(e){
33785         if(disabled){
33786             return;
33787         }
33788         xy = e.getXY();
33789         xy[1] += 18;
33790         if(tm.trackMouse && ce){
33791             el.setXY(xy);
33792         }
33793     };
33794     
33795     var onDown = function(e){
33796         clearTimeout(showProc);
33797         clearTimeout(hideProc);
33798         if(!e.within(el)){
33799             if(tm.hideOnClick){
33800                 hide();
33801                 tm.disable();
33802                 tm.enable.defer(100, tm);
33803             }
33804         }
33805     };
33806     
33807     var getPad = function(){
33808         return 2;//bdLeft.getPadding('l')+bdRight.getPadding('r');
33809     };
33810
33811     var show = function(o){
33812         if(disabled){
33813             return;
33814         }
33815         clearTimeout(dismissProc);
33816         ce = o;
33817         if(removeCls){ // in case manually hidden
33818             el.removeClass(removeCls);
33819             removeCls = null;
33820         }
33821         if(ce.cls){
33822             el.addClass(ce.cls);
33823             removeCls = ce.cls;
33824         }
33825         if(ce.title){
33826             tipTitle.update(ce.title);
33827             tipTitle.show();
33828         }else{
33829             tipTitle.update('');
33830             tipTitle.hide();
33831         }
33832         el.dom.style.width  = tm.maxWidth+'px';
33833         //tipBody.dom.style.width = '';
33834         tipBodyText.update(o.text);
33835         var p = getPad(), w = ce.width;
33836         if(!w){
33837             var td = tipBodyText.dom;
33838             var aw = Math.max(td.offsetWidth, td.clientWidth, td.scrollWidth);
33839             if(aw > tm.maxWidth){
33840                 w = tm.maxWidth;
33841             }else if(aw < tm.minWidth){
33842                 w = tm.minWidth;
33843             }else{
33844                 w = aw;
33845             }
33846         }
33847         //tipBody.setWidth(w);
33848         el.setWidth(parseInt(w, 10) + p);
33849         if(ce.autoHide === false){
33850             close.setDisplayed(true);
33851             if(dd){
33852                 dd.unlock();
33853             }
33854         }else{
33855             close.setDisplayed(false);
33856             if(dd){
33857                 dd.lock();
33858             }
33859         }
33860         if(xy){
33861             el.avoidY = xy[1]-18;
33862             el.setXY(xy);
33863         }
33864         if(tm.animate){
33865             el.setOpacity(.1);
33866             el.setStyle("visibility", "visible");
33867             el.fadeIn({callback: afterShow});
33868         }else{
33869             afterShow();
33870         }
33871     };
33872     
33873     var afterShow = function(){
33874         if(ce){
33875             el.show();
33876             esc.enable();
33877             if(tm.autoDismiss && ce.autoHide !== false){
33878                 dismissProc = setTimeout(hide, tm.autoDismissDelay);
33879             }
33880         }
33881     };
33882     
33883     var hide = function(noanim){
33884         clearTimeout(dismissProc);
33885         clearTimeout(hideProc);
33886         ce = null;
33887         if(el.isVisible()){
33888             esc.disable();
33889             if(noanim !== true && tm.animate){
33890                 el.fadeOut({callback: afterHide});
33891             }else{
33892                 afterHide();
33893             } 
33894         }
33895     };
33896     
33897     var afterHide = function(){
33898         el.hide();
33899         if(removeCls){
33900             el.removeClass(removeCls);
33901             removeCls = null;
33902         }
33903     };
33904     
33905     return {
33906         /**
33907         * @cfg {Number} minWidth
33908         * The minimum width of the quick tip (defaults to 40)
33909         */
33910        minWidth : 40,
33911         /**
33912         * @cfg {Number} maxWidth
33913         * The maximum width of the quick tip (defaults to 300)
33914         */
33915        maxWidth : 300,
33916         /**
33917         * @cfg {Boolean} interceptTitles
33918         * True to automatically use the element's DOM title value if available (defaults to false)
33919         */
33920        interceptTitles : false,
33921         /**
33922         * @cfg {Boolean} trackMouse
33923         * True to have the quick tip follow the mouse as it moves over the target element (defaults to false)
33924         */
33925        trackMouse : false,
33926         /**
33927         * @cfg {Boolean} hideOnClick
33928         * True to hide the quick tip if the user clicks anywhere in the document (defaults to true)
33929         */
33930        hideOnClick : true,
33931         /**
33932         * @cfg {Number} showDelay
33933         * Delay in milliseconds before the quick tip displays after the mouse enters the target element (defaults to 500)
33934         */
33935        showDelay : 500,
33936         /**
33937         * @cfg {Number} hideDelay
33938         * Delay in milliseconds before the quick tip hides when autoHide = true (defaults to 200)
33939         */
33940        hideDelay : 200,
33941         /**
33942         * @cfg {Boolean} autoHide
33943         * True to automatically hide the quick tip after the mouse exits the target element (defaults to true).
33944         * Used in conjunction with hideDelay.
33945         */
33946        autoHide : true,
33947         /**
33948         * @cfg {Boolean}
33949         * True to automatically hide the quick tip after a set period of time, regardless of the user's actions
33950         * (defaults to true).  Used in conjunction with autoDismissDelay.
33951         */
33952        autoDismiss : true,
33953         /**
33954         * @cfg {Number}
33955         * Delay in milliseconds before the quick tip hides when autoDismiss = true (defaults to 5000)
33956         */
33957        autoDismissDelay : 5000,
33958        /**
33959         * @cfg {Boolean} animate
33960         * True to turn on fade animation. Defaults to false (ClearType/scrollbar flicker issues in IE7).
33961         */
33962        animate : false,
33963
33964        /**
33965         * @cfg {String} title
33966         * Title text to display (defaults to '').  This can be any valid HTML markup.
33967         */
33968         title: '',
33969        /**
33970         * @cfg {String} text
33971         * Body text to display (defaults to '').  This can be any valid HTML markup.
33972         */
33973         text : '',
33974        /**
33975         * @cfg {String} cls
33976         * A CSS class to apply to the base quick tip element (defaults to '').
33977         */
33978         cls : '',
33979        /**
33980         * @cfg {Number} width
33981         * Width in pixels of the quick tip (defaults to auto).  Width will be ignored if it exceeds the bounds of
33982         * minWidth or maxWidth.
33983         */
33984         width : null,
33985
33986     /**
33987      * Initialize and enable QuickTips for first use.  This should be called once before the first attempt to access
33988      * or display QuickTips in a page.
33989      */
33990        init : function(){
33991           tm = Roo.QuickTips;
33992           cfg = tm.tagConfig;
33993           if(!inited){
33994               if(!Roo.isReady){ // allow calling of init() before onReady
33995                   Roo.onReady(Roo.QuickTips.init, Roo.QuickTips);
33996                   return;
33997               }
33998               el = new Roo.Layer({cls:"x-tip", shadow:"drop", shim: true, constrain:true, shadowOffset:4});
33999               el.fxDefaults = {stopFx: true};
34000               // maximum custom styling
34001               //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>');
34002               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>');              
34003               tipTitle = el.child('h3');
34004               tipTitle.enableDisplayMode("block");
34005               tipBody = el.child('div.x-tip-bd');
34006               tipBodyText = el.child('div.x-tip-bd-inner');
34007               //bdLeft = el.child('div.x-tip-bd-left');
34008               //bdRight = el.child('div.x-tip-bd-right');
34009               close = el.child('div.x-tip-close');
34010               close.enableDisplayMode("block");
34011               close.on("click", hide);
34012               var d = Roo.get(document);
34013               d.on("mousedown", onDown);
34014               d.on("mouseover", onOver);
34015               d.on("mouseout", onOut);
34016               d.on("mousemove", onMove);
34017               esc = d.addKeyListener(27, hide);
34018               esc.disable();
34019               if(Roo.dd.DD){
34020                   dd = el.initDD("default", null, {
34021                       onDrag : function(){
34022                           el.sync();  
34023                       }
34024                   });
34025                   dd.setHandleElId(tipTitle.id);
34026                   dd.lock();
34027               }
34028               inited = true;
34029           }
34030           this.enable(); 
34031        },
34032
34033     /**
34034      * Configures a new quick tip instance and assigns it to a target element.  The following config options
34035      * are supported:
34036      * <pre>
34037 Property    Type                   Description
34038 ----------  ---------------------  ------------------------------------------------------------------------
34039 target      Element/String/Array   An Element, id or array of ids that this quick tip should be tied to
34040      * </ul>
34041      * @param {Object} config The config object
34042      */
34043        register : function(config){
34044            var cs = config instanceof Array ? config : arguments;
34045            for(var i = 0, len = cs.length; i < len; i++) {
34046                var c = cs[i];
34047                var target = c.target;
34048                if(target){
34049                    if(target instanceof Array){
34050                        for(var j = 0, jlen = target.length; j < jlen; j++){
34051                            tagEls[target[j]] = c;
34052                        }
34053                    }else{
34054                        tagEls[typeof target == 'string' ? target : Roo.id(target)] = c;
34055                    }
34056                }
34057            }
34058        },
34059
34060     /**
34061      * Removes this quick tip from its element and destroys it.
34062      * @param {String/HTMLElement/Element} el The element from which the quick tip is to be removed.
34063      */
34064        unregister : function(el){
34065            delete tagEls[Roo.id(el)];
34066        },
34067
34068     /**
34069      * Enable this quick tip.
34070      */
34071        enable : function(){
34072            if(inited && disabled){
34073                locks.pop();
34074                if(locks.length < 1){
34075                    disabled = false;
34076                }
34077            }
34078        },
34079
34080     /**
34081      * Disable this quick tip.
34082      */
34083        disable : function(){
34084           disabled = true;
34085           clearTimeout(showProc);
34086           clearTimeout(hideProc);
34087           clearTimeout(dismissProc);
34088           if(ce){
34089               hide(true);
34090           }
34091           locks.push(1);
34092        },
34093
34094     /**
34095      * Returns true if the quick tip is enabled, else false.
34096      */
34097        isEnabled : function(){
34098             return !disabled;
34099        },
34100
34101         // private
34102        tagConfig : {
34103            namespace : "roo", // was ext?? this may break..
34104            alt_namespace : "ext",
34105            attribute : "qtip",
34106            width : "width",
34107            target : "target",
34108            title : "qtitle",
34109            hide : "hide",
34110            cls : "qclass"
34111        }
34112    };
34113 }();
34114
34115 // backwards compat
34116 Roo.QuickTips.tips = Roo.QuickTips.register;/*
34117  * Based on:
34118  * Ext JS Library 1.1.1
34119  * Copyright(c) 2006-2007, Ext JS, LLC.
34120  *
34121  * Originally Released Under LGPL - original licence link has changed is not relivant.
34122  *
34123  * Fork - LGPL
34124  * <script type="text/javascript">
34125  */
34126  
34127
34128 /**
34129  * @class Roo.tree.TreePanel
34130  * @extends Roo.data.Tree
34131
34132  * @cfg {Boolean} rootVisible false to hide the root node (defaults to true)
34133  * @cfg {Boolean} lines false to disable tree lines (defaults to true)
34134  * @cfg {Boolean} enableDD true to enable drag and drop
34135  * @cfg {Boolean} enableDrag true to enable just drag
34136  * @cfg {Boolean} enableDrop true to enable just drop
34137  * @cfg {Object} dragConfig Custom config to pass to the {@link Roo.tree.TreeDragZone} instance
34138  * @cfg {Object} dropConfig Custom config to pass to the {@link Roo.tree.TreeDropZone} instance
34139  * @cfg {String} ddGroup The DD group this TreePanel belongs to
34140  * @cfg {String} ddAppendOnly True if the tree should only allow append drops (use for trees which are sorted)
34141  * @cfg {Boolean} ddScroll true to enable YUI body scrolling
34142  * @cfg {Boolean} containerScroll true to register this container with ScrollManager
34143  * @cfg {Boolean} hlDrop false to disable node highlight on drop (defaults to the value of Roo.enableFx)
34144  * @cfg {String} hlColor The color of the node highlight (defaults to C3DAF9)
34145  * @cfg {Boolean} animate true to enable animated expand/collapse (defaults to the value of Roo.enableFx)
34146  * @cfg {Boolean} singleExpand true if only 1 node per branch may be expanded
34147  * @cfg {Boolean} selModel A tree selection model to use with this TreePanel (defaults to a {@link Roo.tree.DefaultSelectionModel})
34148  * @cfg {Boolean} loader A TreeLoader for use with this TreePanel
34149  * @cfg {Object|Roo.tree.TreeEditor} editor The TreeEditor or xtype data to display when clicked.
34150  * @cfg {String} pathSeparator The token used to separate sub-paths in path strings (defaults to '/')
34151  * @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>
34152  * @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>
34153  * 
34154  * @constructor
34155  * @param {String/HTMLElement/Element} el The container element
34156  * @param {Object} config
34157  */
34158 Roo.tree.TreePanel = function(el, config){
34159     var root = false;
34160     var loader = false;
34161     if (config.root) {
34162         root = config.root;
34163         delete config.root;
34164     }
34165     if (config.loader) {
34166         loader = config.loader;
34167         delete config.loader;
34168     }
34169     
34170     Roo.apply(this, config);
34171     Roo.tree.TreePanel.superclass.constructor.call(this);
34172     this.el = Roo.get(el);
34173     this.el.addClass('x-tree');
34174     //console.log(root);
34175     if (root) {
34176         this.setRootNode( Roo.factory(root, Roo.tree));
34177     }
34178     if (loader) {
34179         this.loader = Roo.factory(loader, Roo.tree);
34180     }
34181    /**
34182     * Read-only. The id of the container element becomes this TreePanel's id.
34183     */
34184     this.id = this.el.id;
34185     this.addEvents({
34186         /**
34187         * @event beforeload
34188         * Fires before a node is loaded, return false to cancel
34189         * @param {Node} node The node being loaded
34190         */
34191         "beforeload" : true,
34192         /**
34193         * @event load
34194         * Fires when a node is loaded
34195         * @param {Node} node The node that was loaded
34196         */
34197         "load" : true,
34198         /**
34199         * @event textchange
34200         * Fires when the text for a node is changed
34201         * @param {Node} node The node
34202         * @param {String} text The new text
34203         * @param {String} oldText The old text
34204         */
34205         "textchange" : true,
34206         /**
34207         * @event beforeexpand
34208         * Fires before a node is expanded, return false to cancel.
34209         * @param {Node} node The node
34210         * @param {Boolean} deep
34211         * @param {Boolean} anim
34212         */
34213         "beforeexpand" : true,
34214         /**
34215         * @event beforecollapse
34216         * Fires before a node is collapsed, return false to cancel.
34217         * @param {Node} node The node
34218         * @param {Boolean} deep
34219         * @param {Boolean} anim
34220         */
34221         "beforecollapse" : true,
34222         /**
34223         * @event expand
34224         * Fires when a node is expanded
34225         * @param {Node} node The node
34226         */
34227         "expand" : true,
34228         /**
34229         * @event disabledchange
34230         * Fires when the disabled status of a node changes
34231         * @param {Node} node The node
34232         * @param {Boolean} disabled
34233         */
34234         "disabledchange" : true,
34235         /**
34236         * @event collapse
34237         * Fires when a node is collapsed
34238         * @param {Node} node The node
34239         */
34240         "collapse" : true,
34241         /**
34242         * @event beforeclick
34243         * Fires before click processing on a node. Return false to cancel the default action.
34244         * @param {Node} node The node
34245         * @param {Roo.EventObject} e The event object
34246         */
34247         "beforeclick":true,
34248         /**
34249         * @event checkchange
34250         * Fires when a node with a checkbox's checked property changes
34251         * @param {Node} this This node
34252         * @param {Boolean} checked
34253         */
34254         "checkchange":true,
34255         /**
34256         * @event click
34257         * Fires when a node is clicked
34258         * @param {Node} node The node
34259         * @param {Roo.EventObject} e The event object
34260         */
34261         "click":true,
34262         /**
34263         * @event dblclick
34264         * Fires when a node is double clicked
34265         * @param {Node} node The node
34266         * @param {Roo.EventObject} e The event object
34267         */
34268         "dblclick":true,
34269         /**
34270         * @event contextmenu
34271         * Fires when a node is right clicked
34272         * @param {Node} node The node
34273         * @param {Roo.EventObject} e The event object
34274         */
34275         "contextmenu":true,
34276         /**
34277         * @event beforechildrenrendered
34278         * Fires right before the child nodes for a node are rendered
34279         * @param {Node} node The node
34280         */
34281         "beforechildrenrendered":true,
34282         /**
34283         * @event startdrag
34284         * Fires when a node starts being dragged
34285         * @param {Roo.tree.TreePanel} this
34286         * @param {Roo.tree.TreeNode} node
34287         * @param {event} e The raw browser event
34288         */ 
34289        "startdrag" : true,
34290        /**
34291         * @event enddrag
34292         * Fires when a drag operation is complete
34293         * @param {Roo.tree.TreePanel} this
34294         * @param {Roo.tree.TreeNode} node
34295         * @param {event} e The raw browser event
34296         */
34297        "enddrag" : true,
34298        /**
34299         * @event dragdrop
34300         * Fires when a dragged node is dropped on a valid DD target
34301         * @param {Roo.tree.TreePanel} this
34302         * @param {Roo.tree.TreeNode} node
34303         * @param {DD} dd The dd it was dropped on
34304         * @param {event} e The raw browser event
34305         */
34306        "dragdrop" : true,
34307        /**
34308         * @event beforenodedrop
34309         * Fires when a DD object is dropped on a node in this tree for preprocessing. Return false to cancel the drop. The dropEvent
34310         * passed to handlers has the following properties:<br />
34311         * <ul style="padding:5px;padding-left:16px;">
34312         * <li>tree - The TreePanel</li>
34313         * <li>target - The node being targeted for the drop</li>
34314         * <li>data - The drag data from the drag source</li>
34315         * <li>point - The point of the drop - append, above or below</li>
34316         * <li>source - The drag source</li>
34317         * <li>rawEvent - Raw mouse event</li>
34318         * <li>dropNode - Drop node(s) provided by the source <b>OR</b> you can supply node(s)
34319         * to be inserted by setting them on this object.</li>
34320         * <li>cancel - Set this to true to cancel the drop.</li>
34321         * </ul>
34322         * @param {Object} dropEvent
34323         */
34324        "beforenodedrop" : true,
34325        /**
34326         * @event nodedrop
34327         * Fires after a DD object is dropped on a node in this tree. The dropEvent
34328         * passed to handlers has the following properties:<br />
34329         * <ul style="padding:5px;padding-left:16px;">
34330         * <li>tree - The TreePanel</li>
34331         * <li>target - The node being targeted for the drop</li>
34332         * <li>data - The drag data from the drag source</li>
34333         * <li>point - The point of the drop - append, above or below</li>
34334         * <li>source - The drag source</li>
34335         * <li>rawEvent - Raw mouse event</li>
34336         * <li>dropNode - Dropped node(s).</li>
34337         * </ul>
34338         * @param {Object} dropEvent
34339         */
34340        "nodedrop" : true,
34341         /**
34342         * @event nodedragover
34343         * Fires when a tree node is being targeted for a drag drop, return false to signal drop not allowed. The dragOverEvent
34344         * passed to handlers has the following properties:<br />
34345         * <ul style="padding:5px;padding-left:16px;">
34346         * <li>tree - The TreePanel</li>
34347         * <li>target - The node being targeted for the drop</li>
34348         * <li>data - The drag data from the drag source</li>
34349         * <li>point - The point of the drop - append, above or below</li>
34350         * <li>source - The drag source</li>
34351         * <li>rawEvent - Raw mouse event</li>
34352         * <li>dropNode - Drop node(s) provided by the source.</li>
34353         * <li>cancel - Set this to true to signal drop not allowed.</li>
34354         * </ul>
34355         * @param {Object} dragOverEvent
34356         */
34357        "nodedragover" : true,
34358        /**
34359         * @event appendnode
34360         * Fires when append node to the tree
34361         * @param {Roo.tree.TreePanel} this
34362         * @param {Roo.tree.TreeNode} node
34363         * @param {Number} index The index of the newly appended node
34364         */
34365        "appendnode" : true
34366         
34367     });
34368     if(this.singleExpand){
34369        this.on("beforeexpand", this.restrictExpand, this);
34370     }
34371     if (this.editor) {
34372         this.editor.tree = this;
34373         this.editor = Roo.factory(this.editor, Roo.tree);
34374     }
34375     
34376     if (this.selModel) {
34377         this.selModel = Roo.factory(this.selModel, Roo.tree);
34378     }
34379    
34380 };
34381 Roo.extend(Roo.tree.TreePanel, Roo.data.Tree, {
34382     rootVisible : true,
34383     animate: Roo.enableFx,
34384     lines : true,
34385     enableDD : false,
34386     hlDrop : Roo.enableFx,
34387   
34388     renderer: false,
34389     
34390     rendererTip: false,
34391     // private
34392     restrictExpand : function(node){
34393         var p = node.parentNode;
34394         if(p){
34395             if(p.expandedChild && p.expandedChild.parentNode == p){
34396                 p.expandedChild.collapse();
34397             }
34398             p.expandedChild = node;
34399         }
34400     },
34401
34402     // private override
34403     setRootNode : function(node){
34404         Roo.tree.TreePanel.superclass.setRootNode.call(this, node);
34405         if(!this.rootVisible){
34406             node.ui = new Roo.tree.RootTreeNodeUI(node);
34407         }
34408         return node;
34409     },
34410
34411     /**
34412      * Returns the container element for this TreePanel
34413      */
34414     getEl : function(){
34415         return this.el;
34416     },
34417
34418     /**
34419      * Returns the default TreeLoader for this TreePanel
34420      */
34421     getLoader : function(){
34422         return this.loader;
34423     },
34424
34425     /**
34426      * Expand all nodes
34427      */
34428     expandAll : function(){
34429         this.root.expand(true);
34430     },
34431
34432     /**
34433      * Collapse all nodes
34434      */
34435     collapseAll : function(){
34436         this.root.collapse(true);
34437     },
34438
34439     /**
34440      * Returns the selection model used by this TreePanel
34441      */
34442     getSelectionModel : function(){
34443         if(!this.selModel){
34444             this.selModel = new Roo.tree.DefaultSelectionModel();
34445         }
34446         return this.selModel;
34447     },
34448
34449     /**
34450      * Retrieve an array of checked nodes, or an array of a specific attribute of checked nodes (e.g. "id")
34451      * @param {String} attribute (optional) Defaults to null (return the actual nodes)
34452      * @param {TreeNode} startNode (optional) The node to start from, defaults to the root
34453      * @return {Array}
34454      */
34455     getChecked : function(a, startNode){
34456         startNode = startNode || this.root;
34457         var r = [];
34458         var f = function(){
34459             if(this.attributes.checked){
34460                 r.push(!a ? this : (a == 'id' ? this.id : this.attributes[a]));
34461             }
34462         }
34463         startNode.cascade(f);
34464         return r;
34465     },
34466
34467     /**
34468      * Expands a specified path in this TreePanel. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34469      * @param {String} path
34470      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34471      * @param {Function} callback (optional) The callback to call when the expand is complete. The callback will be called with
34472      * (bSuccess, oLastNode) where bSuccess is if the expand was successful and oLastNode is the last node that was expanded.
34473      */
34474     expandPath : function(path, attr, callback){
34475         attr = attr || "id";
34476         var keys = path.split(this.pathSeparator);
34477         var curNode = this.root;
34478         if(curNode.attributes[attr] != keys[1]){ // invalid root
34479             if(callback){
34480                 callback(false, null);
34481             }
34482             return;
34483         }
34484         var index = 1;
34485         var f = function(){
34486             if(++index == keys.length){
34487                 if(callback){
34488                     callback(true, curNode);
34489                 }
34490                 return;
34491             }
34492             var c = curNode.findChild(attr, keys[index]);
34493             if(!c){
34494                 if(callback){
34495                     callback(false, curNode);
34496                 }
34497                 return;
34498             }
34499             curNode = c;
34500             c.expand(false, false, f);
34501         };
34502         curNode.expand(false, false, f);
34503     },
34504
34505     /**
34506      * Selects the node in this tree at the specified path. A path can be retrieved from a node with {@link Roo.data.Node#getPath}
34507      * @param {String} path
34508      * @param {String} attr (optional) The attribute used in the path (see {@link Roo.data.Node#getPath} for more info)
34509      * @param {Function} callback (optional) The callback to call when the selection is complete. The callback will be called with
34510      * (bSuccess, oSelNode) where bSuccess is if the selection was successful and oSelNode is the selected node.
34511      */
34512     selectPath : function(path, attr, callback){
34513         attr = attr || "id";
34514         var keys = path.split(this.pathSeparator);
34515         var v = keys.pop();
34516         if(keys.length > 0){
34517             var f = function(success, node){
34518                 if(success && node){
34519                     var n = node.findChild(attr, v);
34520                     if(n){
34521                         n.select();
34522                         if(callback){
34523                             callback(true, n);
34524                         }
34525                     }else if(callback){
34526                         callback(false, n);
34527                     }
34528                 }else{
34529                     if(callback){
34530                         callback(false, n);
34531                     }
34532                 }
34533             };
34534             this.expandPath(keys.join(this.pathSeparator), attr, f);
34535         }else{
34536             this.root.select();
34537             if(callback){
34538                 callback(true, this.root);
34539             }
34540         }
34541     },
34542
34543     getTreeEl : function(){
34544         return this.el;
34545     },
34546
34547     /**
34548      * Trigger rendering of this TreePanel
34549      */
34550     render : function(){
34551         if (this.innerCt) {
34552             return this; // stop it rendering more than once!!
34553         }
34554         
34555         this.innerCt = this.el.createChild({tag:"ul",
34556                cls:"x-tree-root-ct " +
34557                (this.lines ? "x-tree-lines" : "x-tree-no-lines")});
34558
34559         if(this.containerScroll){
34560             Roo.dd.ScrollManager.register(this.el);
34561         }
34562         if((this.enableDD || this.enableDrop) && !this.dropZone){
34563            /**
34564             * The dropZone used by this tree if drop is enabled
34565             * @type Roo.tree.TreeDropZone
34566             */
34567              this.dropZone = new Roo.tree.TreeDropZone(this, this.dropConfig || {
34568                ddGroup: this.ddGroup || "TreeDD", appendOnly: this.ddAppendOnly === true
34569            });
34570         }
34571         if((this.enableDD || this.enableDrag) && !this.dragZone){
34572            /**
34573             * The dragZone used by this tree if drag is enabled
34574             * @type Roo.tree.TreeDragZone
34575             */
34576             this.dragZone = new Roo.tree.TreeDragZone(this, this.dragConfig || {
34577                ddGroup: this.ddGroup || "TreeDD",
34578                scroll: this.ddScroll
34579            });
34580         }
34581         this.getSelectionModel().init(this);
34582         if (!this.root) {
34583             Roo.log("ROOT not set in tree");
34584             return this;
34585         }
34586         this.root.render();
34587         if(!this.rootVisible){
34588             this.root.renderChildren();
34589         }
34590         return this;
34591     }
34592 });/*
34593  * Based on:
34594  * Ext JS Library 1.1.1
34595  * Copyright(c) 2006-2007, Ext JS, LLC.
34596  *
34597  * Originally Released Under LGPL - original licence link has changed is not relivant.
34598  *
34599  * Fork - LGPL
34600  * <script type="text/javascript">
34601  */
34602  
34603
34604 /**
34605  * @class Roo.tree.DefaultSelectionModel
34606  * @extends Roo.util.Observable
34607  * The default single selection for a TreePanel.
34608  * @param {Object} cfg Configuration
34609  */
34610 Roo.tree.DefaultSelectionModel = function(cfg){
34611    this.selNode = null;
34612    
34613    
34614    
34615    this.addEvents({
34616        /**
34617         * @event selectionchange
34618         * Fires when the selected node changes
34619         * @param {DefaultSelectionModel} this
34620         * @param {TreeNode} node the new selection
34621         */
34622        "selectionchange" : true,
34623
34624        /**
34625         * @event beforeselect
34626         * Fires before the selected node changes, return false to cancel the change
34627         * @param {DefaultSelectionModel} this
34628         * @param {TreeNode} node the new selection
34629         * @param {TreeNode} node the old selection
34630         */
34631        "beforeselect" : true
34632    });
34633    
34634     Roo.tree.DefaultSelectionModel.superclass.constructor.call(this,cfg);
34635 };
34636
34637 Roo.extend(Roo.tree.DefaultSelectionModel, Roo.util.Observable, {
34638     init : function(tree){
34639         this.tree = tree;
34640         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34641         tree.on("click", this.onNodeClick, this);
34642     },
34643     
34644     onNodeClick : function(node, e){
34645         if (e.ctrlKey && this.selNode == node)  {
34646             this.unselect(node);
34647             return;
34648         }
34649         this.select(node);
34650     },
34651     
34652     /**
34653      * Select a node.
34654      * @param {TreeNode} node The node to select
34655      * @return {TreeNode} The selected node
34656      */
34657     select : function(node){
34658         var last = this.selNode;
34659         if(last != node && this.fireEvent('beforeselect', this, node, last) !== false){
34660             if(last){
34661                 last.ui.onSelectedChange(false);
34662             }
34663             this.selNode = node;
34664             node.ui.onSelectedChange(true);
34665             this.fireEvent("selectionchange", this, node, last);
34666         }
34667         return node;
34668     },
34669     
34670     /**
34671      * Deselect a node.
34672      * @param {TreeNode} node The node to unselect
34673      */
34674     unselect : function(node){
34675         if(this.selNode == node){
34676             this.clearSelections();
34677         }    
34678     },
34679     
34680     /**
34681      * Clear all selections
34682      */
34683     clearSelections : function(){
34684         var n = this.selNode;
34685         if(n){
34686             n.ui.onSelectedChange(false);
34687             this.selNode = null;
34688             this.fireEvent("selectionchange", this, null);
34689         }
34690         return n;
34691     },
34692     
34693     /**
34694      * Get the selected node
34695      * @return {TreeNode} The selected node
34696      */
34697     getSelectedNode : function(){
34698         return this.selNode;    
34699     },
34700     
34701     /**
34702      * Returns true if the node is selected
34703      * @param {TreeNode} node The node to check
34704      * @return {Boolean}
34705      */
34706     isSelected : function(node){
34707         return this.selNode == node;  
34708     },
34709
34710     /**
34711      * Selects the node above the selected node in the tree, intelligently walking the nodes
34712      * @return TreeNode The new selection
34713      */
34714     selectPrevious : function(){
34715         var s = this.selNode || this.lastSelNode;
34716         if(!s){
34717             return null;
34718         }
34719         var ps = s.previousSibling;
34720         if(ps){
34721             if(!ps.isExpanded() || ps.childNodes.length < 1){
34722                 return this.select(ps);
34723             } else{
34724                 var lc = ps.lastChild;
34725                 while(lc && lc.isExpanded() && lc.childNodes.length > 0){
34726                     lc = lc.lastChild;
34727                 }
34728                 return this.select(lc);
34729             }
34730         } else if(s.parentNode && (this.tree.rootVisible || !s.parentNode.isRoot)){
34731             return this.select(s.parentNode);
34732         }
34733         return null;
34734     },
34735
34736     /**
34737      * Selects the node above the selected node in the tree, intelligently walking the nodes
34738      * @return TreeNode The new selection
34739      */
34740     selectNext : function(){
34741         var s = this.selNode || this.lastSelNode;
34742         if(!s){
34743             return null;
34744         }
34745         if(s.firstChild && s.isExpanded()){
34746              return this.select(s.firstChild);
34747          }else if(s.nextSibling){
34748              return this.select(s.nextSibling);
34749          }else if(s.parentNode){
34750             var newS = null;
34751             s.parentNode.bubble(function(){
34752                 if(this.nextSibling){
34753                     newS = this.getOwnerTree().selModel.select(this.nextSibling);
34754                     return false;
34755                 }
34756             });
34757             return newS;
34758          }
34759         return null;
34760     },
34761
34762     onKeyDown : function(e){
34763         var s = this.selNode || this.lastSelNode;
34764         // undesirable, but required
34765         var sm = this;
34766         if(!s){
34767             return;
34768         }
34769         var k = e.getKey();
34770         switch(k){
34771              case e.DOWN:
34772                  e.stopEvent();
34773                  this.selectNext();
34774              break;
34775              case e.UP:
34776                  e.stopEvent();
34777                  this.selectPrevious();
34778              break;
34779              case e.RIGHT:
34780                  e.preventDefault();
34781                  if(s.hasChildNodes()){
34782                      if(!s.isExpanded()){
34783                          s.expand();
34784                      }else if(s.firstChild){
34785                          this.select(s.firstChild, e);
34786                      }
34787                  }
34788              break;
34789              case e.LEFT:
34790                  e.preventDefault();
34791                  if(s.hasChildNodes() && s.isExpanded()){
34792                      s.collapse();
34793                  }else if(s.parentNode && (this.tree.rootVisible || s.parentNode != this.tree.getRootNode())){
34794                      this.select(s.parentNode, e);
34795                  }
34796              break;
34797         };
34798     }
34799 });
34800
34801 /**
34802  * @class Roo.tree.MultiSelectionModel
34803  * @extends Roo.util.Observable
34804  * Multi selection for a TreePanel.
34805  * @param {Object} cfg Configuration
34806  */
34807 Roo.tree.MultiSelectionModel = function(){
34808    this.selNodes = [];
34809    this.selMap = {};
34810    this.addEvents({
34811        /**
34812         * @event selectionchange
34813         * Fires when the selected nodes change
34814         * @param {MultiSelectionModel} this
34815         * @param {Array} nodes Array of the selected nodes
34816         */
34817        "selectionchange" : true
34818    });
34819    Roo.tree.MultiSelectionModel.superclass.constructor.call(this,cfg);
34820    
34821 };
34822
34823 Roo.extend(Roo.tree.MultiSelectionModel, Roo.util.Observable, {
34824     init : function(tree){
34825         this.tree = tree;
34826         tree.getTreeEl().on("keydown", this.onKeyDown, this);
34827         tree.on("click", this.onNodeClick, this);
34828     },
34829     
34830     onNodeClick : function(node, e){
34831         this.select(node, e, e.ctrlKey);
34832     },
34833     
34834     /**
34835      * Select a node.
34836      * @param {TreeNode} node The node to select
34837      * @param {EventObject} e (optional) An event associated with the selection
34838      * @param {Boolean} keepExisting True to retain existing selections
34839      * @return {TreeNode} The selected node
34840      */
34841     select : function(node, e, keepExisting){
34842         if(keepExisting !== true){
34843             this.clearSelections(true);
34844         }
34845         if(this.isSelected(node)){
34846             this.lastSelNode = node;
34847             return node;
34848         }
34849         this.selNodes.push(node);
34850         this.selMap[node.id] = node;
34851         this.lastSelNode = node;
34852         node.ui.onSelectedChange(true);
34853         this.fireEvent("selectionchange", this, this.selNodes);
34854         return node;
34855     },
34856     
34857     /**
34858      * Deselect a node.
34859      * @param {TreeNode} node The node to unselect
34860      */
34861     unselect : function(node){
34862         if(this.selMap[node.id]){
34863             node.ui.onSelectedChange(false);
34864             var sn = this.selNodes;
34865             var index = -1;
34866             if(sn.indexOf){
34867                 index = sn.indexOf(node);
34868             }else{
34869                 for(var i = 0, len = sn.length; i < len; i++){
34870                     if(sn[i] == node){
34871                         index = i;
34872                         break;
34873                     }
34874                 }
34875             }
34876             if(index != -1){
34877                 this.selNodes.splice(index, 1);
34878             }
34879             delete this.selMap[node.id];
34880             this.fireEvent("selectionchange", this, this.selNodes);
34881         }
34882     },
34883     
34884     /**
34885      * Clear all selections
34886      */
34887     clearSelections : function(suppressEvent){
34888         var sn = this.selNodes;
34889         if(sn.length > 0){
34890             for(var i = 0, len = sn.length; i < len; i++){
34891                 sn[i].ui.onSelectedChange(false);
34892             }
34893             this.selNodes = [];
34894             this.selMap = {};
34895             if(suppressEvent !== true){
34896                 this.fireEvent("selectionchange", this, this.selNodes);
34897             }
34898         }
34899     },
34900     
34901     /**
34902      * Returns true if the node is selected
34903      * @param {TreeNode} node The node to check
34904      * @return {Boolean}
34905      */
34906     isSelected : function(node){
34907         return this.selMap[node.id] ? true : false;  
34908     },
34909     
34910     /**
34911      * Returns an array of the selected nodes
34912      * @return {Array}
34913      */
34914     getSelectedNodes : function(){
34915         return this.selNodes;    
34916     },
34917
34918     onKeyDown : Roo.tree.DefaultSelectionModel.prototype.onKeyDown,
34919
34920     selectNext : Roo.tree.DefaultSelectionModel.prototype.selectNext,
34921
34922     selectPrevious : Roo.tree.DefaultSelectionModel.prototype.selectPrevious
34923 });/*
34924  * Based on:
34925  * Ext JS Library 1.1.1
34926  * Copyright(c) 2006-2007, Ext JS, LLC.
34927  *
34928  * Originally Released Under LGPL - original licence link has changed is not relivant.
34929  *
34930  * Fork - LGPL
34931  * <script type="text/javascript">
34932  */
34933  
34934 /**
34935  * @class Roo.tree.TreeNode
34936  * @extends Roo.data.Node
34937  * @cfg {String} text The text for this node
34938  * @cfg {Boolean} expanded true to start the node expanded
34939  * @cfg {Boolean} allowDrag false to make this node undraggable if DD is on (defaults to true)
34940  * @cfg {Boolean} allowDrop false if this node cannot be drop on
34941  * @cfg {Boolean} disabled true to start the node disabled
34942  * @cfg {String} icon The path to an icon for the node. The preferred way to do this
34943  *    is to use the cls or iconCls attributes and add the icon via a CSS background image.
34944  * @cfg {String} cls A css class to be added to the node
34945  * @cfg {String} iconCls A css class to be added to the nodes icon element for applying css background images
34946  * @cfg {String} href URL of the link used for the node (defaults to #)
34947  * @cfg {String} hrefTarget target frame for the link
34948  * @cfg {String} qtip An Ext QuickTip for the node
34949  * @cfg {String} qtipCfg An Ext QuickTip config for the node (used instead of qtip)
34950  * @cfg {Boolean} singleClickExpand True for single click expand on this node
34951  * @cfg {Function} uiProvider A UI <b>class</b> to use for this node (defaults to Roo.tree.TreeNodeUI)
34952  * @cfg {Boolean} checked True to render a checked checkbox for this node, false to render an unchecked checkbox
34953  * (defaults to undefined with no checkbox rendered)
34954  * @constructor
34955  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node
34956  */
34957 Roo.tree.TreeNode = function(attributes){
34958     attributes = attributes || {};
34959     if(typeof attributes == "string"){
34960         attributes = {text: attributes};
34961     }
34962     this.childrenRendered = false;
34963     this.rendered = false;
34964     Roo.tree.TreeNode.superclass.constructor.call(this, attributes);
34965     this.expanded = attributes.expanded === true;
34966     this.isTarget = attributes.isTarget !== false;
34967     this.draggable = attributes.draggable !== false && attributes.allowDrag !== false;
34968     this.allowChildren = attributes.allowChildren !== false && attributes.allowDrop !== false;
34969
34970     /**
34971      * Read-only. The text for this node. To change it use setText().
34972      * @type String
34973      */
34974     this.text = attributes.text;
34975     /**
34976      * True if this node is disabled.
34977      * @type Boolean
34978      */
34979     this.disabled = attributes.disabled === true;
34980
34981     this.addEvents({
34982         /**
34983         * @event textchange
34984         * Fires when the text for this node is changed
34985         * @param {Node} this This node
34986         * @param {String} text The new text
34987         * @param {String} oldText The old text
34988         */
34989         "textchange" : true,
34990         /**
34991         * @event beforeexpand
34992         * Fires before this node is expanded, return false to cancel.
34993         * @param {Node} this This node
34994         * @param {Boolean} deep
34995         * @param {Boolean} anim
34996         */
34997         "beforeexpand" : true,
34998         /**
34999         * @event beforecollapse
35000         * Fires before this node is collapsed, return false to cancel.
35001         * @param {Node} this This node
35002         * @param {Boolean} deep
35003         * @param {Boolean} anim
35004         */
35005         "beforecollapse" : true,
35006         /**
35007         * @event expand
35008         * Fires when this node is expanded
35009         * @param {Node} this This node
35010         */
35011         "expand" : true,
35012         /**
35013         * @event disabledchange
35014         * Fires when the disabled status of this node changes
35015         * @param {Node} this This node
35016         * @param {Boolean} disabled
35017         */
35018         "disabledchange" : true,
35019         /**
35020         * @event collapse
35021         * Fires when this node is collapsed
35022         * @param {Node} this This node
35023         */
35024         "collapse" : true,
35025         /**
35026         * @event beforeclick
35027         * Fires before click processing. Return false to cancel the default action.
35028         * @param {Node} this This node
35029         * @param {Roo.EventObject} e The event object
35030         */
35031         "beforeclick":true,
35032         /**
35033         * @event checkchange
35034         * Fires when a node with a checkbox's checked property changes
35035         * @param {Node} this This node
35036         * @param {Boolean} checked
35037         */
35038         "checkchange":true,
35039         /**
35040         * @event click
35041         * Fires when this node is clicked
35042         * @param {Node} this This node
35043         * @param {Roo.EventObject} e The event object
35044         */
35045         "click":true,
35046         /**
35047         * @event dblclick
35048         * Fires when this node is double clicked
35049         * @param {Node} this This node
35050         * @param {Roo.EventObject} e The event object
35051         */
35052         "dblclick":true,
35053         /**
35054         * @event contextmenu
35055         * Fires when this node is right clicked
35056         * @param {Node} this This node
35057         * @param {Roo.EventObject} e The event object
35058         */
35059         "contextmenu":true,
35060         /**
35061         * @event beforechildrenrendered
35062         * Fires right before the child nodes for this node are rendered
35063         * @param {Node} this This node
35064         */
35065         "beforechildrenrendered":true
35066     });
35067
35068     var uiClass = this.attributes.uiProvider || Roo.tree.TreeNodeUI;
35069
35070     /**
35071      * Read-only. The UI for this node
35072      * @type TreeNodeUI
35073      */
35074     this.ui = new uiClass(this);
35075     
35076     // finally support items[]
35077     if (typeof(this.attributes.items) == 'undefined' || !this.attributes.items) {
35078         return;
35079     }
35080     
35081     
35082     Roo.each(this.attributes.items, function(c) {
35083         this.appendChild(Roo.factory(c,Roo.Tree));
35084     }, this);
35085     delete this.attributes.items;
35086     
35087     
35088     
35089 };
35090 Roo.extend(Roo.tree.TreeNode, Roo.data.Node, {
35091     preventHScroll: true,
35092     /**
35093      * Returns true if this node is expanded
35094      * @return {Boolean}
35095      */
35096     isExpanded : function(){
35097         return this.expanded;
35098     },
35099
35100     /**
35101      * Returns the UI object for this node
35102      * @return {TreeNodeUI}
35103      */
35104     getUI : function(){
35105         return this.ui;
35106     },
35107
35108     // private override
35109     setFirstChild : function(node){
35110         var of = this.firstChild;
35111         Roo.tree.TreeNode.superclass.setFirstChild.call(this, node);
35112         if(this.childrenRendered && of && node != of){
35113             of.renderIndent(true, true);
35114         }
35115         if(this.rendered){
35116             this.renderIndent(true, true);
35117         }
35118     },
35119
35120     // private override
35121     setLastChild : function(node){
35122         var ol = this.lastChild;
35123         Roo.tree.TreeNode.superclass.setLastChild.call(this, node);
35124         if(this.childrenRendered && ol && node != ol){
35125             ol.renderIndent(true, true);
35126         }
35127         if(this.rendered){
35128             this.renderIndent(true, true);
35129         }
35130     },
35131
35132     // these methods are overridden to provide lazy rendering support
35133     // private override
35134     appendChild : function()
35135     {
35136         var node = Roo.tree.TreeNode.superclass.appendChild.apply(this, arguments);
35137         if(node && this.childrenRendered){
35138             node.render();
35139         }
35140         this.ui.updateExpandIcon();
35141         return node;
35142     },
35143
35144     // private override
35145     removeChild : function(node){
35146         this.ownerTree.getSelectionModel().unselect(node);
35147         Roo.tree.TreeNode.superclass.removeChild.apply(this, arguments);
35148         // if it's been rendered remove dom node
35149         if(this.childrenRendered){
35150             node.ui.remove();
35151         }
35152         if(this.childNodes.length < 1){
35153             this.collapse(false, false);
35154         }else{
35155             this.ui.updateExpandIcon();
35156         }
35157         if(!this.firstChild) {
35158             this.childrenRendered = false;
35159         }
35160         return node;
35161     },
35162
35163     // private override
35164     insertBefore : function(node, refNode){
35165         var newNode = Roo.tree.TreeNode.superclass.insertBefore.apply(this, arguments);
35166         if(newNode && refNode && this.childrenRendered){
35167             node.render();
35168         }
35169         this.ui.updateExpandIcon();
35170         return newNode;
35171     },
35172
35173     /**
35174      * Sets the text for this node
35175      * @param {String} text
35176      */
35177     setText : function(text){
35178         var oldText = this.text;
35179         this.text = text;
35180         this.attributes.text = text;
35181         if(this.rendered){ // event without subscribing
35182             this.ui.onTextChange(this, text, oldText);
35183         }
35184         this.fireEvent("textchange", this, text, oldText);
35185     },
35186
35187     /**
35188      * Triggers selection of this node
35189      */
35190     select : function(){
35191         this.getOwnerTree().getSelectionModel().select(this);
35192     },
35193
35194     /**
35195      * Triggers deselection of this node
35196      */
35197     unselect : function(){
35198         this.getOwnerTree().getSelectionModel().unselect(this);
35199     },
35200
35201     /**
35202      * Returns true if this node is selected
35203      * @return {Boolean}
35204      */
35205     isSelected : function(){
35206         return this.getOwnerTree().getSelectionModel().isSelected(this);
35207     },
35208
35209     /**
35210      * Expand this node.
35211      * @param {Boolean} deep (optional) True to expand all children as well
35212      * @param {Boolean} anim (optional) false to cancel the default animation
35213      * @param {Function} callback (optional) A callback to be called when
35214      * expanding this node completes (does not wait for deep expand to complete).
35215      * Called with 1 parameter, this node.
35216      */
35217     expand : function(deep, anim, callback){
35218         if(!this.expanded){
35219             if(this.fireEvent("beforeexpand", this, deep, anim) === false){
35220                 return;
35221             }
35222             if(!this.childrenRendered){
35223                 this.renderChildren();
35224             }
35225             this.expanded = true;
35226             
35227             if(!this.isHiddenRoot() && (this.getOwnerTree() && this.getOwnerTree().animate && anim !== false) || anim){
35228                 this.ui.animExpand(function(){
35229                     this.fireEvent("expand", this);
35230                     if(typeof callback == "function"){
35231                         callback(this);
35232                     }
35233                     if(deep === true){
35234                         this.expandChildNodes(true);
35235                     }
35236                 }.createDelegate(this));
35237                 return;
35238             }else{
35239                 this.ui.expand();
35240                 this.fireEvent("expand", this);
35241                 if(typeof callback == "function"){
35242                     callback(this);
35243                 }
35244             }
35245         }else{
35246            if(typeof callback == "function"){
35247                callback(this);
35248            }
35249         }
35250         if(deep === true){
35251             this.expandChildNodes(true);
35252         }
35253     },
35254
35255     isHiddenRoot : function(){
35256         return this.isRoot && !this.getOwnerTree().rootVisible;
35257     },
35258
35259     /**
35260      * Collapse this node.
35261      * @param {Boolean} deep (optional) True to collapse all children as well
35262      * @param {Boolean} anim (optional) false to cancel the default animation
35263      */
35264     collapse : function(deep, anim){
35265         if(this.expanded && !this.isHiddenRoot()){
35266             if(this.fireEvent("beforecollapse", this, deep, anim) === false){
35267                 return;
35268             }
35269             this.expanded = false;
35270             if((this.getOwnerTree().animate && anim !== false) || anim){
35271                 this.ui.animCollapse(function(){
35272                     this.fireEvent("collapse", this);
35273                     if(deep === true){
35274                         this.collapseChildNodes(true);
35275                     }
35276                 }.createDelegate(this));
35277                 return;
35278             }else{
35279                 this.ui.collapse();
35280                 this.fireEvent("collapse", this);
35281             }
35282         }
35283         if(deep === true){
35284             var cs = this.childNodes;
35285             for(var i = 0, len = cs.length; i < len; i++) {
35286                 cs[i].collapse(true, false);
35287             }
35288         }
35289     },
35290
35291     // private
35292     delayedExpand : function(delay){
35293         if(!this.expandProcId){
35294             this.expandProcId = this.expand.defer(delay, this);
35295         }
35296     },
35297
35298     // private
35299     cancelExpand : function(){
35300         if(this.expandProcId){
35301             clearTimeout(this.expandProcId);
35302         }
35303         this.expandProcId = false;
35304     },
35305
35306     /**
35307      * Toggles expanded/collapsed state of the node
35308      */
35309     toggle : function(){
35310         if(this.expanded){
35311             this.collapse();
35312         }else{
35313             this.expand();
35314         }
35315     },
35316
35317     /**
35318      * Ensures all parent nodes are expanded
35319      */
35320     ensureVisible : function(callback){
35321         var tree = this.getOwnerTree();
35322         tree.expandPath(this.parentNode.getPath(), false, function(){
35323             tree.getTreeEl().scrollChildIntoView(this.ui.anchor);
35324             Roo.callback(callback);
35325         }.createDelegate(this));
35326     },
35327
35328     /**
35329      * Expand all child nodes
35330      * @param {Boolean} deep (optional) true if the child nodes should also expand their child nodes
35331      */
35332     expandChildNodes : function(deep){
35333         var cs = this.childNodes;
35334         for(var i = 0, len = cs.length; i < len; i++) {
35335                 cs[i].expand(deep);
35336         }
35337     },
35338
35339     /**
35340      * Collapse all child nodes
35341      * @param {Boolean} deep (optional) true if the child nodes should also collapse their child nodes
35342      */
35343     collapseChildNodes : function(deep){
35344         var cs = this.childNodes;
35345         for(var i = 0, len = cs.length; i < len; i++) {
35346                 cs[i].collapse(deep);
35347         }
35348     },
35349
35350     /**
35351      * Disables this node
35352      */
35353     disable : function(){
35354         this.disabled = true;
35355         this.unselect();
35356         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35357             this.ui.onDisableChange(this, true);
35358         }
35359         this.fireEvent("disabledchange", this, true);
35360     },
35361
35362     /**
35363      * Enables this node
35364      */
35365     enable : function(){
35366         this.disabled = false;
35367         if(this.rendered && this.ui.onDisableChange){ // event without subscribing
35368             this.ui.onDisableChange(this, false);
35369         }
35370         this.fireEvent("disabledchange", this, false);
35371     },
35372
35373     // private
35374     renderChildren : function(suppressEvent){
35375         if(suppressEvent !== false){
35376             this.fireEvent("beforechildrenrendered", this);
35377         }
35378         var cs = this.childNodes;
35379         for(var i = 0, len = cs.length; i < len; i++){
35380             cs[i].render(true);
35381         }
35382         this.childrenRendered = true;
35383     },
35384
35385     // private
35386     sort : function(fn, scope){
35387         Roo.tree.TreeNode.superclass.sort.apply(this, arguments);
35388         if(this.childrenRendered){
35389             var cs = this.childNodes;
35390             for(var i = 0, len = cs.length; i < len; i++){
35391                 cs[i].render(true);
35392             }
35393         }
35394     },
35395
35396     // private
35397     render : function(bulkRender){
35398         this.ui.render(bulkRender);
35399         if(!this.rendered){
35400             this.rendered = true;
35401             if(this.expanded){
35402                 this.expanded = false;
35403                 this.expand(false, false);
35404             }
35405         }
35406     },
35407
35408     // private
35409     renderIndent : function(deep, refresh){
35410         if(refresh){
35411             this.ui.childIndent = null;
35412         }
35413         this.ui.renderIndent();
35414         if(deep === true && this.childrenRendered){
35415             var cs = this.childNodes;
35416             for(var i = 0, len = cs.length; i < len; i++){
35417                 cs[i].renderIndent(true, refresh);
35418             }
35419         }
35420     }
35421 });/*
35422  * Based on:
35423  * Ext JS Library 1.1.1
35424  * Copyright(c) 2006-2007, Ext JS, LLC.
35425  *
35426  * Originally Released Under LGPL - original licence link has changed is not relivant.
35427  *
35428  * Fork - LGPL
35429  * <script type="text/javascript">
35430  */
35431  
35432 /**
35433  * @class Roo.tree.AsyncTreeNode
35434  * @extends Roo.tree.TreeNode
35435  * @cfg {TreeLoader} loader A TreeLoader to be used by this node (defaults to the loader defined on the tree)
35436  * @constructor
35437  * @param {Object/String} attributes The attributes/config for the node or just a string with the text for the node 
35438  */
35439  Roo.tree.AsyncTreeNode = function(config){
35440     this.loaded = false;
35441     this.loading = false;
35442     Roo.tree.AsyncTreeNode.superclass.constructor.apply(this, arguments);
35443     /**
35444     * @event beforeload
35445     * Fires before this node is loaded, return false to cancel
35446     * @param {Node} this This node
35447     */
35448     this.addEvents({'beforeload':true, 'load': true});
35449     /**
35450     * @event load
35451     * Fires when this node is loaded
35452     * @param {Node} this This node
35453     */
35454     /**
35455      * The loader used by this node (defaults to using the tree's defined loader)
35456      * @type TreeLoader
35457      * @property loader
35458      */
35459 };
35460 Roo.extend(Roo.tree.AsyncTreeNode, Roo.tree.TreeNode, {
35461     expand : function(deep, anim, callback){
35462         if(this.loading){ // if an async load is already running, waiting til it's done
35463             var timer;
35464             var f = function(){
35465                 if(!this.loading){ // done loading
35466                     clearInterval(timer);
35467                     this.expand(deep, anim, callback);
35468                 }
35469             }.createDelegate(this);
35470             timer = setInterval(f, 200);
35471             return;
35472         }
35473         if(!this.loaded){
35474             if(this.fireEvent("beforeload", this) === false){
35475                 return;
35476             }
35477             this.loading = true;
35478             this.ui.beforeLoad(this);
35479             var loader = this.loader || this.attributes.loader || this.getOwnerTree().getLoader();
35480             if(loader){
35481                 loader.load(this, this.loadComplete.createDelegate(this, [deep, anim, callback]));
35482                 return;
35483             }
35484         }
35485         Roo.tree.AsyncTreeNode.superclass.expand.call(this, deep, anim, callback);
35486     },
35487     
35488     /**
35489      * Returns true if this node is currently loading
35490      * @return {Boolean}
35491      */
35492     isLoading : function(){
35493         return this.loading;  
35494     },
35495     
35496     loadComplete : function(deep, anim, callback){
35497         this.loading = false;
35498         this.loaded = true;
35499         this.ui.afterLoad(this);
35500         this.fireEvent("load", this);
35501         this.expand(deep, anim, callback);
35502     },
35503     
35504     /**
35505      * Returns true if this node has been loaded
35506      * @return {Boolean}
35507      */
35508     isLoaded : function(){
35509         return this.loaded;
35510     },
35511     
35512     hasChildNodes : function(){
35513         if(!this.isLeaf() && !this.loaded){
35514             return true;
35515         }else{
35516             return Roo.tree.AsyncTreeNode.superclass.hasChildNodes.call(this);
35517         }
35518     },
35519
35520     /**
35521      * Trigger a reload for this node
35522      * @param {Function} callback
35523      */
35524     reload : function(callback){
35525         this.collapse(false, false);
35526         while(this.firstChild){
35527             this.removeChild(this.firstChild);
35528         }
35529         this.childrenRendered = false;
35530         this.loaded = false;
35531         if(this.isHiddenRoot()){
35532             this.expanded = false;
35533         }
35534         this.expand(false, false, callback);
35535     }
35536 });/*
35537  * Based on:
35538  * Ext JS Library 1.1.1
35539  * Copyright(c) 2006-2007, Ext JS, LLC.
35540  *
35541  * Originally Released Under LGPL - original licence link has changed is not relivant.
35542  *
35543  * Fork - LGPL
35544  * <script type="text/javascript">
35545  */
35546  
35547 /**
35548  * @class Roo.tree.TreeNodeUI
35549  * @constructor
35550  * @param {Object} node The node to render
35551  * The TreeNode UI implementation is separate from the
35552  * tree implementation. Unless you are customizing the tree UI,
35553  * you should never have to use this directly.
35554  */
35555 Roo.tree.TreeNodeUI = function(node){
35556     this.node = node;
35557     this.rendered = false;
35558     this.animating = false;
35559     this.emptyIcon = Roo.BLANK_IMAGE_URL;
35560 };
35561
35562 Roo.tree.TreeNodeUI.prototype = {
35563     removeChild : function(node){
35564         if(this.rendered){
35565             this.ctNode.removeChild(node.ui.getEl());
35566         }
35567     },
35568
35569     beforeLoad : function(){
35570          this.addClass("x-tree-node-loading");
35571     },
35572
35573     afterLoad : function(){
35574          this.removeClass("x-tree-node-loading");
35575     },
35576
35577     onTextChange : function(node, text, oldText){
35578         if(this.rendered){
35579             this.textNode.innerHTML = text;
35580         }
35581     },
35582
35583     onDisableChange : function(node, state){
35584         this.disabled = state;
35585         if(state){
35586             this.addClass("x-tree-node-disabled");
35587         }else{
35588             this.removeClass("x-tree-node-disabled");
35589         }
35590     },
35591
35592     onSelectedChange : function(state){
35593         if(state){
35594             this.focus();
35595             this.addClass("x-tree-selected");
35596         }else{
35597             //this.blur();
35598             this.removeClass("x-tree-selected");
35599         }
35600     },
35601
35602     onMove : function(tree, node, oldParent, newParent, index, refNode){
35603         this.childIndent = null;
35604         if(this.rendered){
35605             var targetNode = newParent.ui.getContainer();
35606             if(!targetNode){//target not rendered
35607                 this.holder = document.createElement("div");
35608                 this.holder.appendChild(this.wrap);
35609                 return;
35610             }
35611             var insertBefore = refNode ? refNode.ui.getEl() : null;
35612             if(insertBefore){
35613                 targetNode.insertBefore(this.wrap, insertBefore);
35614             }else{
35615                 targetNode.appendChild(this.wrap);
35616             }
35617             this.node.renderIndent(true);
35618         }
35619     },
35620
35621     addClass : function(cls){
35622         if(this.elNode){
35623             Roo.fly(this.elNode).addClass(cls);
35624         }
35625     },
35626
35627     removeClass : function(cls){
35628         if(this.elNode){
35629             Roo.fly(this.elNode).removeClass(cls);
35630         }
35631     },
35632
35633     remove : function(){
35634         if(this.rendered){
35635             this.holder = document.createElement("div");
35636             this.holder.appendChild(this.wrap);
35637         }
35638     },
35639
35640     fireEvent : function(){
35641         return this.node.fireEvent.apply(this.node, arguments);
35642     },
35643
35644     initEvents : function(){
35645         this.node.on("move", this.onMove, this);
35646         var E = Roo.EventManager;
35647         var a = this.anchor;
35648
35649         var el = Roo.fly(a, '_treeui');
35650
35651         if(Roo.isOpera){ // opera render bug ignores the CSS
35652             el.setStyle("text-decoration", "none");
35653         }
35654
35655         el.on("click", this.onClick, this);
35656         el.on("dblclick", this.onDblClick, this);
35657
35658         if(this.checkbox){
35659             Roo.EventManager.on(this.checkbox,
35660                     Roo.isIE ? 'click' : 'change', this.onCheckChange, this);
35661         }
35662
35663         el.on("contextmenu", this.onContextMenu, this);
35664
35665         var icon = Roo.fly(this.iconNode);
35666         icon.on("click", this.onClick, this);
35667         icon.on("dblclick", this.onDblClick, this);
35668         icon.on("contextmenu", this.onContextMenu, this);
35669         E.on(this.ecNode, "click", this.ecClick, this, true);
35670
35671         if(this.node.disabled){
35672             this.addClass("x-tree-node-disabled");
35673         }
35674         if(this.node.hidden){
35675             this.addClass("x-tree-node-disabled");
35676         }
35677         var ot = this.node.getOwnerTree();
35678         var dd = ot ? (ot.enableDD || ot.enableDrag || ot.enableDrop) : false;
35679         if(dd && (!this.node.isRoot || ot.rootVisible)){
35680             Roo.dd.Registry.register(this.elNode, {
35681                 node: this.node,
35682                 handles: this.getDDHandles(),
35683                 isHandle: false
35684             });
35685         }
35686     },
35687
35688     getDDHandles : function(){
35689         return [this.iconNode, this.textNode];
35690     },
35691
35692     hide : function(){
35693         if(this.rendered){
35694             this.wrap.style.display = "none";
35695         }
35696     },
35697
35698     show : function(){
35699         if(this.rendered){
35700             this.wrap.style.display = "";
35701         }
35702     },
35703
35704     onContextMenu : function(e){
35705         if (this.node.hasListener("contextmenu") || this.node.getOwnerTree().hasListener("contextmenu")) {
35706             e.preventDefault();
35707             this.focus();
35708             this.fireEvent("contextmenu", this.node, e);
35709         }
35710     },
35711
35712     onClick : function(e){
35713         if(this.dropping){
35714             e.stopEvent();
35715             return;
35716         }
35717         if(this.fireEvent("beforeclick", this.node, e) !== false){
35718             if(!this.disabled && this.node.attributes.href){
35719                 this.fireEvent("click", this.node, e);
35720                 return;
35721             }
35722             e.preventDefault();
35723             if(this.disabled){
35724                 return;
35725             }
35726
35727             if(this.node.attributes.singleClickExpand && !this.animating && this.node.hasChildNodes()){
35728                 this.node.toggle();
35729             }
35730
35731             this.fireEvent("click", this.node, e);
35732         }else{
35733             e.stopEvent();
35734         }
35735     },
35736
35737     onDblClick : function(e){
35738         e.preventDefault();
35739         if(this.disabled){
35740             return;
35741         }
35742         if(this.checkbox){
35743             this.toggleCheck();
35744         }
35745         if(!this.animating && this.node.hasChildNodes()){
35746             this.node.toggle();
35747         }
35748         this.fireEvent("dblclick", this.node, e);
35749     },
35750
35751     onCheckChange : function(){
35752         var checked = this.checkbox.checked;
35753         this.node.attributes.checked = checked;
35754         this.fireEvent('checkchange', this.node, checked);
35755     },
35756
35757     ecClick : function(e){
35758         if(!this.animating && this.node.hasChildNodes()){
35759             this.node.toggle();
35760         }
35761     },
35762
35763     startDrop : function(){
35764         this.dropping = true;
35765     },
35766
35767     // delayed drop so the click event doesn't get fired on a drop
35768     endDrop : function(){
35769        setTimeout(function(){
35770            this.dropping = false;
35771        }.createDelegate(this), 50);
35772     },
35773
35774     expand : function(){
35775         this.updateExpandIcon();
35776         this.ctNode.style.display = "";
35777     },
35778
35779     focus : function(){
35780         if(!this.node.preventHScroll){
35781             try{this.anchor.focus();
35782             }catch(e){}
35783         }else if(!Roo.isIE){
35784             try{
35785                 var noscroll = this.node.getOwnerTree().getTreeEl().dom;
35786                 var l = noscroll.scrollLeft;
35787                 this.anchor.focus();
35788                 noscroll.scrollLeft = l;
35789             }catch(e){}
35790         }
35791     },
35792
35793     toggleCheck : function(value){
35794         var cb = this.checkbox;
35795         if(cb){
35796             cb.checked = (value === undefined ? !cb.checked : value);
35797         }
35798     },
35799
35800     blur : function(){
35801         try{
35802             this.anchor.blur();
35803         }catch(e){}
35804     },
35805
35806     animExpand : function(callback){
35807         var ct = Roo.get(this.ctNode);
35808         ct.stopFx();
35809         if(!this.node.hasChildNodes()){
35810             this.updateExpandIcon();
35811             this.ctNode.style.display = "";
35812             Roo.callback(callback);
35813             return;
35814         }
35815         this.animating = true;
35816         this.updateExpandIcon();
35817
35818         ct.slideIn('t', {
35819            callback : function(){
35820                this.animating = false;
35821                Roo.callback(callback);
35822             },
35823             scope: this,
35824             duration: this.node.ownerTree.duration || .25
35825         });
35826     },
35827
35828     highlight : function(){
35829         var tree = this.node.getOwnerTree();
35830         Roo.fly(this.wrap).highlight(
35831             tree.hlColor || "C3DAF9",
35832             {endColor: tree.hlBaseColor}
35833         );
35834     },
35835
35836     collapse : function(){
35837         this.updateExpandIcon();
35838         this.ctNode.style.display = "none";
35839     },
35840
35841     animCollapse : function(callback){
35842         var ct = Roo.get(this.ctNode);
35843         ct.enableDisplayMode('block');
35844         ct.stopFx();
35845
35846         this.animating = true;
35847         this.updateExpandIcon();
35848
35849         ct.slideOut('t', {
35850             callback : function(){
35851                this.animating = false;
35852                Roo.callback(callback);
35853             },
35854             scope: this,
35855             duration: this.node.ownerTree.duration || .25
35856         });
35857     },
35858
35859     getContainer : function(){
35860         return this.ctNode;
35861     },
35862
35863     getEl : function(){
35864         return this.wrap;
35865     },
35866
35867     appendDDGhost : function(ghostNode){
35868         ghostNode.appendChild(this.elNode.cloneNode(true));
35869     },
35870
35871     getDDRepairXY : function(){
35872         return Roo.lib.Dom.getXY(this.iconNode);
35873     },
35874
35875     onRender : function(){
35876         this.render();
35877     },
35878
35879     render : function(bulkRender){
35880         var n = this.node, a = n.attributes;
35881         var targetNode = n.parentNode ?
35882               n.parentNode.ui.getContainer() : n.ownerTree.innerCt.dom;
35883
35884         if(!this.rendered){
35885             this.rendered = true;
35886
35887             this.renderElements(n, a, targetNode, bulkRender);
35888
35889             if(a.qtip){
35890                if(this.textNode.setAttributeNS){
35891                    this.textNode.setAttributeNS("ext", "qtip", a.qtip);
35892                    if(a.qtipTitle){
35893                        this.textNode.setAttributeNS("ext", "qtitle", a.qtipTitle);
35894                    }
35895                }else{
35896                    this.textNode.setAttribute("ext:qtip", a.qtip);
35897                    if(a.qtipTitle){
35898                        this.textNode.setAttribute("ext:qtitle", a.qtipTitle);
35899                    }
35900                }
35901             }else if(a.qtipCfg){
35902                 a.qtipCfg.target = Roo.id(this.textNode);
35903                 Roo.QuickTips.register(a.qtipCfg);
35904             }
35905             this.initEvents();
35906             if(!this.node.expanded){
35907                 this.updateExpandIcon();
35908             }
35909         }else{
35910             if(bulkRender === true) {
35911                 targetNode.appendChild(this.wrap);
35912             }
35913         }
35914     },
35915
35916     renderElements : function(n, a, targetNode, bulkRender)
35917     {
35918         // add some indent caching, this helps performance when rendering a large tree
35919         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
35920         var t = n.getOwnerTree();
35921         var txt = t && t.renderer ? t.renderer(n.attributes) : Roo.util.Format.htmlEncode(n.text);
35922         if (typeof(n.attributes.html) != 'undefined') {
35923             txt = n.attributes.html;
35924         }
35925         var tip = t && t.rendererTip ? t.rendererTip(n.attributes) : txt;
35926         var cb = typeof a.checked == 'boolean';
35927         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
35928         var buf = ['<li class="x-tree-node"><div class="x-tree-node-el ', a.cls,'">',
35929             '<span class="x-tree-node-indent">',this.indentMarkup,"</span>",
35930             '<img src="', this.emptyIcon, '" class="x-tree-ec-icon" />',
35931             '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',(a.icon ? " x-tree-node-inline-icon" : ""),(a.iconCls ? " "+a.iconCls : ""),'" unselectable="on" />',
35932             cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + (a.checked ? 'checked="checked" />' : ' />')) : '',
35933             '<a hidefocus="on" href="',href,'" tabIndex="1" ',
35934              a.hrefTarget ? ' target="'+a.hrefTarget+'"' : "", 
35935                 '><span unselectable="on" qtip="' , tip ,'">',txt,"</span></a></div>",
35936             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
35937             "</li>"];
35938
35939         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
35940             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
35941                                 n.nextSibling.ui.getEl(), buf.join(""));
35942         }else{
35943             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
35944         }
35945
35946         this.elNode = this.wrap.childNodes[0];
35947         this.ctNode = this.wrap.childNodes[1];
35948         var cs = this.elNode.childNodes;
35949         this.indentNode = cs[0];
35950         this.ecNode = cs[1];
35951         this.iconNode = cs[2];
35952         var index = 3;
35953         if(cb){
35954             this.checkbox = cs[3];
35955             index++;
35956         }
35957         this.anchor = cs[index];
35958         this.textNode = cs[index].firstChild;
35959     },
35960
35961     getAnchor : function(){
35962         return this.anchor;
35963     },
35964
35965     getTextEl : function(){
35966         return this.textNode;
35967     },
35968
35969     getIconEl : function(){
35970         return this.iconNode;
35971     },
35972
35973     isChecked : function(){
35974         return this.checkbox ? this.checkbox.checked : false;
35975     },
35976
35977     updateExpandIcon : function(){
35978         if(this.rendered){
35979             var n = this.node, c1, c2;
35980             var cls = n.isLast() ? "x-tree-elbow-end" : "x-tree-elbow";
35981             var hasChild = n.hasChildNodes();
35982             if(hasChild){
35983                 if(n.expanded){
35984                     cls += "-minus";
35985                     c1 = "x-tree-node-collapsed";
35986                     c2 = "x-tree-node-expanded";
35987                 }else{
35988                     cls += "-plus";
35989                     c1 = "x-tree-node-expanded";
35990                     c2 = "x-tree-node-collapsed";
35991                 }
35992                 if(this.wasLeaf){
35993                     this.removeClass("x-tree-node-leaf");
35994                     this.wasLeaf = false;
35995                 }
35996                 if(this.c1 != c1 || this.c2 != c2){
35997                     Roo.fly(this.elNode).replaceClass(c1, c2);
35998                     this.c1 = c1; this.c2 = c2;
35999                 }
36000             }else{
36001                 // this changes non-leafs into leafs if they have no children.
36002                 // it's not very rational behaviour..
36003                 
36004                 if(!this.wasLeaf && this.node.leaf){
36005                     Roo.fly(this.elNode).replaceClass("x-tree-node-expanded", "x-tree-node-leaf");
36006                     delete this.c1;
36007                     delete this.c2;
36008                     this.wasLeaf = true;
36009                 }
36010             }
36011             var ecc = "x-tree-ec-icon "+cls;
36012             if(this.ecc != ecc){
36013                 this.ecNode.className = ecc;
36014                 this.ecc = ecc;
36015             }
36016         }
36017     },
36018
36019     getChildIndent : function(){
36020         if(!this.childIndent){
36021             var buf = [];
36022             var p = this.node;
36023             while(p){
36024                 if(!p.isRoot || (p.isRoot && p.ownerTree.rootVisible)){
36025                     if(!p.isLast()) {
36026                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-elbow-line" />');
36027                     } else {
36028                         buf.unshift('<img src="'+this.emptyIcon+'" class="x-tree-icon" />');
36029                     }
36030                 }
36031                 p = p.parentNode;
36032             }
36033             this.childIndent = buf.join("");
36034         }
36035         return this.childIndent;
36036     },
36037
36038     renderIndent : function(){
36039         if(this.rendered){
36040             var indent = "";
36041             var p = this.node.parentNode;
36042             if(p){
36043                 indent = p.ui.getChildIndent();
36044             }
36045             if(this.indentMarkup != indent){ // don't rerender if not required
36046                 this.indentNode.innerHTML = indent;
36047                 this.indentMarkup = indent;
36048             }
36049             this.updateExpandIcon();
36050         }
36051     }
36052 };
36053
36054 Roo.tree.RootTreeNodeUI = function(){
36055     Roo.tree.RootTreeNodeUI.superclass.constructor.apply(this, arguments);
36056 };
36057 Roo.extend(Roo.tree.RootTreeNodeUI, Roo.tree.TreeNodeUI, {
36058     render : function(){
36059         if(!this.rendered){
36060             var targetNode = this.node.ownerTree.innerCt.dom;
36061             this.node.expanded = true;
36062             targetNode.innerHTML = '<div class="x-tree-root-node"></div>';
36063             this.wrap = this.ctNode = targetNode.firstChild;
36064         }
36065     },
36066     collapse : function(){
36067     },
36068     expand : function(){
36069     }
36070 });/*
36071  * Based on:
36072  * Ext JS Library 1.1.1
36073  * Copyright(c) 2006-2007, Ext JS, LLC.
36074  *
36075  * Originally Released Under LGPL - original licence link has changed is not relivant.
36076  *
36077  * Fork - LGPL
36078  * <script type="text/javascript">
36079  */
36080 /**
36081  * @class Roo.tree.TreeLoader
36082  * @extends Roo.util.Observable
36083  * A TreeLoader provides for lazy loading of an {@link Roo.tree.TreeNode}'s child
36084  * nodes from a specified URL. The response must be a javascript Array definition
36085  * who's elements are node definition objects. eg:
36086  * <pre><code>
36087 {  success : true,
36088    data :      [
36089    
36090     { 'id': 1, 'text': 'A folder Node', 'leaf': false },
36091     { 'id': 2, 'text': 'A leaf Node', 'leaf': true }
36092     ]
36093 }
36094
36095
36096 </code></pre>
36097  * <br><br>
36098  * The old style respose with just an array is still supported, but not recommended.
36099  * <br><br>
36100  *
36101  * A server request is sent, and child nodes are loaded only when a node is expanded.
36102  * The loading node's id is passed to the server under the parameter name "node" to
36103  * enable the server to produce the correct child nodes.
36104  * <br><br>
36105  * To pass extra parameters, an event handler may be attached to the "beforeload"
36106  * event, and the parameters specified in the TreeLoader's baseParams property:
36107  * <pre><code>
36108     myTreeLoader.on("beforeload", function(treeLoader, node) {
36109         this.baseParams.category = node.attributes.category;
36110     }, this);
36111     
36112 </code></pre>
36113  *
36114  * This would pass an HTTP parameter called "category" to the server containing
36115  * the value of the Node's "category" attribute.
36116  * @constructor
36117  * Creates a new Treeloader.
36118  * @param {Object} config A config object containing config properties.
36119  */
36120 Roo.tree.TreeLoader = function(config){
36121     this.baseParams = {};
36122     this.requestMethod = "POST";
36123     Roo.apply(this, config);
36124
36125     this.addEvents({
36126     
36127         /**
36128          * @event beforeload
36129          * Fires before a network request is made to retrieve the Json text which specifies a node's children.
36130          * @param {Object} This TreeLoader object.
36131          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36132          * @param {Object} callback The callback function specified in the {@link #load} call.
36133          */
36134         beforeload : true,
36135         /**
36136          * @event load
36137          * Fires when the node has been successfuly loaded.
36138          * @param {Object} This TreeLoader object.
36139          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36140          * @param {Object} response The response object containing the data from the server.
36141          */
36142         load : true,
36143         /**
36144          * @event loadexception
36145          * Fires if the network request failed.
36146          * @param {Object} This TreeLoader object.
36147          * @param {Object} node The {@link Roo.tree.TreeNode} object being loaded.
36148          * @param {Object} response The response object containing the data from the server.
36149          */
36150         loadexception : true,
36151         /**
36152          * @event create
36153          * Fires before a node is created, enabling you to return custom Node types 
36154          * @param {Object} This TreeLoader object.
36155          * @param {Object} attr - the data returned from the AJAX call (modify it to suit)
36156          */
36157         create : true
36158     });
36159
36160     Roo.tree.TreeLoader.superclass.constructor.call(this);
36161 };
36162
36163 Roo.extend(Roo.tree.TreeLoader, Roo.util.Observable, {
36164     /**
36165     * @cfg {String} dataUrl The URL from which to request a Json string which
36166     * specifies an array of node definition object representing the child nodes
36167     * to be loaded.
36168     */
36169     /**
36170     * @cfg {String} requestMethod either GET or POST
36171     * defaults to POST (due to BC)
36172     * to be loaded.
36173     */
36174     /**
36175     * @cfg {Object} baseParams (optional) An object containing properties which
36176     * specify HTTP parameters to be passed to each request for child nodes.
36177     */
36178     /**
36179     * @cfg {Object} baseAttrs (optional) An object containing attributes to be added to all nodes
36180     * created by this loader. If the attributes sent by the server have an attribute in this object,
36181     * they take priority.
36182     */
36183     /**
36184     * @cfg {Object} uiProviders (optional) An object containing properties which
36185     * 
36186     * DEPRECATED - use 'create' event handler to modify attributes - which affect creation.
36187     * specify custom {@link Roo.tree.TreeNodeUI} implementations. If the optional
36188     * <i>uiProvider</i> attribute of a returned child node is a string rather
36189     * than a reference to a TreeNodeUI implementation, this that string value
36190     * is used as a property name in the uiProviders object. You can define the provider named
36191     * 'default' , and this will be used for all nodes (if no uiProvider is delivered by the node data)
36192     */
36193     uiProviders : {},
36194
36195     /**
36196     * @cfg {Boolean} clearOnLoad (optional) Default to true. Remove previously existing
36197     * child nodes before loading.
36198     */
36199     clearOnLoad : true,
36200
36201     /**
36202     * @cfg {String} root (optional) Default to false. Use this to read data from an object 
36203     * property on loading, rather than expecting an array. (eg. more compatible to a standard
36204     * Grid query { data : [ .....] }
36205     */
36206     
36207     root : false,
36208      /**
36209     * @cfg {String} queryParam (optional) 
36210     * Name of the query as it will be passed on the querystring (defaults to 'node')
36211     * eg. the request will be ?node=[id]
36212     */
36213     
36214     
36215     queryParam: false,
36216     
36217     /**
36218      * Load an {@link Roo.tree.TreeNode} from the URL specified in the constructor.
36219      * This is called automatically when a node is expanded, but may be used to reload
36220      * a node (or append new children if the {@link #clearOnLoad} option is false.)
36221      * @param {Roo.tree.TreeNode} node
36222      * @param {Function} callback
36223      */
36224     load : function(node, callback){
36225         if(this.clearOnLoad){
36226             while(node.firstChild){
36227                 node.removeChild(node.firstChild);
36228             }
36229         }
36230         if(node.attributes.children){ // preloaded json children
36231             var cs = node.attributes.children;
36232             for(var i = 0, len = cs.length; i < len; i++){
36233                 node.appendChild(this.createNode(cs[i]));
36234             }
36235             if(typeof callback == "function"){
36236                 callback();
36237             }
36238         }else if(this.dataUrl){
36239             this.requestData(node, callback);
36240         }
36241     },
36242
36243     getParams: function(node){
36244         var buf = [], bp = this.baseParams;
36245         for(var key in bp){
36246             if(typeof bp[key] != "function"){
36247                 buf.push(encodeURIComponent(key), "=", encodeURIComponent(bp[key]), "&");
36248             }
36249         }
36250         var n = this.queryParam === false ? 'node' : this.queryParam;
36251         buf.push(n + "=", encodeURIComponent(node.id));
36252         return buf.join("");
36253     },
36254
36255     requestData : function(node, callback){
36256         if(this.fireEvent("beforeload", this, node, callback) !== false){
36257             this.transId = Roo.Ajax.request({
36258                 method:this.requestMethod,
36259                 url: this.dataUrl||this.url,
36260                 success: this.handleResponse,
36261                 failure: this.handleFailure,
36262                 scope: this,
36263                 argument: {callback: callback, node: node},
36264                 params: this.getParams(node)
36265             });
36266         }else{
36267             // if the load is cancelled, make sure we notify
36268             // the node that we are done
36269             if(typeof callback == "function"){
36270                 callback();
36271             }
36272         }
36273     },
36274
36275     isLoading : function(){
36276         return this.transId ? true : false;
36277     },
36278
36279     abort : function(){
36280         if(this.isLoading()){
36281             Roo.Ajax.abort(this.transId);
36282         }
36283     },
36284
36285     // private
36286     createNode : function(attr)
36287     {
36288         // apply baseAttrs, nice idea Corey!
36289         if(this.baseAttrs){
36290             Roo.applyIf(attr, this.baseAttrs);
36291         }
36292         if(this.applyLoader !== false){
36293             attr.loader = this;
36294         }
36295         // uiProvider = depreciated..
36296         
36297         if(typeof(attr.uiProvider) == 'string'){
36298            attr.uiProvider = this.uiProviders[attr.uiProvider] || 
36299                 /**  eval:var:attr */ eval(attr.uiProvider);
36300         }
36301         if(typeof(this.uiProviders['default']) != 'undefined') {
36302             attr.uiProvider = this.uiProviders['default'];
36303         }
36304         
36305         this.fireEvent('create', this, attr);
36306         
36307         attr.leaf  = typeof(attr.leaf) == 'string' ? attr.leaf * 1 : attr.leaf;
36308         return(attr.leaf ?
36309                         new Roo.tree.TreeNode(attr) :
36310                         new Roo.tree.AsyncTreeNode(attr));
36311     },
36312
36313     processResponse : function(response, node, callback)
36314     {
36315         var json = response.responseText;
36316         try {
36317             
36318             var o = Roo.decode(json);
36319             
36320             if (this.root === false && typeof(o.success) != undefined) {
36321                 this.root = 'data'; // the default behaviour for list like data..
36322                 }
36323                 
36324             if (this.root !== false &&  !o.success) {
36325                 // it's a failure condition.
36326                 var a = response.argument;
36327                 this.fireEvent("loadexception", this, a.node, response);
36328                 Roo.log("Load failed - should have a handler really");
36329                 return;
36330             }
36331             
36332             
36333             
36334             if (this.root !== false) {
36335                  o = o[this.root];
36336             }
36337             
36338             for(var i = 0, len = o.length; i < len; i++){
36339                 var n = this.createNode(o[i]);
36340                 if(n){
36341                     node.appendChild(n);
36342                 }
36343             }
36344             if(typeof callback == "function"){
36345                 callback(this, node);
36346             }
36347         }catch(e){
36348             this.handleFailure(response);
36349         }
36350     },
36351
36352     handleResponse : function(response){
36353         this.transId = false;
36354         var a = response.argument;
36355         this.processResponse(response, a.node, a.callback);
36356         this.fireEvent("load", this, a.node, response);
36357     },
36358
36359     handleFailure : function(response)
36360     {
36361         // should handle failure better..
36362         this.transId = false;
36363         var a = response.argument;
36364         this.fireEvent("loadexception", this, a.node, response);
36365         if(typeof a.callback == "function"){
36366             a.callback(this, a.node);
36367         }
36368     }
36369 });/*
36370  * Based on:
36371  * Ext JS Library 1.1.1
36372  * Copyright(c) 2006-2007, Ext JS, LLC.
36373  *
36374  * Originally Released Under LGPL - original licence link has changed is not relivant.
36375  *
36376  * Fork - LGPL
36377  * <script type="text/javascript">
36378  */
36379
36380 /**
36381 * @class Roo.tree.TreeFilter
36382 * Note this class is experimental and doesn't update the indent (lines) or expand collapse icons of the nodes
36383 * @param {TreePanel} tree
36384 * @param {Object} config (optional)
36385  */
36386 Roo.tree.TreeFilter = function(tree, config){
36387     this.tree = tree;
36388     this.filtered = {};
36389     Roo.apply(this, config);
36390 };
36391
36392 Roo.tree.TreeFilter.prototype = {
36393     clearBlank:false,
36394     reverse:false,
36395     autoClear:false,
36396     remove:false,
36397
36398      /**
36399      * Filter the data by a specific attribute.
36400      * @param {String/RegExp} value Either string that the attribute value
36401      * should start with or a RegExp to test against the attribute
36402      * @param {String} attr (optional) The attribute passed in your node's attributes collection. Defaults to "text".
36403      * @param {TreeNode} startNode (optional) The node to start the filter at.
36404      */
36405     filter : function(value, attr, startNode){
36406         attr = attr || "text";
36407         var f;
36408         if(typeof value == "string"){
36409             var vlen = value.length;
36410             // auto clear empty filter
36411             if(vlen == 0 && this.clearBlank){
36412                 this.clear();
36413                 return;
36414             }
36415             value = value.toLowerCase();
36416             f = function(n){
36417                 return n.attributes[attr].substr(0, vlen).toLowerCase() == value;
36418             };
36419         }else if(value.exec){ // regex?
36420             f = function(n){
36421                 return value.test(n.attributes[attr]);
36422             };
36423         }else{
36424             throw 'Illegal filter type, must be string or regex';
36425         }
36426         this.filterBy(f, null, startNode);
36427         },
36428
36429     /**
36430      * Filter by a function. The passed function will be called with each
36431      * node in the tree (or from the startNode). If the function returns true, the node is kept
36432      * otherwise it is filtered. If a node is filtered, its children are also filtered.
36433      * @param {Function} fn The filter function
36434      * @param {Object} scope (optional) The scope of the function (defaults to the current node)
36435      */
36436     filterBy : function(fn, scope, startNode){
36437         startNode = startNode || this.tree.root;
36438         if(this.autoClear){
36439             this.clear();
36440         }
36441         var af = this.filtered, rv = this.reverse;
36442         var f = function(n){
36443             if(n == startNode){
36444                 return true;
36445             }
36446             if(af[n.id]){
36447                 return false;
36448             }
36449             var m = fn.call(scope || n, n);
36450             if(!m || rv){
36451                 af[n.id] = n;
36452                 n.ui.hide();
36453                 return false;
36454             }
36455             return true;
36456         };
36457         startNode.cascade(f);
36458         if(this.remove){
36459            for(var id in af){
36460                if(typeof id != "function"){
36461                    var n = af[id];
36462                    if(n && n.parentNode){
36463                        n.parentNode.removeChild(n);
36464                    }
36465                }
36466            }
36467         }
36468     },
36469
36470     /**
36471      * Clears the current filter. Note: with the "remove" option
36472      * set a filter cannot be cleared.
36473      */
36474     clear : function(){
36475         var t = this.tree;
36476         var af = this.filtered;
36477         for(var id in af){
36478             if(typeof id != "function"){
36479                 var n = af[id];
36480                 if(n){
36481                     n.ui.show();
36482                 }
36483             }
36484         }
36485         this.filtered = {};
36486     }
36487 };
36488 /*
36489  * Based on:
36490  * Ext JS Library 1.1.1
36491  * Copyright(c) 2006-2007, Ext JS, LLC.
36492  *
36493  * Originally Released Under LGPL - original licence link has changed is not relivant.
36494  *
36495  * Fork - LGPL
36496  * <script type="text/javascript">
36497  */
36498  
36499
36500 /**
36501  * @class Roo.tree.TreeSorter
36502  * Provides sorting of nodes in a TreePanel
36503  * 
36504  * @cfg {Boolean} folderSort True to sort leaf nodes under non leaf nodes
36505  * @cfg {String} property The named attribute on the node to sort by (defaults to text)
36506  * @cfg {String} dir The direction to sort (asc or desc) (defaults to asc)
36507  * @cfg {String} leafAttr The attribute used to determine leaf nodes in folder sort (defaults to "leaf")
36508  * @cfg {Boolean} caseSensitive true for case sensitive sort (defaults to false)
36509  * @cfg {Function} sortType A custom "casting" function used to convert node values before sorting
36510  * @constructor
36511  * @param {TreePanel} tree
36512  * @param {Object} config
36513  */
36514 Roo.tree.TreeSorter = function(tree, config){
36515     Roo.apply(this, config);
36516     tree.on("beforechildrenrendered", this.doSort, this);
36517     tree.on("append", this.updateSort, this);
36518     tree.on("insert", this.updateSort, this);
36519     
36520     var dsc = this.dir && this.dir.toLowerCase() == "desc";
36521     var p = this.property || "text";
36522     var sortType = this.sortType;
36523     var fs = this.folderSort;
36524     var cs = this.caseSensitive === true;
36525     var leafAttr = this.leafAttr || 'leaf';
36526
36527     this.sortFn = function(n1, n2){
36528         if(fs){
36529             if(n1.attributes[leafAttr] && !n2.attributes[leafAttr]){
36530                 return 1;
36531             }
36532             if(!n1.attributes[leafAttr] && n2.attributes[leafAttr]){
36533                 return -1;
36534             }
36535         }
36536         var v1 = sortType ? sortType(n1) : (cs ? n1.attributes[p] : n1.attributes[p].toUpperCase());
36537         var v2 = sortType ? sortType(n2) : (cs ? n2.attributes[p] : n2.attributes[p].toUpperCase());
36538         if(v1 < v2){
36539                         return dsc ? +1 : -1;
36540                 }else if(v1 > v2){
36541                         return dsc ? -1 : +1;
36542         }else{
36543                 return 0;
36544         }
36545     };
36546 };
36547
36548 Roo.tree.TreeSorter.prototype = {
36549     doSort : function(node){
36550         node.sort(this.sortFn);
36551     },
36552     
36553     compareNodes : function(n1, n2){
36554         return (n1.text.toUpperCase() > n2.text.toUpperCase() ? 1 : -1);
36555     },
36556     
36557     updateSort : function(tree, node){
36558         if(node.childrenRendered){
36559             this.doSort.defer(1, this, [node]);
36560         }
36561     }
36562 };/*
36563  * Based on:
36564  * Ext JS Library 1.1.1
36565  * Copyright(c) 2006-2007, Ext JS, LLC.
36566  *
36567  * Originally Released Under LGPL - original licence link has changed is not relivant.
36568  *
36569  * Fork - LGPL
36570  * <script type="text/javascript">
36571  */
36572
36573 if(Roo.dd.DropZone){
36574     
36575 Roo.tree.TreeDropZone = function(tree, config){
36576     this.allowParentInsert = false;
36577     this.allowContainerDrop = false;
36578     this.appendOnly = false;
36579     Roo.tree.TreeDropZone.superclass.constructor.call(this, tree.innerCt, config);
36580     this.tree = tree;
36581     this.lastInsertClass = "x-tree-no-status";
36582     this.dragOverData = {};
36583 };
36584
36585 Roo.extend(Roo.tree.TreeDropZone, Roo.dd.DropZone, {
36586     ddGroup : "TreeDD",
36587     scroll:  true,
36588     
36589     expandDelay : 1000,
36590     
36591     expandNode : function(node){
36592         if(node.hasChildNodes() && !node.isExpanded()){
36593             node.expand(false, null, this.triggerCacheRefresh.createDelegate(this));
36594         }
36595     },
36596     
36597     queueExpand : function(node){
36598         this.expandProcId = this.expandNode.defer(this.expandDelay, this, [node]);
36599     },
36600     
36601     cancelExpand : function(){
36602         if(this.expandProcId){
36603             clearTimeout(this.expandProcId);
36604             this.expandProcId = false;
36605         }
36606     },
36607     
36608     isValidDropPoint : function(n, pt, dd, e, data){
36609         if(!n || !data){ return false; }
36610         var targetNode = n.node;
36611         var dropNode = data.node;
36612         // default drop rules
36613         if(!(targetNode && targetNode.isTarget && pt)){
36614             return false;
36615         }
36616         if(pt == "append" && targetNode.allowChildren === false){
36617             return false;
36618         }
36619         if((pt == "above" || pt == "below") && (targetNode.parentNode && targetNode.parentNode.allowChildren === false)){
36620             return false;
36621         }
36622         if(dropNode && (targetNode == dropNode || dropNode.contains(targetNode))){
36623             return false;
36624         }
36625         // reuse the object
36626         var overEvent = this.dragOverData;
36627         overEvent.tree = this.tree;
36628         overEvent.target = targetNode;
36629         overEvent.data = data;
36630         overEvent.point = pt;
36631         overEvent.source = dd;
36632         overEvent.rawEvent = e;
36633         overEvent.dropNode = dropNode;
36634         overEvent.cancel = false;  
36635         var result = this.tree.fireEvent("nodedragover", overEvent);
36636         return overEvent.cancel === false && result !== false;
36637     },
36638     
36639     getDropPoint : function(e, n, dd)
36640     {
36641         var tn = n.node;
36642         if(tn.isRoot){
36643             return tn.allowChildren !== false ? "append" : false; // always append for root
36644         }
36645         var dragEl = n.ddel;
36646         var t = Roo.lib.Dom.getY(dragEl), b = t + dragEl.offsetHeight;
36647         var y = Roo.lib.Event.getPageY(e);
36648         //var noAppend = tn.allowChildren === false || tn.isLeaf();
36649         
36650         // we may drop nodes anywhere, as long as allowChildren has not been set to false..
36651         var noAppend = tn.allowChildren === false;
36652         if(this.appendOnly || tn.parentNode.allowChildren === false){
36653             return noAppend ? false : "append";
36654         }
36655         var noBelow = false;
36656         if(!this.allowParentInsert){
36657             noBelow = tn.hasChildNodes() && tn.isExpanded();
36658         }
36659         var q = (b - t) / (noAppend ? 2 : 3);
36660         if(y >= t && y < (t + q)){
36661             return "above";
36662         }else if(!noBelow && (noAppend || y >= b-q && y <= b)){
36663             return "below";
36664         }else{
36665             return "append";
36666         }
36667     },
36668     
36669     onNodeEnter : function(n, dd, e, data)
36670     {
36671         this.cancelExpand();
36672     },
36673     
36674     onNodeOver : function(n, dd, e, data)
36675     {
36676        
36677         var pt = this.getDropPoint(e, n, dd);
36678         var node = n.node;
36679         
36680         // auto node expand check
36681         if(!this.expandProcId && pt == "append" && node.hasChildNodes() && !n.node.isExpanded()){
36682             this.queueExpand(node);
36683         }else if(pt != "append"){
36684             this.cancelExpand();
36685         }
36686         
36687         // set the insert point style on the target node
36688         var returnCls = this.dropNotAllowed;
36689         if(this.isValidDropPoint(n, pt, dd, e, data)){
36690            if(pt){
36691                var el = n.ddel;
36692                var cls;
36693                if(pt == "above"){
36694                    returnCls = n.node.isFirst() ? "x-tree-drop-ok-above" : "x-tree-drop-ok-between";
36695                    cls = "x-tree-drag-insert-above";
36696                }else if(pt == "below"){
36697                    returnCls = n.node.isLast() ? "x-tree-drop-ok-below" : "x-tree-drop-ok-between";
36698                    cls = "x-tree-drag-insert-below";
36699                }else{
36700                    returnCls = "x-tree-drop-ok-append";
36701                    cls = "x-tree-drag-append";
36702                }
36703                if(this.lastInsertClass != cls){
36704                    Roo.fly(el).replaceClass(this.lastInsertClass, cls);
36705                    this.lastInsertClass = cls;
36706                }
36707            }
36708        }
36709        return returnCls;
36710     },
36711     
36712     onNodeOut : function(n, dd, e, data){
36713         
36714         this.cancelExpand();
36715         this.removeDropIndicators(n);
36716     },
36717     
36718     onNodeDrop : function(n, dd, e, data){
36719         var point = this.getDropPoint(e, n, dd);
36720         var targetNode = n.node;
36721         targetNode.ui.startDrop();
36722         if(!this.isValidDropPoint(n, point, dd, e, data)){
36723             targetNode.ui.endDrop();
36724             return false;
36725         }
36726         // first try to find the drop node
36727         var dropNode = data.node || (dd.getTreeNode ? dd.getTreeNode(data, targetNode, point, e) : null);
36728         var dropEvent = {
36729             tree : this.tree,
36730             target: targetNode,
36731             data: data,
36732             point: point,
36733             source: dd,
36734             rawEvent: e,
36735             dropNode: dropNode,
36736             cancel: !dropNode   
36737         };
36738         var retval = this.tree.fireEvent("beforenodedrop", dropEvent);
36739         if(retval === false || dropEvent.cancel === true || !dropEvent.dropNode){
36740             targetNode.ui.endDrop();
36741             return false;
36742         }
36743         // allow target changing
36744         targetNode = dropEvent.target;
36745         if(point == "append" && !targetNode.isExpanded()){
36746             targetNode.expand(false, null, function(){
36747                 this.completeDrop(dropEvent);
36748             }.createDelegate(this));
36749         }else{
36750             this.completeDrop(dropEvent);
36751         }
36752         return true;
36753     },
36754     
36755     completeDrop : function(de){
36756         var ns = de.dropNode, p = de.point, t = de.target;
36757         if(!(ns instanceof Array)){
36758             ns = [ns];
36759         }
36760         var n;
36761         for(var i = 0, len = ns.length; i < len; i++){
36762             n = ns[i];
36763             if(p == "above"){
36764                 t.parentNode.insertBefore(n, t);
36765             }else if(p == "below"){
36766                 t.parentNode.insertBefore(n, t.nextSibling);
36767             }else{
36768                 t.appendChild(n);
36769             }
36770         }
36771         n.ui.focus();
36772         if(this.tree.hlDrop){
36773             n.ui.highlight();
36774         }
36775         t.ui.endDrop();
36776         this.tree.fireEvent("nodedrop", de);
36777     },
36778     
36779     afterNodeMoved : function(dd, data, e, targetNode, dropNode){
36780         if(this.tree.hlDrop){
36781             dropNode.ui.focus();
36782             dropNode.ui.highlight();
36783         }
36784         this.tree.fireEvent("nodedrop", this.tree, targetNode, data, dd, e);
36785     },
36786     
36787     getTree : function(){
36788         return this.tree;
36789     },
36790     
36791     removeDropIndicators : function(n){
36792         if(n && n.ddel){
36793             var el = n.ddel;
36794             Roo.fly(el).removeClass([
36795                     "x-tree-drag-insert-above",
36796                     "x-tree-drag-insert-below",
36797                     "x-tree-drag-append"]);
36798             this.lastInsertClass = "_noclass";
36799         }
36800     },
36801     
36802     beforeDragDrop : function(target, e, id){
36803         this.cancelExpand();
36804         return true;
36805     },
36806     
36807     afterRepair : function(data){
36808         if(data && Roo.enableFx){
36809             data.node.ui.highlight();
36810         }
36811         this.hideProxy();
36812     } 
36813     
36814 });
36815
36816 }
36817 /*
36818  * Based on:
36819  * Ext JS Library 1.1.1
36820  * Copyright(c) 2006-2007, Ext JS, LLC.
36821  *
36822  * Originally Released Under LGPL - original licence link has changed is not relivant.
36823  *
36824  * Fork - LGPL
36825  * <script type="text/javascript">
36826  */
36827  
36828
36829 if(Roo.dd.DragZone){
36830 Roo.tree.TreeDragZone = function(tree, config){
36831     Roo.tree.TreeDragZone.superclass.constructor.call(this, tree.getTreeEl(), config);
36832     this.tree = tree;
36833 };
36834
36835 Roo.extend(Roo.tree.TreeDragZone, Roo.dd.DragZone, {
36836     ddGroup : "TreeDD",
36837    
36838     onBeforeDrag : function(data, e){
36839         var n = data.node;
36840         return n && n.draggable && !n.disabled;
36841     },
36842      
36843     
36844     onInitDrag : function(e){
36845         var data = this.dragData;
36846         this.tree.getSelectionModel().select(data.node);
36847         this.proxy.update("");
36848         data.node.ui.appendDDGhost(this.proxy.ghost.dom);
36849         this.tree.fireEvent("startdrag", this.tree, data.node, e);
36850     },
36851     
36852     getRepairXY : function(e, data){
36853         return data.node.ui.getDDRepairXY();
36854     },
36855     
36856     onEndDrag : function(data, e){
36857         this.tree.fireEvent("enddrag", this.tree, data.node, e);
36858         
36859         
36860     },
36861     
36862     onValidDrop : function(dd, e, id){
36863         this.tree.fireEvent("dragdrop", this.tree, this.dragData.node, dd, e);
36864         this.hideProxy();
36865     },
36866     
36867     beforeInvalidDrop : function(e, id){
36868         // this scrolls the original position back into view
36869         var sm = this.tree.getSelectionModel();
36870         sm.clearSelections();
36871         sm.select(this.dragData.node);
36872     }
36873 });
36874 }/*
36875  * Based on:
36876  * Ext JS Library 1.1.1
36877  * Copyright(c) 2006-2007, Ext JS, LLC.
36878  *
36879  * Originally Released Under LGPL - original licence link has changed is not relivant.
36880  *
36881  * Fork - LGPL
36882  * <script type="text/javascript">
36883  */
36884 /**
36885  * @class Roo.tree.TreeEditor
36886  * @extends Roo.Editor
36887  * Provides editor functionality for inline tree node editing.  Any valid {@link Roo.form.Field} can be used
36888  * as the editor field.
36889  * @constructor
36890  * @param {Object} config (used to be the tree panel.)
36891  * @param {Object} oldconfig DEPRECIATED Either a prebuilt {@link Roo.form.Field} instance or a Field config object
36892  * 
36893  * @cfg {Roo.tree.TreePanel} tree The tree to bind to.
36894  * @cfg {Roo.form.TextField|Object} field The field configuration
36895  *
36896  * 
36897  */
36898 Roo.tree.TreeEditor = function(config, oldconfig) { // was -- (tree, config){
36899     var tree = config;
36900     var field;
36901     if (oldconfig) { // old style..
36902         field = oldconfig.events ? oldconfig : new Roo.form.TextField(oldconfig);
36903     } else {
36904         // new style..
36905         tree = config.tree;
36906         config.field = config.field  || {};
36907         config.field.xtype = 'TextField';
36908         field = Roo.factory(config.field, Roo.form);
36909     }
36910     config = config || {};
36911     
36912     
36913     this.addEvents({
36914         /**
36915          * @event beforenodeedit
36916          * Fires when editing is initiated, but before the value changes.  Editing can be canceled by returning
36917          * false from the handler of this event.
36918          * @param {Editor} this
36919          * @param {Roo.tree.Node} node 
36920          */
36921         "beforenodeedit" : true
36922     });
36923     
36924     //Roo.log(config);
36925     Roo.tree.TreeEditor.superclass.constructor.call(this, field, config);
36926
36927     this.tree = tree;
36928
36929     tree.on('beforeclick', this.beforeNodeClick, this);
36930     tree.getTreeEl().on('mousedown', this.hide, this);
36931     this.on('complete', this.updateNode, this);
36932     this.on('beforestartedit', this.fitToTree, this);
36933     this.on('startedit', this.bindScroll, this, {delay:10});
36934     this.on('specialkey', this.onSpecialKey, this);
36935 };
36936
36937 Roo.extend(Roo.tree.TreeEditor, Roo.Editor, {
36938     /**
36939      * @cfg {String} alignment
36940      * The position to align to (see {@link Roo.Element#alignTo} for more details, defaults to "l-l").
36941      */
36942     alignment: "l-l",
36943     // inherit
36944     autoSize: false,
36945     /**
36946      * @cfg {Boolean} hideEl
36947      * True to hide the bound element while the editor is displayed (defaults to false)
36948      */
36949     hideEl : false,
36950     /**
36951      * @cfg {String} cls
36952      * CSS class to apply to the editor (defaults to "x-small-editor x-tree-editor")
36953      */
36954     cls: "x-small-editor x-tree-editor",
36955     /**
36956      * @cfg {Boolean} shim
36957      * True to shim the editor if selects/iframes could be displayed beneath it (defaults to false)
36958      */
36959     shim:false,
36960     // inherit
36961     shadow:"frame",
36962     /**
36963      * @cfg {Number} maxWidth
36964      * The maximum width in pixels of the editor field (defaults to 250).  Note that if the maxWidth would exceed
36965      * the containing tree element's size, it will be automatically limited for you to the container width, taking
36966      * scroll and client offsets into account prior to each edit.
36967      */
36968     maxWidth: 250,
36969
36970     editDelay : 350,
36971
36972     // private
36973     fitToTree : function(ed, el){
36974         var td = this.tree.getTreeEl().dom, nd = el.dom;
36975         if(td.scrollLeft >  nd.offsetLeft){ // ensure the node left point is visible
36976             td.scrollLeft = nd.offsetLeft;
36977         }
36978         var w = Math.min(
36979                 this.maxWidth,
36980                 (td.clientWidth > 20 ? td.clientWidth : td.offsetWidth) - Math.max(0, nd.offsetLeft-td.scrollLeft) - /*cushion*/5);
36981         this.setSize(w, '');
36982         
36983         return this.fireEvent('beforenodeedit', this, this.editNode);
36984         
36985     },
36986
36987     // private
36988     triggerEdit : function(node){
36989         this.completeEdit();
36990         this.editNode = node;
36991         this.startEdit(node.ui.textNode, node.text);
36992     },
36993
36994     // private
36995     bindScroll : function(){
36996         this.tree.getTreeEl().on('scroll', this.cancelEdit, this);
36997     },
36998
36999     // private
37000     beforeNodeClick : function(node, e){
37001         var sinceLast = (this.lastClick ? this.lastClick.getElapsed() : 0);
37002         this.lastClick = new Date();
37003         if(sinceLast > this.editDelay && this.tree.getSelectionModel().isSelected(node)){
37004             e.stopEvent();
37005             this.triggerEdit(node);
37006             return false;
37007         }
37008         return true;
37009     },
37010
37011     // private
37012     updateNode : function(ed, value){
37013         this.tree.getTreeEl().un('scroll', this.cancelEdit, this);
37014         this.editNode.setText(value);
37015     },
37016
37017     // private
37018     onHide : function(){
37019         Roo.tree.TreeEditor.superclass.onHide.call(this);
37020         if(this.editNode){
37021             this.editNode.ui.focus();
37022         }
37023     },
37024
37025     // private
37026     onSpecialKey : function(field, e){
37027         var k = e.getKey();
37028         if(k == e.ESC){
37029             e.stopEvent();
37030             this.cancelEdit();
37031         }else if(k == e.ENTER && !e.hasModifier()){
37032             e.stopEvent();
37033             this.completeEdit();
37034         }
37035     }
37036 });//<Script type="text/javascript">
37037 /*
37038  * Based on:
37039  * Ext JS Library 1.1.1
37040  * Copyright(c) 2006-2007, Ext JS, LLC.
37041  *
37042  * Originally Released Under LGPL - original licence link has changed is not relivant.
37043  *
37044  * Fork - LGPL
37045  * <script type="text/javascript">
37046  */
37047  
37048 /**
37049  * Not documented??? - probably should be...
37050  */
37051
37052 Roo.tree.ColumnNodeUI = Roo.extend(Roo.tree.TreeNodeUI, {
37053     //focus: Roo.emptyFn, // prevent odd scrolling behavior
37054     
37055     renderElements : function(n, a, targetNode, bulkRender){
37056         //consel.log("renderElements?");
37057         this.indentMarkup = n.parentNode ? n.parentNode.ui.getChildIndent() : '';
37058
37059         var t = n.getOwnerTree();
37060         var tid = Pman.Tab.Document_TypesTree.tree.el.id;
37061         
37062         var cols = t.columns;
37063         var bw = t.borderWidth;
37064         var c = cols[0];
37065         var href = a.href ? a.href : Roo.isGecko ? "" : "#";
37066          var cb = typeof a.checked == "boolean";
37067         var tx = String.format('{0}',n.text || (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37068         var colcls = 'x-t-' + tid + '-c0';
37069         var buf = [
37070             '<li class="x-tree-node">',
37071             
37072                 
37073                 '<div class="x-tree-node-el ', a.cls,'">',
37074                     // extran...
37075                     '<div class="x-tree-col ', colcls, '" style="width:', c.width-bw, 'px;">',
37076                 
37077                 
37078                         '<span class="x-tree-node-indent">',this.indentMarkup,'</span>',
37079                         '<img src="', this.emptyIcon, '" class="x-tree-ec-icon  " />',
37080                         '<img src="', a.icon || this.emptyIcon, '" class="x-tree-node-icon',
37081                            (a.icon ? ' x-tree-node-inline-icon' : ''),
37082                            (a.iconCls ? ' '+a.iconCls : ''),
37083                            '" unselectable="on" />',
37084                         (cb ? ('<input class="x-tree-node-cb" type="checkbox" ' + 
37085                              (a.checked ? 'checked="checked" />' : ' />')) : ''),
37086                              
37087                         '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37088                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>',
37089                             '<span unselectable="on" qtip="' + tx + '">',
37090                              tx,
37091                              '</span></a>' ,
37092                     '</div>',
37093                      '<a class="x-tree-node-anchor" hidefocus="on" href="',href,'" tabIndex="1" ',
37094                             (a.hrefTarget ? ' target="' +a.hrefTarget + '"' : ''), '>'
37095                  ];
37096         for(var i = 1, len = cols.length; i < len; i++){
37097             c = cols[i];
37098             colcls = 'x-t-' + tid + '-c' +i;
37099             tx = String.format('{0}', (c.renderer ? c.renderer(a[c.dataIndex], n, a) : a[c.dataIndex]));
37100             buf.push('<div class="x-tree-col ', colcls, ' ' ,(c.cls?c.cls:''),'" style="width:',c.width-bw,'px;">',
37101                         '<div class="x-tree-col-text" qtip="' + tx +'">',tx,"</div>",
37102                       "</div>");
37103          }
37104          
37105          buf.push(
37106             '</a>',
37107             '<div class="x-clear"></div></div>',
37108             '<ul class="x-tree-node-ct" style="display:none;"></ul>',
37109             "</li>");
37110         
37111         if(bulkRender !== true && n.nextSibling && n.nextSibling.ui.getEl()){
37112             this.wrap = Roo.DomHelper.insertHtml("beforeBegin",
37113                                 n.nextSibling.ui.getEl(), buf.join(""));
37114         }else{
37115             this.wrap = Roo.DomHelper.insertHtml("beforeEnd", targetNode, buf.join(""));
37116         }
37117         var el = this.wrap.firstChild;
37118         this.elRow = el;
37119         this.elNode = el.firstChild;
37120         this.ranchor = el.childNodes[1];
37121         this.ctNode = this.wrap.childNodes[1];
37122         var cs = el.firstChild.childNodes;
37123         this.indentNode = cs[0];
37124         this.ecNode = cs[1];
37125         this.iconNode = cs[2];
37126         var index = 3;
37127         if(cb){
37128             this.checkbox = cs[3];
37129             index++;
37130         }
37131         this.anchor = cs[index];
37132         
37133         this.textNode = cs[index].firstChild;
37134         
37135         //el.on("click", this.onClick, this);
37136         //el.on("dblclick", this.onDblClick, this);
37137         
37138         
37139        // console.log(this);
37140     },
37141     initEvents : function(){
37142         Roo.tree.ColumnNodeUI.superclass.initEvents.call(this);
37143         
37144             
37145         var a = this.ranchor;
37146
37147         var el = Roo.get(a);
37148
37149         if(Roo.isOpera){ // opera render bug ignores the CSS
37150             el.setStyle("text-decoration", "none");
37151         }
37152
37153         el.on("click", this.onClick, this);
37154         el.on("dblclick", this.onDblClick, this);
37155         el.on("contextmenu", this.onContextMenu, this);
37156         
37157     },
37158     
37159     /*onSelectedChange : function(state){
37160         if(state){
37161             this.focus();
37162             this.addClass("x-tree-selected");
37163         }else{
37164             //this.blur();
37165             this.removeClass("x-tree-selected");
37166         }
37167     },*/
37168     addClass : function(cls){
37169         if(this.elRow){
37170             Roo.fly(this.elRow).addClass(cls);
37171         }
37172         
37173     },
37174     
37175     
37176     removeClass : function(cls){
37177         if(this.elRow){
37178             Roo.fly(this.elRow).removeClass(cls);
37179         }
37180     }
37181
37182     
37183     
37184 });//<Script type="text/javascript">
37185
37186 /*
37187  * Based on:
37188  * Ext JS Library 1.1.1
37189  * Copyright(c) 2006-2007, Ext JS, LLC.
37190  *
37191  * Originally Released Under LGPL - original licence link has changed is not relivant.
37192  *
37193  * Fork - LGPL
37194  * <script type="text/javascript">
37195  */
37196  
37197
37198 /**
37199  * @class Roo.tree.ColumnTree
37200  * @extends Roo.data.TreePanel
37201  * @cfg {Object} columns  Including width, header, renderer, cls, dataIndex 
37202  * @cfg {int} borderWidth  compined right/left border allowance
37203  * @constructor
37204  * @param {String/HTMLElement/Element} el The container element
37205  * @param {Object} config
37206  */
37207 Roo.tree.ColumnTree =  function(el, config)
37208 {
37209    Roo.tree.ColumnTree.superclass.constructor.call(this, el , config);
37210    this.addEvents({
37211         /**
37212         * @event resize
37213         * Fire this event on a container when it resizes
37214         * @param {int} w Width
37215         * @param {int} h Height
37216         */
37217        "resize" : true
37218     });
37219     this.on('resize', this.onResize, this);
37220 };
37221
37222 Roo.extend(Roo.tree.ColumnTree, Roo.tree.TreePanel, {
37223     //lines:false,
37224     
37225     
37226     borderWidth: Roo.isBorderBox ? 0 : 2, 
37227     headEls : false,
37228     
37229     render : function(){
37230         // add the header.....
37231        
37232         Roo.tree.ColumnTree.superclass.render.apply(this);
37233         
37234         this.el.addClass('x-column-tree');
37235         
37236         this.headers = this.el.createChild(
37237             {cls:'x-tree-headers'},this.innerCt.dom);
37238    
37239         var cols = this.columns, c;
37240         var totalWidth = 0;
37241         this.headEls = [];
37242         var  len = cols.length;
37243         for(var i = 0; i < len; i++){
37244              c = cols[i];
37245              totalWidth += c.width;
37246             this.headEls.push(this.headers.createChild({
37247                  cls:'x-tree-hd ' + (c.cls?c.cls+'-hd':''),
37248                  cn: {
37249                      cls:'x-tree-hd-text',
37250                      html: c.header
37251                  },
37252                  style:'width:'+(c.width-this.borderWidth)+'px;'
37253              }));
37254         }
37255         this.headers.createChild({cls:'x-clear'});
37256         // prevent floats from wrapping when clipped
37257         this.headers.setWidth(totalWidth);
37258         //this.innerCt.setWidth(totalWidth);
37259         this.innerCt.setStyle({ overflow: 'auto' });
37260         this.onResize(this.width, this.height);
37261              
37262         
37263     },
37264     onResize : function(w,h)
37265     {
37266         this.height = h;
37267         this.width = w;
37268         // resize cols..
37269         this.innerCt.setWidth(this.width);
37270         this.innerCt.setHeight(this.height-20);
37271         
37272         // headers...
37273         var cols = this.columns, c;
37274         var totalWidth = 0;
37275         var expEl = false;
37276         var len = cols.length;
37277         for(var i = 0; i < len; i++){
37278             c = cols[i];
37279             if (this.autoExpandColumn !== false && c.dataIndex == this.autoExpandColumn) {
37280                 // it's the expander..
37281                 expEl  = this.headEls[i];
37282                 continue;
37283             }
37284             totalWidth += c.width;
37285             
37286         }
37287         if (expEl) {
37288             expEl.setWidth(  ((w - totalWidth)-this.borderWidth - 20));
37289         }
37290         this.headers.setWidth(w-20);
37291
37292         
37293         
37294         
37295     }
37296 });
37297 /*
37298  * Based on:
37299  * Ext JS Library 1.1.1
37300  * Copyright(c) 2006-2007, Ext JS, LLC.
37301  *
37302  * Originally Released Under LGPL - original licence link has changed is not relivant.
37303  *
37304  * Fork - LGPL
37305  * <script type="text/javascript">
37306  */
37307  
37308 /**
37309  * @class Roo.menu.Menu
37310  * @extends Roo.util.Observable
37311  * A menu object.  This is the container to which you add all other menu items.  Menu can also serve a as a base class
37312  * when you want a specialzed menu based off of another component (like {@link Roo.menu.DateMenu} for example).
37313  * @constructor
37314  * Creates a new Menu
37315  * @param {Object} config Configuration options
37316  */
37317 Roo.menu.Menu = function(config){
37318     
37319     Roo.menu.Menu.superclass.constructor.call(this, config);
37320     
37321     this.id = this.id || Roo.id();
37322     this.addEvents({
37323         /**
37324          * @event beforeshow
37325          * Fires before this menu is displayed
37326          * @param {Roo.menu.Menu} this
37327          */
37328         beforeshow : true,
37329         /**
37330          * @event beforehide
37331          * Fires before this menu is hidden
37332          * @param {Roo.menu.Menu} this
37333          */
37334         beforehide : true,
37335         /**
37336          * @event show
37337          * Fires after this menu is displayed
37338          * @param {Roo.menu.Menu} this
37339          */
37340         show : true,
37341         /**
37342          * @event hide
37343          * Fires after this menu is hidden
37344          * @param {Roo.menu.Menu} this
37345          */
37346         hide : true,
37347         /**
37348          * @event click
37349          * Fires when this menu is clicked (or when the enter key is pressed while it is active)
37350          * @param {Roo.menu.Menu} this
37351          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37352          * @param {Roo.EventObject} e
37353          */
37354         click : true,
37355         /**
37356          * @event mouseover
37357          * Fires when the mouse is hovering over this menu
37358          * @param {Roo.menu.Menu} this
37359          * @param {Roo.EventObject} e
37360          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37361          */
37362         mouseover : true,
37363         /**
37364          * @event mouseout
37365          * Fires when the mouse exits this menu
37366          * @param {Roo.menu.Menu} this
37367          * @param {Roo.EventObject} e
37368          * @param {Roo.menu.Item} menuItem The menu item that was clicked
37369          */
37370         mouseout : true,
37371         /**
37372          * @event itemclick
37373          * Fires when a menu item contained in this menu is clicked
37374          * @param {Roo.menu.BaseItem} baseItem The BaseItem that was clicked
37375          * @param {Roo.EventObject} e
37376          */
37377         itemclick: true
37378     });
37379     if (this.registerMenu) {
37380         Roo.menu.MenuMgr.register(this);
37381     }
37382     
37383     var mis = this.items;
37384     this.items = new Roo.util.MixedCollection();
37385     if(mis){
37386         this.add.apply(this, mis);
37387     }
37388 };
37389
37390 Roo.extend(Roo.menu.Menu, Roo.util.Observable, {
37391     /**
37392      * @cfg {Number} minWidth The minimum width of the menu in pixels (defaults to 120)
37393      */
37394     minWidth : 120,
37395     /**
37396      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop"
37397      * for bottom-right shadow (defaults to "sides")
37398      */
37399     shadow : "sides",
37400     /**
37401      * @cfg {String} subMenuAlign The {@link Roo.Element#alignTo} anchor position value to use for submenus of
37402      * this menu (defaults to "tl-tr?")
37403      */
37404     subMenuAlign : "tl-tr?",
37405     /**
37406      * @cfg {String} defaultAlign The default {@link Roo.Element#alignTo) anchor position value for this menu
37407      * relative to its element of origin (defaults to "tl-bl?")
37408      */
37409     defaultAlign : "tl-bl?",
37410     /**
37411      * @cfg {Boolean} allowOtherMenus True to allow multiple menus to be displayed at the same time (defaults to false)
37412      */
37413     allowOtherMenus : false,
37414     /**
37415      * @cfg {Boolean} registerMenu True (default) - means that clicking on screen etc. hides it.
37416      */
37417     registerMenu : true,
37418
37419     hidden:true,
37420
37421     // private
37422     render : function(){
37423         if(this.el){
37424             return;
37425         }
37426         var el = this.el = new Roo.Layer({
37427             cls: "x-menu",
37428             shadow:this.shadow,
37429             constrain: false,
37430             parentEl: this.parentEl || document.body,
37431             zindex:15000
37432         });
37433
37434         this.keyNav = new Roo.menu.MenuNav(this);
37435
37436         if(this.plain){
37437             el.addClass("x-menu-plain");
37438         }
37439         if(this.cls){
37440             el.addClass(this.cls);
37441         }
37442         // generic focus element
37443         this.focusEl = el.createChild({
37444             tag: "a", cls: "x-menu-focus", href: "#", onclick: "return false;", tabIndex:"-1"
37445         });
37446         var ul = el.createChild({tag: "ul", cls: "x-menu-list"});
37447         //disabling touch- as it's causing issues ..
37448         //ul.on(Roo.isTouch ? 'touchstart' : 'click'   , this.onClick, this);
37449         ul.on('click'   , this.onClick, this);
37450         
37451         
37452         ul.on("mouseover", this.onMouseOver, this);
37453         ul.on("mouseout", this.onMouseOut, this);
37454         this.items.each(function(item){
37455             if (item.hidden) {
37456                 return;
37457             }
37458             
37459             var li = document.createElement("li");
37460             li.className = "x-menu-list-item";
37461             ul.dom.appendChild(li);
37462             item.render(li, this);
37463         }, this);
37464         this.ul = ul;
37465         this.autoWidth();
37466     },
37467
37468     // private
37469     autoWidth : function(){
37470         var el = this.el, ul = this.ul;
37471         if(!el){
37472             return;
37473         }
37474         var w = this.width;
37475         if(w){
37476             el.setWidth(w);
37477         }else if(Roo.isIE){
37478             el.setWidth(this.minWidth);
37479             var t = el.dom.offsetWidth; // force recalc
37480             el.setWidth(ul.getWidth()+el.getFrameWidth("lr"));
37481         }
37482     },
37483
37484     // private
37485     delayAutoWidth : function(){
37486         if(this.rendered){
37487             if(!this.awTask){
37488                 this.awTask = new Roo.util.DelayedTask(this.autoWidth, this);
37489             }
37490             this.awTask.delay(20);
37491         }
37492     },
37493
37494     // private
37495     findTargetItem : function(e){
37496         var t = e.getTarget(".x-menu-list-item", this.ul,  true);
37497         if(t && t.menuItemId){
37498             return this.items.get(t.menuItemId);
37499         }
37500     },
37501
37502     // private
37503     onClick : function(e){
37504         Roo.log("menu.onClick");
37505         var t = this.findTargetItem(e);
37506         if(!t){
37507             return;
37508         }
37509         Roo.log(e);
37510         if (Roo.isTouch && e.type == 'touchstart' && t.menu  && !t.disabled) {
37511             if(t == this.activeItem && t.shouldDeactivate(e)){
37512                 this.activeItem.deactivate();
37513                 delete this.activeItem;
37514                 return;
37515             }
37516             if(t.canActivate){
37517                 this.setActiveItem(t, true);
37518             }
37519             return;
37520             
37521             
37522         }
37523         
37524         t.onClick(e);
37525         this.fireEvent("click", this, t, e);
37526     },
37527
37528     // private
37529     setActiveItem : function(item, autoExpand){
37530         if(item != this.activeItem){
37531             if(this.activeItem){
37532                 this.activeItem.deactivate();
37533             }
37534             this.activeItem = item;
37535             item.activate(autoExpand);
37536         }else if(autoExpand){
37537             item.expandMenu();
37538         }
37539     },
37540
37541     // private
37542     tryActivate : function(start, step){
37543         var items = this.items;
37544         for(var i = start, len = items.length; i >= 0 && i < len; i+= step){
37545             var item = items.get(i);
37546             if(!item.disabled && item.canActivate){
37547                 this.setActiveItem(item, false);
37548                 return item;
37549             }
37550         }
37551         return false;
37552     },
37553
37554     // private
37555     onMouseOver : function(e){
37556         var t;
37557         if(t = this.findTargetItem(e)){
37558             if(t.canActivate && !t.disabled){
37559                 this.setActiveItem(t, true);
37560             }
37561         }
37562         this.fireEvent("mouseover", this, e, t);
37563     },
37564
37565     // private
37566     onMouseOut : function(e){
37567         var t;
37568         if(t = this.findTargetItem(e)){
37569             if(t == this.activeItem && t.shouldDeactivate(e)){
37570                 this.activeItem.deactivate();
37571                 delete this.activeItem;
37572             }
37573         }
37574         this.fireEvent("mouseout", this, e, t);
37575     },
37576
37577     /**
37578      * Read-only.  Returns true if the menu is currently displayed, else false.
37579      * @type Boolean
37580      */
37581     isVisible : function(){
37582         return this.el && !this.hidden;
37583     },
37584
37585     /**
37586      * Displays this menu relative to another element
37587      * @param {String/HTMLElement/Roo.Element} element The element to align to
37588      * @param {String} position (optional) The {@link Roo.Element#alignTo} anchor position to use in aligning to
37589      * the element (defaults to this.defaultAlign)
37590      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37591      */
37592     show : function(el, pos, parentMenu){
37593         this.parentMenu = parentMenu;
37594         if(!this.el){
37595             this.render();
37596         }
37597         this.fireEvent("beforeshow", this);
37598         this.showAt(this.el.getAlignToXY(el, pos || this.defaultAlign), parentMenu, false);
37599     },
37600
37601     /**
37602      * Displays this menu at a specific xy position
37603      * @param {Array} xyPosition Contains X & Y [x, y] values for the position at which to show the menu (coordinates are page-based)
37604      * @param {Roo.menu.Menu} parentMenu (optional) This menu's parent menu, if applicable (defaults to undefined)
37605      */
37606     showAt : function(xy, parentMenu, /* private: */_e){
37607         this.parentMenu = parentMenu;
37608         if(!this.el){
37609             this.render();
37610         }
37611         if(_e !== false){
37612             this.fireEvent("beforeshow", this);
37613             xy = this.el.adjustForConstraints(xy);
37614         }
37615         this.el.setXY(xy);
37616         this.el.show();
37617         this.hidden = false;
37618         this.focus();
37619         this.fireEvent("show", this);
37620     },
37621
37622     focus : function(){
37623         if(!this.hidden){
37624             this.doFocus.defer(50, this);
37625         }
37626     },
37627
37628     doFocus : function(){
37629         if(!this.hidden){
37630             this.focusEl.focus();
37631         }
37632     },
37633
37634     /**
37635      * Hides this menu and optionally all parent menus
37636      * @param {Boolean} deep (optional) True to hide all parent menus recursively, if any (defaults to false)
37637      */
37638     hide : function(deep){
37639         if(this.el && this.isVisible()){
37640             this.fireEvent("beforehide", this);
37641             if(this.activeItem){
37642                 this.activeItem.deactivate();
37643                 this.activeItem = null;
37644             }
37645             this.el.hide();
37646             this.hidden = true;
37647             this.fireEvent("hide", this);
37648         }
37649         if(deep === true && this.parentMenu){
37650             this.parentMenu.hide(true);
37651         }
37652     },
37653
37654     /**
37655      * Addds one or more items of any type supported by the Menu class, or that can be converted into menu items.
37656      * Any of the following are valid:
37657      * <ul>
37658      * <li>Any menu item object based on {@link Roo.menu.Item}</li>
37659      * <li>An HTMLElement object which will be converted to a menu item</li>
37660      * <li>A menu item config object that will be created as a new menu item</li>
37661      * <li>A string, which can either be '-' or 'separator' to add a menu separator, otherwise
37662      * it will be converted into a {@link Roo.menu.TextItem} and added</li>
37663      * </ul>
37664      * Usage:
37665      * <pre><code>
37666 // Create the menu
37667 var menu = new Roo.menu.Menu();
37668
37669 // Create a menu item to add by reference
37670 var menuItem = new Roo.menu.Item({ text: 'New Item!' });
37671
37672 // Add a bunch of items at once using different methods.
37673 // Only the last item added will be returned.
37674 var item = menu.add(
37675     menuItem,                // add existing item by ref
37676     'Dynamic Item',          // new TextItem
37677     '-',                     // new separator
37678     { text: 'Config Item' }  // new item by config
37679 );
37680 </code></pre>
37681      * @param {Mixed} args One or more menu items, menu item configs or other objects that can be converted to menu items
37682      * @return {Roo.menu.Item} The menu item that was added, or the last one if multiple items were added
37683      */
37684     add : function(){
37685         var a = arguments, l = a.length, item;
37686         for(var i = 0; i < l; i++){
37687             var el = a[i];
37688             if ((typeof(el) == "object") && el.xtype && el.xns) {
37689                 el = Roo.factory(el, Roo.menu);
37690             }
37691             
37692             if(el.render){ // some kind of Item
37693                 item = this.addItem(el);
37694             }else if(typeof el == "string"){ // string
37695                 if(el == "separator" || el == "-"){
37696                     item = this.addSeparator();
37697                 }else{
37698                     item = this.addText(el);
37699                 }
37700             }else if(el.tagName || el.el){ // element
37701                 item = this.addElement(el);
37702             }else if(typeof el == "object"){ // must be menu item config?
37703                 item = this.addMenuItem(el);
37704             }
37705         }
37706         return item;
37707     },
37708
37709     /**
37710      * Returns this menu's underlying {@link Roo.Element} object
37711      * @return {Roo.Element} The element
37712      */
37713     getEl : function(){
37714         if(!this.el){
37715             this.render();
37716         }
37717         return this.el;
37718     },
37719
37720     /**
37721      * Adds a separator bar to the menu
37722      * @return {Roo.menu.Item} The menu item that was added
37723      */
37724     addSeparator : function(){
37725         return this.addItem(new Roo.menu.Separator());
37726     },
37727
37728     /**
37729      * Adds an {@link Roo.Element} object to the menu
37730      * @param {String/HTMLElement/Roo.Element} el The element or DOM node to add, or its id
37731      * @return {Roo.menu.Item} The menu item that was added
37732      */
37733     addElement : function(el){
37734         return this.addItem(new Roo.menu.BaseItem(el));
37735     },
37736
37737     /**
37738      * Adds an existing object based on {@link Roo.menu.Item} to the menu
37739      * @param {Roo.menu.Item} item The menu item to add
37740      * @return {Roo.menu.Item} The menu item that was added
37741      */
37742     addItem : function(item){
37743         this.items.add(item);
37744         if(this.ul){
37745             var li = document.createElement("li");
37746             li.className = "x-menu-list-item";
37747             this.ul.dom.appendChild(li);
37748             item.render(li, this);
37749             this.delayAutoWidth();
37750         }
37751         return item;
37752     },
37753
37754     /**
37755      * Creates a new {@link Roo.menu.Item} based an the supplied config object and adds it to the menu
37756      * @param {Object} config A MenuItem config object
37757      * @return {Roo.menu.Item} The menu item that was added
37758      */
37759     addMenuItem : function(config){
37760         if(!(config instanceof Roo.menu.Item)){
37761             if(typeof config.checked == "boolean"){ // must be check menu item config?
37762                 config = new Roo.menu.CheckItem(config);
37763             }else{
37764                 config = new Roo.menu.Item(config);
37765             }
37766         }
37767         return this.addItem(config);
37768     },
37769
37770     /**
37771      * Creates a new {@link Roo.menu.TextItem} with the supplied text and adds it to the menu
37772      * @param {String} text The text to display in the menu item
37773      * @return {Roo.menu.Item} The menu item that was added
37774      */
37775     addText : function(text){
37776         return this.addItem(new Roo.menu.TextItem({ text : text }));
37777     },
37778
37779     /**
37780      * Inserts an existing object based on {@link Roo.menu.Item} to the menu at a specified index
37781      * @param {Number} index The index in the menu's list of current items where the new item should be inserted
37782      * @param {Roo.menu.Item} item The menu item to add
37783      * @return {Roo.menu.Item} The menu item that was added
37784      */
37785     insert : function(index, item){
37786         this.items.insert(index, item);
37787         if(this.ul){
37788             var li = document.createElement("li");
37789             li.className = "x-menu-list-item";
37790             this.ul.dom.insertBefore(li, this.ul.dom.childNodes[index]);
37791             item.render(li, this);
37792             this.delayAutoWidth();
37793         }
37794         return item;
37795     },
37796
37797     /**
37798      * Removes an {@link Roo.menu.Item} from the menu and destroys the object
37799      * @param {Roo.menu.Item} item The menu item to remove
37800      */
37801     remove : function(item){
37802         this.items.removeKey(item.id);
37803         item.destroy();
37804     },
37805
37806     /**
37807      * Removes and destroys all items in the menu
37808      */
37809     removeAll : function(){
37810         var f;
37811         while(f = this.items.first()){
37812             this.remove(f);
37813         }
37814     }
37815 });
37816
37817 // MenuNav is a private utility class used internally by the Menu
37818 Roo.menu.MenuNav = function(menu){
37819     Roo.menu.MenuNav.superclass.constructor.call(this, menu.el);
37820     this.scope = this.menu = menu;
37821 };
37822
37823 Roo.extend(Roo.menu.MenuNav, Roo.KeyNav, {
37824     doRelay : function(e, h){
37825         var k = e.getKey();
37826         if(!this.menu.activeItem && e.isNavKeyPress() && k != e.SPACE && k != e.RETURN){
37827             this.menu.tryActivate(0, 1);
37828             return false;
37829         }
37830         return h.call(this.scope || this, e, this.menu);
37831     },
37832
37833     up : function(e, m){
37834         if(!m.tryActivate(m.items.indexOf(m.activeItem)-1, -1)){
37835             m.tryActivate(m.items.length-1, -1);
37836         }
37837     },
37838
37839     down : function(e, m){
37840         if(!m.tryActivate(m.items.indexOf(m.activeItem)+1, 1)){
37841             m.tryActivate(0, 1);
37842         }
37843     },
37844
37845     right : function(e, m){
37846         if(m.activeItem){
37847             m.activeItem.expandMenu(true);
37848         }
37849     },
37850
37851     left : function(e, m){
37852         m.hide();
37853         if(m.parentMenu && m.parentMenu.activeItem){
37854             m.parentMenu.activeItem.activate();
37855         }
37856     },
37857
37858     enter : function(e, m){
37859         if(m.activeItem){
37860             e.stopPropagation();
37861             m.activeItem.onClick(e);
37862             m.fireEvent("click", this, m.activeItem);
37863             return true;
37864         }
37865     }
37866 });/*
37867  * Based on:
37868  * Ext JS Library 1.1.1
37869  * Copyright(c) 2006-2007, Ext JS, LLC.
37870  *
37871  * Originally Released Under LGPL - original licence link has changed is not relivant.
37872  *
37873  * Fork - LGPL
37874  * <script type="text/javascript">
37875  */
37876  
37877 /**
37878  * @class Roo.menu.MenuMgr
37879  * Provides a common registry of all menu items on a page so that they can be easily accessed by id.
37880  * @singleton
37881  */
37882 Roo.menu.MenuMgr = function(){
37883    var menus, active, groups = {}, attached = false, lastShow = new Date();
37884
37885    // private - called when first menu is created
37886    function init(){
37887        menus = {};
37888        active = new Roo.util.MixedCollection();
37889        Roo.get(document).addKeyListener(27, function(){
37890            if(active.length > 0){
37891                hideAll();
37892            }
37893        });
37894    }
37895
37896    // private
37897    function hideAll(){
37898        if(active && active.length > 0){
37899            var c = active.clone();
37900            c.each(function(m){
37901                m.hide();
37902            });
37903        }
37904    }
37905
37906    // private
37907    function onHide(m){
37908        active.remove(m);
37909        if(active.length < 1){
37910            Roo.get(document).un("mousedown", onMouseDown);
37911            attached = false;
37912        }
37913    }
37914
37915    // private
37916    function onShow(m){
37917        var last = active.last();
37918        lastShow = new Date();
37919        active.add(m);
37920        if(!attached){
37921            Roo.get(document).on("mousedown", onMouseDown);
37922            attached = true;
37923        }
37924        if(m.parentMenu){
37925           m.getEl().setZIndex(parseInt(m.parentMenu.getEl().getStyle("z-index"), 10) + 3);
37926           m.parentMenu.activeChild = m;
37927        }else if(last && last.isVisible()){
37928           m.getEl().setZIndex(parseInt(last.getEl().getStyle("z-index"), 10) + 3);
37929        }
37930    }
37931
37932    // private
37933    function onBeforeHide(m){
37934        if(m.activeChild){
37935            m.activeChild.hide();
37936        }
37937        if(m.autoHideTimer){
37938            clearTimeout(m.autoHideTimer);
37939            delete m.autoHideTimer;
37940        }
37941    }
37942
37943    // private
37944    function onBeforeShow(m){
37945        var pm = m.parentMenu;
37946        if(!pm && !m.allowOtherMenus){
37947            hideAll();
37948        }else if(pm && pm.activeChild && active != m){
37949            pm.activeChild.hide();
37950        }
37951    }
37952
37953    // private
37954    function onMouseDown(e){
37955        if(lastShow.getElapsed() > 50 && active.length > 0 && !e.getTarget(".x-menu")){
37956            hideAll();
37957        }
37958    }
37959
37960    // private
37961    function onBeforeCheck(mi, state){
37962        if(state){
37963            var g = groups[mi.group];
37964            for(var i = 0, l = g.length; i < l; i++){
37965                if(g[i] != mi){
37966                    g[i].setChecked(false);
37967                }
37968            }
37969        }
37970    }
37971
37972    return {
37973
37974        /**
37975         * Hides all menus that are currently visible
37976         */
37977        hideAll : function(){
37978             hideAll();  
37979        },
37980
37981        // private
37982        register : function(menu){
37983            if(!menus){
37984                init();
37985            }
37986            menus[menu.id] = menu;
37987            menu.on("beforehide", onBeforeHide);
37988            menu.on("hide", onHide);
37989            menu.on("beforeshow", onBeforeShow);
37990            menu.on("show", onShow);
37991            var g = menu.group;
37992            if(g && menu.events["checkchange"]){
37993                if(!groups[g]){
37994                    groups[g] = [];
37995                }
37996                groups[g].push(menu);
37997                menu.on("checkchange", onCheck);
37998            }
37999        },
38000
38001         /**
38002          * Returns a {@link Roo.menu.Menu} object
38003          * @param {String/Object} menu The string menu id, an existing menu object reference, or a Menu config that will
38004          * be used to generate and return a new Menu instance.
38005          */
38006        get : function(menu){
38007            if(typeof menu == "string"){ // menu id
38008                return menus[menu];
38009            }else if(menu.events){  // menu instance
38010                return menu;
38011            }else if(typeof menu.length == 'number'){ // array of menu items?
38012                return new Roo.menu.Menu({items:menu});
38013            }else{ // otherwise, must be a config
38014                return new Roo.menu.Menu(menu);
38015            }
38016        },
38017
38018        // private
38019        unregister : function(menu){
38020            delete menus[menu.id];
38021            menu.un("beforehide", onBeforeHide);
38022            menu.un("hide", onHide);
38023            menu.un("beforeshow", onBeforeShow);
38024            menu.un("show", onShow);
38025            var g = menu.group;
38026            if(g && menu.events["checkchange"]){
38027                groups[g].remove(menu);
38028                menu.un("checkchange", onCheck);
38029            }
38030        },
38031
38032        // private
38033        registerCheckable : function(menuItem){
38034            var g = menuItem.group;
38035            if(g){
38036                if(!groups[g]){
38037                    groups[g] = [];
38038                }
38039                groups[g].push(menuItem);
38040                menuItem.on("beforecheckchange", onBeforeCheck);
38041            }
38042        },
38043
38044        // private
38045        unregisterCheckable : function(menuItem){
38046            var g = menuItem.group;
38047            if(g){
38048                groups[g].remove(menuItem);
38049                menuItem.un("beforecheckchange", onBeforeCheck);
38050            }
38051        }
38052    };
38053 }();/*
38054  * Based on:
38055  * Ext JS Library 1.1.1
38056  * Copyright(c) 2006-2007, Ext JS, LLC.
38057  *
38058  * Originally Released Under LGPL - original licence link has changed is not relivant.
38059  *
38060  * Fork - LGPL
38061  * <script type="text/javascript">
38062  */
38063  
38064
38065 /**
38066  * @class Roo.menu.BaseItem
38067  * @extends Roo.Component
38068  * The base class for all items that render into menus.  BaseItem provides default rendering, activated state
38069  * management and base configuration options shared by all menu components.
38070  * @constructor
38071  * Creates a new BaseItem
38072  * @param {Object} config Configuration options
38073  */
38074 Roo.menu.BaseItem = function(config){
38075     Roo.menu.BaseItem.superclass.constructor.call(this, config);
38076
38077     this.addEvents({
38078         /**
38079          * @event click
38080          * Fires when this item is clicked
38081          * @param {Roo.menu.BaseItem} this
38082          * @param {Roo.EventObject} e
38083          */
38084         click: true,
38085         /**
38086          * @event activate
38087          * Fires when this item is activated
38088          * @param {Roo.menu.BaseItem} this
38089          */
38090         activate : true,
38091         /**
38092          * @event deactivate
38093          * Fires when this item is deactivated
38094          * @param {Roo.menu.BaseItem} this
38095          */
38096         deactivate : true
38097     });
38098
38099     if(this.handler){
38100         this.on("click", this.handler, this.scope, true);
38101     }
38102 };
38103
38104 Roo.extend(Roo.menu.BaseItem, Roo.Component, {
38105     /**
38106      * @cfg {Function} handler
38107      * A function that will handle the click event of this menu item (defaults to undefined)
38108      */
38109     /**
38110      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to false)
38111      */
38112     canActivate : false,
38113     
38114      /**
38115      * @cfg {Boolean} hidden True to prevent creation of this menu item (defaults to false)
38116      */
38117     hidden: false,
38118     
38119     /**
38120      * @cfg {String} activeClass The CSS class to use when the item becomes activated (defaults to "x-menu-item-active")
38121      */
38122     activeClass : "x-menu-item-active",
38123     /**
38124      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to true)
38125      */
38126     hideOnClick : true,
38127     /**
38128      * @cfg {Number} hideDelay Length of time in milliseconds to wait before hiding after a click (defaults to 100)
38129      */
38130     hideDelay : 100,
38131
38132     // private
38133     ctype: "Roo.menu.BaseItem",
38134
38135     // private
38136     actionMode : "container",
38137
38138     // private
38139     render : function(container, parentMenu){
38140         this.parentMenu = parentMenu;
38141         Roo.menu.BaseItem.superclass.render.call(this, container);
38142         this.container.menuItemId = this.id;
38143     },
38144
38145     // private
38146     onRender : function(container, position){
38147         this.el = Roo.get(this.el);
38148         container.dom.appendChild(this.el.dom);
38149     },
38150
38151     // private
38152     onClick : function(e){
38153         if(!this.disabled && this.fireEvent("click", this, e) !== false
38154                 && this.parentMenu.fireEvent("itemclick", this, e) !== false){
38155             this.handleClick(e);
38156         }else{
38157             e.stopEvent();
38158         }
38159     },
38160
38161     // private
38162     activate : function(){
38163         if(this.disabled){
38164             return false;
38165         }
38166         var li = this.container;
38167         li.addClass(this.activeClass);
38168         this.region = li.getRegion().adjust(2, 2, -2, -2);
38169         this.fireEvent("activate", this);
38170         return true;
38171     },
38172
38173     // private
38174     deactivate : function(){
38175         this.container.removeClass(this.activeClass);
38176         this.fireEvent("deactivate", this);
38177     },
38178
38179     // private
38180     shouldDeactivate : function(e){
38181         return !this.region || !this.region.contains(e.getPoint());
38182     },
38183
38184     // private
38185     handleClick : function(e){
38186         if(this.hideOnClick){
38187             this.parentMenu.hide.defer(this.hideDelay, this.parentMenu, [true]);
38188         }
38189     },
38190
38191     // private
38192     expandMenu : function(autoActivate){
38193         // do nothing
38194     },
38195
38196     // private
38197     hideMenu : function(){
38198         // do nothing
38199     }
38200 });/*
38201  * Based on:
38202  * Ext JS Library 1.1.1
38203  * Copyright(c) 2006-2007, Ext JS, LLC.
38204  *
38205  * Originally Released Under LGPL - original licence link has changed is not relivant.
38206  *
38207  * Fork - LGPL
38208  * <script type="text/javascript">
38209  */
38210  
38211 /**
38212  * @class Roo.menu.Adapter
38213  * @extends Roo.menu.BaseItem
38214  * 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.
38215  * It provides basic rendering, activation management and enable/disable logic required to work in menus.
38216  * @constructor
38217  * Creates a new Adapter
38218  * @param {Object} config Configuration options
38219  */
38220 Roo.menu.Adapter = function(component, config){
38221     Roo.menu.Adapter.superclass.constructor.call(this, config);
38222     this.component = component;
38223 };
38224 Roo.extend(Roo.menu.Adapter, Roo.menu.BaseItem, {
38225     // private
38226     canActivate : true,
38227
38228     // private
38229     onRender : function(container, position){
38230         this.component.render(container);
38231         this.el = this.component.getEl();
38232     },
38233
38234     // private
38235     activate : function(){
38236         if(this.disabled){
38237             return false;
38238         }
38239         this.component.focus();
38240         this.fireEvent("activate", this);
38241         return true;
38242     },
38243
38244     // private
38245     deactivate : function(){
38246         this.fireEvent("deactivate", this);
38247     },
38248
38249     // private
38250     disable : function(){
38251         this.component.disable();
38252         Roo.menu.Adapter.superclass.disable.call(this);
38253     },
38254
38255     // private
38256     enable : function(){
38257         this.component.enable();
38258         Roo.menu.Adapter.superclass.enable.call(this);
38259     }
38260 });/*
38261  * Based on:
38262  * Ext JS Library 1.1.1
38263  * Copyright(c) 2006-2007, Ext JS, LLC.
38264  *
38265  * Originally Released Under LGPL - original licence link has changed is not relivant.
38266  *
38267  * Fork - LGPL
38268  * <script type="text/javascript">
38269  */
38270
38271 /**
38272  * @class Roo.menu.TextItem
38273  * @extends Roo.menu.BaseItem
38274  * Adds a static text string to a menu, usually used as either a heading or group separator.
38275  * Note: old style constructor with text is still supported.
38276  * 
38277  * @constructor
38278  * Creates a new TextItem
38279  * @param {Object} cfg Configuration
38280  */
38281 Roo.menu.TextItem = function(cfg){
38282     if (typeof(cfg) == 'string') {
38283         this.text = cfg;
38284     } else {
38285         Roo.apply(this,cfg);
38286     }
38287     
38288     Roo.menu.TextItem.superclass.constructor.call(this);
38289 };
38290
38291 Roo.extend(Roo.menu.TextItem, Roo.menu.BaseItem, {
38292     /**
38293      * @cfg {Boolean} text Text to show on item.
38294      */
38295     text : '',
38296     
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      * @cfg {String} itemCls The default CSS class to use for text items (defaults to "x-menu-text")
38303      */
38304     itemCls : "x-menu-text",
38305
38306     // private
38307     onRender : function(){
38308         var s = document.createElement("span");
38309         s.className = this.itemCls;
38310         s.innerHTML = this.text;
38311         this.el = s;
38312         Roo.menu.TextItem.superclass.onRender.apply(this, arguments);
38313     }
38314 });/*
38315  * Based on:
38316  * Ext JS Library 1.1.1
38317  * Copyright(c) 2006-2007, Ext JS, LLC.
38318  *
38319  * Originally Released Under LGPL - original licence link has changed is not relivant.
38320  *
38321  * Fork - LGPL
38322  * <script type="text/javascript">
38323  */
38324
38325 /**
38326  * @class Roo.menu.Separator
38327  * @extends Roo.menu.BaseItem
38328  * Adds a separator bar to a menu, used to divide logical groups of menu items. Generally you will
38329  * add one of these by using "-" in you call to add() or in your items config rather than creating one directly.
38330  * @constructor
38331  * @param {Object} config Configuration options
38332  */
38333 Roo.menu.Separator = function(config){
38334     Roo.menu.Separator.superclass.constructor.call(this, config);
38335 };
38336
38337 Roo.extend(Roo.menu.Separator, Roo.menu.BaseItem, {
38338     /**
38339      * @cfg {String} itemCls The default CSS class to use for separators (defaults to "x-menu-sep")
38340      */
38341     itemCls : "x-menu-sep",
38342     /**
38343      * @cfg {Boolean} hideOnClick True to hide the containing menu after this item is clicked (defaults to false)
38344      */
38345     hideOnClick : false,
38346
38347     // private
38348     onRender : function(li){
38349         var s = document.createElement("span");
38350         s.className = this.itemCls;
38351         s.innerHTML = "&#160;";
38352         this.el = s;
38353         li.addClass("x-menu-sep-li");
38354         Roo.menu.Separator.superclass.onRender.apply(this, arguments);
38355     }
38356 });/*
38357  * Based on:
38358  * Ext JS Library 1.1.1
38359  * Copyright(c) 2006-2007, Ext JS, LLC.
38360  *
38361  * Originally Released Under LGPL - original licence link has changed is not relivant.
38362  *
38363  * Fork - LGPL
38364  * <script type="text/javascript">
38365  */
38366 /**
38367  * @class Roo.menu.Item
38368  * @extends Roo.menu.BaseItem
38369  * A base class for all menu items that require menu-related functionality (like sub-menus) and are not static
38370  * display items.  Item extends the base functionality of {@link Roo.menu.BaseItem} by adding menu-specific
38371  * activation and click handling.
38372  * @constructor
38373  * Creates a new Item
38374  * @param {Object} config Configuration options
38375  */
38376 Roo.menu.Item = function(config){
38377     Roo.menu.Item.superclass.constructor.call(this, config);
38378     if(this.menu){
38379         this.menu = Roo.menu.MenuMgr.get(this.menu);
38380     }
38381 };
38382 Roo.extend(Roo.menu.Item, Roo.menu.BaseItem, {
38383     
38384     /**
38385      * @cfg {String} text
38386      * The text to show on the menu item.
38387      */
38388     text: '',
38389      /**
38390      * @cfg {String} HTML to render in menu
38391      * The text to show on the menu item (HTML version).
38392      */
38393     html: '',
38394     /**
38395      * @cfg {String} icon
38396      * The path to an icon to display in this menu item (defaults to Roo.BLANK_IMAGE_URL)
38397      */
38398     icon: undefined,
38399     /**
38400      * @cfg {String} itemCls The default CSS class to use for menu items (defaults to "x-menu-item")
38401      */
38402     itemCls : "x-menu-item",
38403     /**
38404      * @cfg {Boolean} canActivate True if this item can be visually activated (defaults to true)
38405      */
38406     canActivate : true,
38407     /**
38408      * @cfg {Number} showDelay Length of time in milliseconds to wait before showing this item (defaults to 200)
38409      */
38410     showDelay: 200,
38411     // doc'd in BaseItem
38412     hideDelay: 200,
38413
38414     // private
38415     ctype: "Roo.menu.Item",
38416     
38417     // private
38418     onRender : function(container, position){
38419         var el = document.createElement("a");
38420         el.hideFocus = true;
38421         el.unselectable = "on";
38422         el.href = this.href || "#";
38423         if(this.hrefTarget){
38424             el.target = this.hrefTarget;
38425         }
38426         el.className = this.itemCls + (this.menu ?  " x-menu-item-arrow" : "") + (this.cls ?  " " + this.cls : "");
38427         
38428         var html = this.html.length ? this.html  : String.format('{0}',this.text);
38429         
38430         el.innerHTML = String.format(
38431                 '<img src="{0}" class="x-menu-item-icon {1}" />' + html,
38432                 this.icon || Roo.BLANK_IMAGE_URL, this.iconCls || '');
38433         this.el = el;
38434         Roo.menu.Item.superclass.onRender.call(this, container, position);
38435     },
38436
38437     /**
38438      * Sets the text to display in this menu item
38439      * @param {String} text The text to display
38440      * @param {Boolean} isHTML true to indicate text is pure html.
38441      */
38442     setText : function(text, isHTML){
38443         if (isHTML) {
38444             this.html = text;
38445         } else {
38446             this.text = text;
38447             this.html = '';
38448         }
38449         if(this.rendered){
38450             var html = this.html.length ? this.html  : String.format('{0}',this.text);
38451      
38452             this.el.update(String.format(
38453                 '<img src="{0}" class="x-menu-item-icon {2}">' + html,
38454                 this.icon || Roo.BLANK_IMAGE_URL, this.text, this.iconCls || ''));
38455             this.parentMenu.autoWidth();
38456         }
38457     },
38458
38459     // private
38460     handleClick : function(e){
38461         if(!this.href){ // if no link defined, stop the event automatically
38462             e.stopEvent();
38463         }
38464         Roo.menu.Item.superclass.handleClick.apply(this, arguments);
38465     },
38466
38467     // private
38468     activate : function(autoExpand){
38469         if(Roo.menu.Item.superclass.activate.apply(this, arguments)){
38470             this.focus();
38471             if(autoExpand){
38472                 this.expandMenu();
38473             }
38474         }
38475         return true;
38476     },
38477
38478     // private
38479     shouldDeactivate : function(e){
38480         if(Roo.menu.Item.superclass.shouldDeactivate.call(this, e)){
38481             if(this.menu && this.menu.isVisible()){
38482                 return !this.menu.getEl().getRegion().contains(e.getPoint());
38483             }
38484             return true;
38485         }
38486         return false;
38487     },
38488
38489     // private
38490     deactivate : function(){
38491         Roo.menu.Item.superclass.deactivate.apply(this, arguments);
38492         this.hideMenu();
38493     },
38494
38495     // private
38496     expandMenu : function(autoActivate){
38497         if(!this.disabled && this.menu){
38498             clearTimeout(this.hideTimer);
38499             delete this.hideTimer;
38500             if(!this.menu.isVisible() && !this.showTimer){
38501                 this.showTimer = this.deferExpand.defer(this.showDelay, this, [autoActivate]);
38502             }else if (this.menu.isVisible() && autoActivate){
38503                 this.menu.tryActivate(0, 1);
38504             }
38505         }
38506     },
38507
38508     // private
38509     deferExpand : function(autoActivate){
38510         delete this.showTimer;
38511         this.menu.show(this.container, this.parentMenu.subMenuAlign || "tl-tr?", this.parentMenu);
38512         if(autoActivate){
38513             this.menu.tryActivate(0, 1);
38514         }
38515     },
38516
38517     // private
38518     hideMenu : function(){
38519         clearTimeout(this.showTimer);
38520         delete this.showTimer;
38521         if(!this.hideTimer && this.menu && this.menu.isVisible()){
38522             this.hideTimer = this.deferHide.defer(this.hideDelay, this);
38523         }
38524     },
38525
38526     // private
38527     deferHide : function(){
38528         delete this.hideTimer;
38529         this.menu.hide();
38530     }
38531 });/*
38532  * Based on:
38533  * Ext JS Library 1.1.1
38534  * Copyright(c) 2006-2007, Ext JS, LLC.
38535  *
38536  * Originally Released Under LGPL - original licence link has changed is not relivant.
38537  *
38538  * Fork - LGPL
38539  * <script type="text/javascript">
38540  */
38541  
38542 /**
38543  * @class Roo.menu.CheckItem
38544  * @extends Roo.menu.Item
38545  * Adds a menu item that contains a checkbox by default, but can also be part of a radio group.
38546  * @constructor
38547  * Creates a new CheckItem
38548  * @param {Object} config Configuration options
38549  */
38550 Roo.menu.CheckItem = function(config){
38551     Roo.menu.CheckItem.superclass.constructor.call(this, config);
38552     this.addEvents({
38553         /**
38554          * @event beforecheckchange
38555          * Fires before the checked value is set, providing an opportunity to cancel if needed
38556          * @param {Roo.menu.CheckItem} this
38557          * @param {Boolean} checked The new checked value that will be set
38558          */
38559         "beforecheckchange" : true,
38560         /**
38561          * @event checkchange
38562          * Fires after the checked value has been set
38563          * @param {Roo.menu.CheckItem} this
38564          * @param {Boolean} checked The checked value that was set
38565          */
38566         "checkchange" : true
38567     });
38568     if(this.checkHandler){
38569         this.on('checkchange', this.checkHandler, this.scope);
38570     }
38571 };
38572 Roo.extend(Roo.menu.CheckItem, Roo.menu.Item, {
38573     /**
38574      * @cfg {String} group
38575      * All check items with the same group name will automatically be grouped into a single-select
38576      * radio button group (defaults to '')
38577      */
38578     /**
38579      * @cfg {String} itemCls The default CSS class to use for check items (defaults to "x-menu-item x-menu-check-item")
38580      */
38581     itemCls : "x-menu-item x-menu-check-item",
38582     /**
38583      * @cfg {String} groupClass The default CSS class to use for radio group check items (defaults to "x-menu-group-item")
38584      */
38585     groupClass : "x-menu-group-item",
38586
38587     /**
38588      * @cfg {Boolean} checked True to initialize this checkbox as checked (defaults to false).  Note that
38589      * if this checkbox is part of a radio group (group = true) only the last item in the group that is
38590      * initialized with checked = true will be rendered as checked.
38591      */
38592     checked: false,
38593
38594     // private
38595     ctype: "Roo.menu.CheckItem",
38596
38597     // private
38598     onRender : function(c){
38599         Roo.menu.CheckItem.superclass.onRender.apply(this, arguments);
38600         if(this.group){
38601             this.el.addClass(this.groupClass);
38602         }
38603         Roo.menu.MenuMgr.registerCheckable(this);
38604         if(this.checked){
38605             this.checked = false;
38606             this.setChecked(true, true);
38607         }
38608     },
38609
38610     // private
38611     destroy : function(){
38612         if(this.rendered){
38613             Roo.menu.MenuMgr.unregisterCheckable(this);
38614         }
38615         Roo.menu.CheckItem.superclass.destroy.apply(this, arguments);
38616     },
38617
38618     /**
38619      * Set the checked state of this item
38620      * @param {Boolean} checked The new checked value
38621      * @param {Boolean} suppressEvent (optional) True to prevent the checkchange event from firing (defaults to false)
38622      */
38623     setChecked : function(state, suppressEvent){
38624         if(this.checked != state && this.fireEvent("beforecheckchange", this, state) !== false){
38625             if(this.container){
38626                 this.container[state ? "addClass" : "removeClass"]("x-menu-item-checked");
38627             }
38628             this.checked = state;
38629             if(suppressEvent !== true){
38630                 this.fireEvent("checkchange", this, state);
38631             }
38632         }
38633     },
38634
38635     // private
38636     handleClick : function(e){
38637        if(!this.disabled && !(this.checked && this.group)){// disable unselect on radio item
38638            this.setChecked(!this.checked);
38639        }
38640        Roo.menu.CheckItem.superclass.handleClick.apply(this, arguments);
38641     }
38642 });/*
38643  * Based on:
38644  * Ext JS Library 1.1.1
38645  * Copyright(c) 2006-2007, Ext JS, LLC.
38646  *
38647  * Originally Released Under LGPL - original licence link has changed is not relivant.
38648  *
38649  * Fork - LGPL
38650  * <script type="text/javascript">
38651  */
38652  
38653 /**
38654  * @class Roo.menu.DateItem
38655  * @extends Roo.menu.Adapter
38656  * A menu item that wraps the {@link Roo.DatPicker} component.
38657  * @constructor
38658  * Creates a new DateItem
38659  * @param {Object} config Configuration options
38660  */
38661 Roo.menu.DateItem = function(config){
38662     Roo.menu.DateItem.superclass.constructor.call(this, new Roo.DatePicker(config), config);
38663     /** The Roo.DatePicker object @type Roo.DatePicker */
38664     this.picker = this.component;
38665     this.addEvents({select: true});
38666     
38667     this.picker.on("render", function(picker){
38668         picker.getEl().swallowEvent("click");
38669         picker.container.addClass("x-menu-date-item");
38670     });
38671
38672     this.picker.on("select", this.onSelect, this);
38673 };
38674
38675 Roo.extend(Roo.menu.DateItem, Roo.menu.Adapter, {
38676     // private
38677     onSelect : function(picker, date){
38678         this.fireEvent("select", this, date, picker);
38679         Roo.menu.DateItem.superclass.handleClick.call(this);
38680     }
38681 });/*
38682  * Based on:
38683  * Ext JS Library 1.1.1
38684  * Copyright(c) 2006-2007, Ext JS, LLC.
38685  *
38686  * Originally Released Under LGPL - original licence link has changed is not relivant.
38687  *
38688  * Fork - LGPL
38689  * <script type="text/javascript">
38690  */
38691  
38692 /**
38693  * @class Roo.menu.ColorItem
38694  * @extends Roo.menu.Adapter
38695  * A menu item that wraps the {@link Roo.ColorPalette} component.
38696  * @constructor
38697  * Creates a new ColorItem
38698  * @param {Object} config Configuration options
38699  */
38700 Roo.menu.ColorItem = function(config){
38701     Roo.menu.ColorItem.superclass.constructor.call(this, new Roo.ColorPalette(config), config);
38702     /** The Roo.ColorPalette object @type Roo.ColorPalette */
38703     this.palette = this.component;
38704     this.relayEvents(this.palette, ["select"]);
38705     if(this.selectHandler){
38706         this.on('select', this.selectHandler, this.scope);
38707     }
38708 };
38709 Roo.extend(Roo.menu.ColorItem, Roo.menu.Adapter);/*
38710  * Based on:
38711  * Ext JS Library 1.1.1
38712  * Copyright(c) 2006-2007, Ext JS, LLC.
38713  *
38714  * Originally Released Under LGPL - original licence link has changed is not relivant.
38715  *
38716  * Fork - LGPL
38717  * <script type="text/javascript">
38718  */
38719  
38720
38721 /**
38722  * @class Roo.menu.DateMenu
38723  * @extends Roo.menu.Menu
38724  * A menu containing a {@link Roo.menu.DateItem} component (which provides a date picker).
38725  * @constructor
38726  * Creates a new DateMenu
38727  * @param {Object} config Configuration options
38728  */
38729 Roo.menu.DateMenu = function(config){
38730     Roo.menu.DateMenu.superclass.constructor.call(this, config);
38731     this.plain = true;
38732     var di = new Roo.menu.DateItem(config);
38733     this.add(di);
38734     /**
38735      * The {@link Roo.DatePicker} instance for this DateMenu
38736      * @type DatePicker
38737      */
38738     this.picker = di.picker;
38739     /**
38740      * @event select
38741      * @param {DatePicker} picker
38742      * @param {Date} date
38743      */
38744     this.relayEvents(di, ["select"]);
38745     this.on('beforeshow', function(){
38746         if(this.picker){
38747             this.picker.hideMonthPicker(false);
38748         }
38749     }, this);
38750 };
38751 Roo.extend(Roo.menu.DateMenu, Roo.menu.Menu, {
38752     cls:'x-date-menu'
38753 });/*
38754  * Based on:
38755  * Ext JS Library 1.1.1
38756  * Copyright(c) 2006-2007, Ext JS, LLC.
38757  *
38758  * Originally Released Under LGPL - original licence link has changed is not relivant.
38759  *
38760  * Fork - LGPL
38761  * <script type="text/javascript">
38762  */
38763  
38764
38765 /**
38766  * @class Roo.menu.ColorMenu
38767  * @extends Roo.menu.Menu
38768  * A menu containing a {@link Roo.menu.ColorItem} component (which provides a basic color picker).
38769  * @constructor
38770  * Creates a new ColorMenu
38771  * @param {Object} config Configuration options
38772  */
38773 Roo.menu.ColorMenu = function(config){
38774     Roo.menu.ColorMenu.superclass.constructor.call(this, config);
38775     this.plain = true;
38776     var ci = new Roo.menu.ColorItem(config);
38777     this.add(ci);
38778     /**
38779      * The {@link Roo.ColorPalette} instance for this ColorMenu
38780      * @type ColorPalette
38781      */
38782     this.palette = ci.palette;
38783     /**
38784      * @event select
38785      * @param {ColorPalette} palette
38786      * @param {String} color
38787      */
38788     this.relayEvents(ci, ["select"]);
38789 };
38790 Roo.extend(Roo.menu.ColorMenu, Roo.menu.Menu);/*
38791  * Based on:
38792  * Ext JS Library 1.1.1
38793  * Copyright(c) 2006-2007, Ext JS, LLC.
38794  *
38795  * Originally Released Under LGPL - original licence link has changed is not relivant.
38796  *
38797  * Fork - LGPL
38798  * <script type="text/javascript">
38799  */
38800  
38801 /**
38802  * @class Roo.form.TextItem
38803  * @extends Roo.BoxComponent
38804  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38805  * @constructor
38806  * Creates a new TextItem
38807  * @param {Object} config Configuration options
38808  */
38809 Roo.form.TextItem = function(config){
38810     Roo.form.TextItem.superclass.constructor.call(this, config);
38811 };
38812
38813 Roo.extend(Roo.form.TextItem, Roo.BoxComponent,  {
38814     
38815     /**
38816      * @cfg {String} tag the tag for this item (default div)
38817      */
38818     tag : 'div',
38819     /**
38820      * @cfg {String} html the content for this item
38821      */
38822     html : '',
38823     
38824     getAutoCreate : function()
38825     {
38826         var cfg = {
38827             id: this.id,
38828             tag: this.tag,
38829             html: this.html,
38830             cls: 'x-form-item'
38831         };
38832         
38833         return cfg;
38834         
38835     },
38836     
38837     onRender : function(ct, position)
38838     {
38839         Roo.form.TextItem.superclass.onRender.call(this, ct, position);
38840         
38841         if(!this.el){
38842             var cfg = this.getAutoCreate();
38843             if(!cfg.name){
38844                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
38845             }
38846             if (!cfg.name.length) {
38847                 delete cfg.name;
38848             }
38849             this.el = ct.createChild(cfg, position);
38850         }
38851     }
38852     
38853 });/*
38854  * Based on:
38855  * Ext JS Library 1.1.1
38856  * Copyright(c) 2006-2007, Ext JS, LLC.
38857  *
38858  * Originally Released Under LGPL - original licence link has changed is not relivant.
38859  *
38860  * Fork - LGPL
38861  * <script type="text/javascript">
38862  */
38863  
38864 /**
38865  * @class Roo.form.Field
38866  * @extends Roo.BoxComponent
38867  * Base class for form fields that provides default event handling, sizing, value handling and other functionality.
38868  * @constructor
38869  * Creates a new Field
38870  * @param {Object} config Configuration options
38871  */
38872 Roo.form.Field = function(config){
38873     Roo.form.Field.superclass.constructor.call(this, config);
38874 };
38875
38876 Roo.extend(Roo.form.Field, Roo.BoxComponent,  {
38877     /**
38878      * @cfg {String} fieldLabel Label to use when rendering a form.
38879      */
38880        /**
38881      * @cfg {String} qtip Mouse over tip
38882      */
38883      
38884     /**
38885      * @cfg {String} invalidClass The CSS class to use when marking a field invalid (defaults to "x-form-invalid")
38886      */
38887     invalidClass : "x-form-invalid",
38888     /**
38889      * @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")
38890      */
38891     invalidText : "The value in this field is invalid",
38892     /**
38893      * @cfg {String} focusClass The CSS class to use when the field receives focus (defaults to "x-form-focus")
38894      */
38895     focusClass : "x-form-focus",
38896     /**
38897      * @cfg {String/Boolean} validationEvent The event that should initiate field validation. Set to false to disable
38898       automatic validation (defaults to "keyup").
38899      */
38900     validationEvent : "keyup",
38901     /**
38902      * @cfg {Boolean} validateOnBlur Whether the field should validate when it loses focus (defaults to true).
38903      */
38904     validateOnBlur : true,
38905     /**
38906      * @cfg {Number} validationDelay The length of time in milliseconds after user input begins until validation is initiated (defaults to 250)
38907      */
38908     validationDelay : 250,
38909     /**
38910      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
38911      * {tag: "input", type: "text", size: "20", autocomplete: "off"})
38912      */
38913     defaultAutoCreate : {tag: "input", type: "text", size: "20", autocomplete: "new-password"},
38914     /**
38915      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field")
38916      */
38917     fieldClass : "x-form-field",
38918     /**
38919      * @cfg {String} msgTarget The location where error text should display.  Should be one of the following values (defaults to 'qtip'):
38920      *<pre>
38921 Value         Description
38922 -----------   ----------------------------------------------------------------------
38923 qtip          Display a quick tip when the user hovers over the field
38924 title         Display a default browser title attribute popup
38925 under         Add a block div beneath the field containing the error text
38926 side          Add an error icon to the right of the field with a popup on hover
38927 [element id]  Add the error text directly to the innerHTML of the specified element
38928 </pre>
38929      */
38930     msgTarget : 'qtip',
38931     /**
38932      * @cfg {String} msgFx <b>Experimental</b> The effect used when displaying a validation message under the field (defaults to 'normal').
38933      */
38934     msgFx : 'normal',
38935
38936     /**
38937      * @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.
38938      */
38939     readOnly : false,
38940
38941     /**
38942      * @cfg {Boolean} disabled True to disable the field (defaults to false).
38943      */
38944     disabled : false,
38945
38946     /**
38947      * @cfg {String} inputType The type attribute for input fields -- e.g. radio, text, password (defaults to "text").
38948      */
38949     inputType : undefined,
38950     
38951     /**
38952      * @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).
38953          */
38954         tabIndex : undefined,
38955         
38956     // private
38957     isFormField : true,
38958
38959     // private
38960     hasFocus : false,
38961     /**
38962      * @property {Roo.Element} fieldEl
38963      * Element Containing the rendered Field (with label etc.)
38964      */
38965     /**
38966      * @cfg {Mixed} value A value to initialize this field with.
38967      */
38968     value : undefined,
38969
38970     /**
38971      * @cfg {String} name The field's HTML name attribute.
38972      */
38973     /**
38974      * @cfg {String} cls A CSS class to apply to the field's underlying element.
38975      */
38976     // private
38977     loadedValue : false,
38978      
38979      
38980         // private ??
38981         initComponent : function(){
38982         Roo.form.Field.superclass.initComponent.call(this);
38983         this.addEvents({
38984             /**
38985              * @event focus
38986              * Fires when this field receives input focus.
38987              * @param {Roo.form.Field} this
38988              */
38989             focus : true,
38990             /**
38991              * @event blur
38992              * Fires when this field loses input focus.
38993              * @param {Roo.form.Field} this
38994              */
38995             blur : true,
38996             /**
38997              * @event specialkey
38998              * Fires when any key related to navigation (arrows, tab, enter, esc, etc.) is pressed.  You can check
38999              * {@link Roo.EventObject#getKey} to determine which key was pressed.
39000              * @param {Roo.form.Field} this
39001              * @param {Roo.EventObject} e The event object
39002              */
39003             specialkey : true,
39004             /**
39005              * @event change
39006              * Fires just before the field blurs if the field value has changed.
39007              * @param {Roo.form.Field} this
39008              * @param {Mixed} newValue The new value
39009              * @param {Mixed} oldValue The original value
39010              */
39011             change : true,
39012             /**
39013              * @event invalid
39014              * Fires after the field has been marked as invalid.
39015              * @param {Roo.form.Field} this
39016              * @param {String} msg The validation message
39017              */
39018             invalid : true,
39019             /**
39020              * @event valid
39021              * Fires after the field has been validated with no errors.
39022              * @param {Roo.form.Field} this
39023              */
39024             valid : true,
39025              /**
39026              * @event keyup
39027              * Fires after the key up
39028              * @param {Roo.form.Field} this
39029              * @param {Roo.EventObject}  e The event Object
39030              */
39031             keyup : true
39032         });
39033     },
39034
39035     /**
39036      * Returns the name attribute of the field if available
39037      * @return {String} name The field name
39038      */
39039     getName: function(){
39040          return this.rendered && this.el.dom.name ? this.el.dom.name : (this.hiddenName || '');
39041     },
39042
39043     // private
39044     onRender : function(ct, position){
39045         Roo.form.Field.superclass.onRender.call(this, ct, position);
39046         if(!this.el){
39047             var cfg = this.getAutoCreate();
39048             if(!cfg.name){
39049                 cfg.name = typeof(this.name) == 'undefined' ? this.id : this.name;
39050             }
39051             if (!cfg.name.length) {
39052                 delete cfg.name;
39053             }
39054             if(this.inputType){
39055                 cfg.type = this.inputType;
39056             }
39057             this.el = ct.createChild(cfg, position);
39058         }
39059         var type = this.el.dom.type;
39060         if(type){
39061             if(type == 'password'){
39062                 type = 'text';
39063             }
39064             this.el.addClass('x-form-'+type);
39065         }
39066         if(this.readOnly){
39067             this.el.dom.readOnly = true;
39068         }
39069         if(this.tabIndex !== undefined){
39070             this.el.dom.setAttribute('tabIndex', this.tabIndex);
39071         }
39072
39073         this.el.addClass([this.fieldClass, this.cls]);
39074         this.initValue();
39075     },
39076
39077     /**
39078      * Apply the behaviors of this component to an existing element. <b>This is used instead of render().</b>
39079      * @param {String/HTMLElement/Element} el The id of the node, a DOM node or an existing Element
39080      * @return {Roo.form.Field} this
39081      */
39082     applyTo : function(target){
39083         this.allowDomMove = false;
39084         this.el = Roo.get(target);
39085         this.render(this.el.dom.parentNode);
39086         return this;
39087     },
39088
39089     // private
39090     initValue : function(){
39091         if(this.value !== undefined){
39092             this.setValue(this.value);
39093         }else if(this.el.dom.value.length > 0){
39094             this.setValue(this.el.dom.value);
39095         }
39096     },
39097
39098     /**
39099      * Returns true if this field has been changed since it was originally loaded and is not disabled.
39100      * DEPRICATED  - it never worked well - use hasChanged/resetHasChanged.
39101      */
39102     isDirty : function() {
39103         if(this.disabled) {
39104             return false;
39105         }
39106         return String(this.getValue()) !== String(this.originalValue);
39107     },
39108
39109     /**
39110      * stores the current value in loadedValue
39111      */
39112     resetHasChanged : function()
39113     {
39114         this.loadedValue = String(this.getValue());
39115     },
39116     /**
39117      * checks the current value against the 'loaded' value.
39118      * Note - will return false if 'resetHasChanged' has not been called first.
39119      */
39120     hasChanged : function()
39121     {
39122         if(this.disabled || this.readOnly) {
39123             return false;
39124         }
39125         return this.loadedValue !== false && String(this.getValue()) !== this.loadedValue;
39126     },
39127     
39128     
39129     
39130     // private
39131     afterRender : function(){
39132         Roo.form.Field.superclass.afterRender.call(this);
39133         this.initEvents();
39134     },
39135
39136     // private
39137     fireKey : function(e){
39138         //Roo.log('field ' + e.getKey());
39139         if(e.isNavKeyPress()){
39140             this.fireEvent("specialkey", this, e);
39141         }
39142     },
39143
39144     /**
39145      * Resets the current field value to the originally loaded value and clears any validation messages
39146      */
39147     reset : function(){
39148         this.setValue(this.resetValue);
39149         this.originalValue = this.getValue();
39150         this.clearInvalid();
39151     },
39152
39153     // private
39154     initEvents : function(){
39155         // safari killled keypress - so keydown is now used..
39156         this.el.on("keydown" , this.fireKey,  this);
39157         this.el.on("focus", this.onFocus,  this);
39158         this.el.on("blur", this.onBlur,  this);
39159         this.el.relayEvent('keyup', this);
39160
39161         // reference to original value for reset
39162         this.originalValue = this.getValue();
39163         this.resetValue =  this.getValue();
39164     },
39165
39166     // private
39167     onFocus : function(){
39168         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39169             this.el.addClass(this.focusClass);
39170         }
39171         if(!this.hasFocus){
39172             this.hasFocus = true;
39173             this.startValue = this.getValue();
39174             this.fireEvent("focus", this);
39175         }
39176     },
39177
39178     beforeBlur : Roo.emptyFn,
39179
39180     // private
39181     onBlur : function(){
39182         this.beforeBlur();
39183         if(!Roo.isOpera && this.focusClass){ // don't touch in Opera
39184             this.el.removeClass(this.focusClass);
39185         }
39186         this.hasFocus = false;
39187         if(this.validationEvent !== false && this.validateOnBlur && this.validationEvent != "blur"){
39188             this.validate();
39189         }
39190         var v = this.getValue();
39191         if(String(v) !== String(this.startValue)){
39192             this.fireEvent('change', this, v, this.startValue);
39193         }
39194         this.fireEvent("blur", this);
39195     },
39196
39197     /**
39198      * Returns whether or not the field value is currently valid
39199      * @param {Boolean} preventMark True to disable marking the field invalid
39200      * @return {Boolean} True if the value is valid, else false
39201      */
39202     isValid : function(preventMark){
39203         if(this.disabled){
39204             return true;
39205         }
39206         var restore = this.preventMark;
39207         this.preventMark = preventMark === true;
39208         var v = this.validateValue(this.processValue(this.getRawValue()));
39209         this.preventMark = restore;
39210         return v;
39211     },
39212
39213     /**
39214      * Validates the field value
39215      * @return {Boolean} True if the value is valid, else false
39216      */
39217     validate : function(){
39218         if(this.disabled || this.validateValue(this.processValue(this.getRawValue()))){
39219             this.clearInvalid();
39220             return true;
39221         }
39222         return false;
39223     },
39224
39225     processValue : function(value){
39226         return value;
39227     },
39228
39229     // private
39230     // Subclasses should provide the validation implementation by overriding this
39231     validateValue : function(value){
39232         return true;
39233     },
39234
39235     /**
39236      * Mark this field as invalid
39237      * @param {String} msg The validation message
39238      */
39239     markInvalid : function(msg){
39240         if(!this.rendered || this.preventMark){ // not rendered
39241             return;
39242         }
39243         
39244         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39245         
39246         obj.el.addClass(this.invalidClass);
39247         msg = msg || this.invalidText;
39248         switch(this.msgTarget){
39249             case 'qtip':
39250                 obj.el.dom.qtip = msg;
39251                 obj.el.dom.qclass = 'x-form-invalid-tip';
39252                 if(Roo.QuickTips){ // fix for floating editors interacting with DND
39253                     Roo.QuickTips.enable();
39254                 }
39255                 break;
39256             case 'title':
39257                 this.el.dom.title = msg;
39258                 break;
39259             case 'under':
39260                 if(!this.errorEl){
39261                     var elp = this.el.findParent('.x-form-element', 5, true);
39262                     this.errorEl = elp.createChild({cls:'x-form-invalid-msg'});
39263                     this.errorEl.setWidth(elp.getWidth(true)-20);
39264                 }
39265                 this.errorEl.update(msg);
39266                 Roo.form.Field.msgFx[this.msgFx].show(this.errorEl, this);
39267                 break;
39268             case 'side':
39269                 if(!this.errorIcon){
39270                     var elp = this.el.findParent('.x-form-element', 5, true);
39271                     this.errorIcon = elp.createChild({cls:'x-form-invalid-icon'});
39272                 }
39273                 this.alignErrorIcon();
39274                 this.errorIcon.dom.qtip = msg;
39275                 this.errorIcon.dom.qclass = 'x-form-invalid-tip';
39276                 this.errorIcon.show();
39277                 this.on('resize', this.alignErrorIcon, this);
39278                 break;
39279             default:
39280                 var t = Roo.getDom(this.msgTarget);
39281                 t.innerHTML = msg;
39282                 t.style.display = this.msgDisplay;
39283                 break;
39284         }
39285         this.fireEvent('invalid', this, msg);
39286     },
39287
39288     // private
39289     alignErrorIcon : function(){
39290         this.errorIcon.alignTo(this.el, 'tl-tr', [2, 0]);
39291     },
39292
39293     /**
39294      * Clear any invalid styles/messages for this field
39295      */
39296     clearInvalid : function(){
39297         if(!this.rendered || this.preventMark){ // not rendered
39298             return;
39299         }
39300         var obj = (typeof(this.combo) != 'undefined') ? this.combo : this; // fix the combox array!!
39301         
39302         obj.el.removeClass(this.invalidClass);
39303         switch(this.msgTarget){
39304             case 'qtip':
39305                 obj.el.dom.qtip = '';
39306                 break;
39307             case 'title':
39308                 this.el.dom.title = '';
39309                 break;
39310             case 'under':
39311                 if(this.errorEl){
39312                     Roo.form.Field.msgFx[this.msgFx].hide(this.errorEl, this);
39313                 }
39314                 break;
39315             case 'side':
39316                 if(this.errorIcon){
39317                     this.errorIcon.dom.qtip = '';
39318                     this.errorIcon.hide();
39319                     this.un('resize', this.alignErrorIcon, this);
39320                 }
39321                 break;
39322             default:
39323                 var t = Roo.getDom(this.msgTarget);
39324                 t.innerHTML = '';
39325                 t.style.display = 'none';
39326                 break;
39327         }
39328         this.fireEvent('valid', this);
39329     },
39330
39331     /**
39332      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
39333      * @return {Mixed} value The field value
39334      */
39335     getRawValue : function(){
39336         var v = this.el.getValue();
39337         
39338         return v;
39339     },
39340
39341     /**
39342      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
39343      * @return {Mixed} value The field value
39344      */
39345     getValue : function(){
39346         var v = this.el.getValue();
39347          
39348         return v;
39349     },
39350
39351     /**
39352      * Sets the underlying DOM field's value directly, bypassing validation.  To set the value with validation see {@link #setValue}.
39353      * @param {Mixed} value The value to set
39354      */
39355     setRawValue : function(v){
39356         return this.el.dom.value = (v === null || v === undefined ? '' : v);
39357     },
39358
39359     /**
39360      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
39361      * @param {Mixed} value The value to set
39362      */
39363     setValue : function(v){
39364         this.value = v;
39365         if(this.rendered){
39366             this.el.dom.value = (v === null || v === undefined ? '' : v);
39367              this.validate();
39368         }
39369     },
39370
39371     adjustSize : function(w, h){
39372         var s = Roo.form.Field.superclass.adjustSize.call(this, w, h);
39373         s.width = this.adjustWidth(this.el.dom.tagName, s.width);
39374         return s;
39375     },
39376
39377     adjustWidth : function(tag, w){
39378         tag = tag.toLowerCase();
39379         if(typeof w == 'number' && Roo.isStrict && !Roo.isSafari){
39380             if(Roo.isIE && (tag == 'input' || tag == 'textarea')){
39381                 if(tag == 'input'){
39382                     return w + 2;
39383                 }
39384                 if(tag == 'textarea'){
39385                     return w-2;
39386                 }
39387             }else if(Roo.isOpera){
39388                 if(tag == 'input'){
39389                     return w + 2;
39390                 }
39391                 if(tag == 'textarea'){
39392                     return w-2;
39393                 }
39394             }
39395         }
39396         return w;
39397     }
39398 });
39399
39400
39401 // anything other than normal should be considered experimental
39402 Roo.form.Field.msgFx = {
39403     normal : {
39404         show: function(msgEl, f){
39405             msgEl.setDisplayed('block');
39406         },
39407
39408         hide : function(msgEl, f){
39409             msgEl.setDisplayed(false).update('');
39410         }
39411     },
39412
39413     slide : {
39414         show: function(msgEl, f){
39415             msgEl.slideIn('t', {stopFx:true});
39416         },
39417
39418         hide : function(msgEl, f){
39419             msgEl.slideOut('t', {stopFx:true,useDisplay:true});
39420         }
39421     },
39422
39423     slideRight : {
39424         show: function(msgEl, f){
39425             msgEl.fixDisplay();
39426             msgEl.alignTo(f.el, 'tl-tr');
39427             msgEl.slideIn('l', {stopFx:true});
39428         },
39429
39430         hide : function(msgEl, f){
39431             msgEl.slideOut('l', {stopFx:true,useDisplay:true});
39432         }
39433     }
39434 };/*
39435  * Based on:
39436  * Ext JS Library 1.1.1
39437  * Copyright(c) 2006-2007, Ext JS, LLC.
39438  *
39439  * Originally Released Under LGPL - original licence link has changed is not relivant.
39440  *
39441  * Fork - LGPL
39442  * <script type="text/javascript">
39443  */
39444  
39445
39446 /**
39447  * @class Roo.form.TextField
39448  * @extends Roo.form.Field
39449  * Basic text field.  Can be used as a direct replacement for traditional text inputs, or as the base
39450  * class for more sophisticated input controls (like {@link Roo.form.TextArea} and {@link Roo.form.ComboBox}).
39451  * @constructor
39452  * Creates a new TextField
39453  * @param {Object} config Configuration options
39454  */
39455 Roo.form.TextField = function(config){
39456     Roo.form.TextField.superclass.constructor.call(this, config);
39457     this.addEvents({
39458         /**
39459          * @event autosize
39460          * Fires when the autosize function is triggered.  The field may or may not have actually changed size
39461          * according to the default logic, but this event provides a hook for the developer to apply additional
39462          * logic at runtime to resize the field if needed.
39463              * @param {Roo.form.Field} this This text field
39464              * @param {Number} width The new field width
39465              */
39466         autosize : true
39467     });
39468 };
39469
39470 Roo.extend(Roo.form.TextField, Roo.form.Field,  {
39471     /**
39472      * @cfg {Boolean} grow True if this field should automatically grow and shrink to its content
39473      */
39474     grow : false,
39475     /**
39476      * @cfg {Number} growMin The minimum width to allow when grow = true (defaults to 30)
39477      */
39478     growMin : 30,
39479     /**
39480      * @cfg {Number} growMax The maximum width to allow when grow = true (defaults to 800)
39481      */
39482     growMax : 800,
39483     /**
39484      * @cfg {String} vtype A validation type name as defined in {@link Roo.form.VTypes} (defaults to null)
39485      */
39486     vtype : null,
39487     /**
39488      * @cfg {String} maskRe An input mask regular expression that will be used to filter keystrokes that don't match (defaults to null)
39489      */
39490     maskRe : null,
39491     /**
39492      * @cfg {Boolean} disableKeyFilter True to disable input keystroke filtering (defaults to false)
39493      */
39494     disableKeyFilter : false,
39495     /**
39496      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to true)
39497      */
39498     allowBlank : true,
39499     /**
39500      * @cfg {Number} minLength Minimum input field length required (defaults to 0)
39501      */
39502     minLength : 0,
39503     /**
39504      * @cfg {Number} maxLength Maximum input field length allowed (defaults to Number.MAX_VALUE)
39505      */
39506     maxLength : Number.MAX_VALUE,
39507     /**
39508      * @cfg {String} minLengthText Error text to display if the minimum length validation fails (defaults to "The minimum length for this field is {minLength}")
39509      */
39510     minLengthText : "The minimum length for this field is {0}",
39511     /**
39512      * @cfg {String} maxLengthText Error text to display if the maximum length validation fails (defaults to "The maximum length for this field is {maxLength}")
39513      */
39514     maxLengthText : "The maximum length for this field is {0}",
39515     /**
39516      * @cfg {Boolean} selectOnFocus True to automatically select any existing field text when the field receives input focus (defaults to false)
39517      */
39518     selectOnFocus : false,
39519     /**
39520      * @cfg {Boolean} allowLeadingSpace True to prevent the stripping of leading white space 
39521      */    
39522     allowLeadingSpace : false,
39523     /**
39524      * @cfg {String} blankText Error text to display if the allow blank validation fails (defaults to "This field is required")
39525      */
39526     blankText : "This field is required",
39527     /**
39528      * @cfg {Function} validator A custom validation function to be called during field validation (defaults to null).
39529      * If available, this function will be called only after the basic validators all return true, and will be passed the
39530      * current field value and expected to return boolean true if the value is valid or a string error message if invalid.
39531      */
39532     validator : null,
39533     /**
39534      * @cfg {RegExp} regex A JavaScript RegExp object to be tested against the field value during validation (defaults to null).
39535      * If available, this regex will be evaluated only after the basic validators all return true, and will be passed the
39536      * current field value.  If the test fails, the field will be marked invalid using {@link #regexText}.
39537      */
39538     regex : null,
39539     /**
39540      * @cfg {String} regexText The error text to display if {@link #regex} is used and the test fails during validation (defaults to "")
39541      */
39542     regexText : "",
39543     /**
39544      * @cfg {String} emptyText The default text to display in an empty field - placeholder... (defaults to null).
39545      */
39546     emptyText : null,
39547    
39548
39549     // private
39550     initEvents : function()
39551     {
39552         if (this.emptyText) {
39553             this.el.attr('placeholder', this.emptyText);
39554         }
39555         
39556         Roo.form.TextField.superclass.initEvents.call(this);
39557         if(this.validationEvent == 'keyup'){
39558             this.validationTask = new Roo.util.DelayedTask(this.validate, this);
39559             this.el.on('keyup', this.filterValidation, this);
39560         }
39561         else if(this.validationEvent !== false){
39562             this.el.on(this.validationEvent, this.validate, this, {buffer: this.validationDelay});
39563         }
39564         
39565         if(this.selectOnFocus){
39566             this.on("focus", this.preFocus, this);
39567         }
39568         if (!this.allowLeadingSpace) {
39569             this.on('blur', this.cleanLeadingSpace, this);
39570         }
39571         
39572         if(this.maskRe || (this.vtype && this.disableKeyFilter !== true && (this.maskRe = Roo.form.VTypes[this.vtype+'Mask']))){
39573             this.el.on("keypress", this.filterKeys, this);
39574         }
39575         if(this.grow){
39576             this.el.on("keyup", this.onKeyUp,  this, {buffer:50});
39577             this.el.on("click", this.autoSize,  this);
39578         }
39579         if(this.el.is('input[type=password]') && Roo.isSafari){
39580             this.el.on('keydown', this.SafariOnKeyDown, this);
39581         }
39582     },
39583
39584     processValue : function(value){
39585         if(this.stripCharsRe){
39586             var newValue = value.replace(this.stripCharsRe, '');
39587             if(newValue !== value){
39588                 this.setRawValue(newValue);
39589                 return newValue;
39590             }
39591         }
39592         return value;
39593     },
39594
39595     filterValidation : function(e){
39596         if(!e.isNavKeyPress()){
39597             this.validationTask.delay(this.validationDelay);
39598         }
39599     },
39600
39601     // private
39602     onKeyUp : function(e){
39603         if(!e.isNavKeyPress()){
39604             this.autoSize();
39605         }
39606     },
39607     // private - clean the leading white space
39608     cleanLeadingSpace : function(e)
39609     {
39610         if ( this.inputType == 'file') {
39611             return;
39612         }
39613         
39614         this.setValue((this.getValue() + '').replace(/^\s+/,''));
39615     },
39616     /**
39617      * Resets the current field value to the originally-loaded value and clears any validation messages.
39618      *  
39619      */
39620     reset : function(){
39621         Roo.form.TextField.superclass.reset.call(this);
39622        
39623     }, 
39624     // private
39625     preFocus : function(){
39626         
39627         if(this.selectOnFocus){
39628             this.el.dom.select();
39629         }
39630     },
39631
39632     
39633     // private
39634     filterKeys : function(e){
39635         var k = e.getKey();
39636         if(!Roo.isIE && (e.isNavKeyPress() || k == e.BACKSPACE || (k == e.DELETE && e.button == -1))){
39637             return;
39638         }
39639         var c = e.getCharCode(), cc = String.fromCharCode(c);
39640         if(Roo.isIE && (e.isSpecialKey() || !cc)){
39641             return;
39642         }
39643         if(!this.maskRe.test(cc)){
39644             e.stopEvent();
39645         }
39646     },
39647
39648     setValue : function(v){
39649         
39650         Roo.form.TextField.superclass.setValue.apply(this, arguments);
39651         
39652         this.autoSize();
39653     },
39654
39655     /**
39656      * Validates a value according to the field's validation rules and marks the field as invalid
39657      * if the validation fails
39658      * @param {Mixed} value The value to validate
39659      * @return {Boolean} True if the value is valid, else false
39660      */
39661     validateValue : function(value){
39662         if(value.length < 1)  { // if it's blank
39663              if(this.allowBlank){
39664                 this.clearInvalid();
39665                 return true;
39666              }else{
39667                 this.markInvalid(this.blankText);
39668                 return false;
39669              }
39670         }
39671         if(value.length < this.minLength){
39672             this.markInvalid(String.format(this.minLengthText, this.minLength));
39673             return false;
39674         }
39675         if(value.length > this.maxLength){
39676             this.markInvalid(String.format(this.maxLengthText, this.maxLength));
39677             return false;
39678         }
39679         if(this.vtype){
39680             var vt = Roo.form.VTypes;
39681             if(!vt[this.vtype](value, this)){
39682                 this.markInvalid(this.vtypeText || vt[this.vtype +'Text']);
39683                 return false;
39684             }
39685         }
39686         if(typeof this.validator == "function"){
39687             var msg = this.validator(value);
39688             if(msg !== true){
39689                 this.markInvalid(msg);
39690                 return false;
39691             }
39692         }
39693         if(this.regex && !this.regex.test(value)){
39694             this.markInvalid(this.regexText);
39695             return false;
39696         }
39697         return true;
39698     },
39699
39700     /**
39701      * Selects text in this field
39702      * @param {Number} start (optional) The index where the selection should start (defaults to 0)
39703      * @param {Number} end (optional) The index where the selection should end (defaults to the text length)
39704      */
39705     selectText : function(start, end){
39706         var v = this.getRawValue();
39707         if(v.length > 0){
39708             start = start === undefined ? 0 : start;
39709             end = end === undefined ? v.length : end;
39710             var d = this.el.dom;
39711             if(d.setSelectionRange){
39712                 d.setSelectionRange(start, end);
39713             }else if(d.createTextRange){
39714                 var range = d.createTextRange();
39715                 range.moveStart("character", start);
39716                 range.moveEnd("character", v.length-end);
39717                 range.select();
39718             }
39719         }
39720     },
39721
39722     /**
39723      * Automatically grows the field to accomodate the width of the text up to the maximum field width allowed.
39724      * This only takes effect if grow = true, and fires the autosize event.
39725      */
39726     autoSize : function(){
39727         if(!this.grow || !this.rendered){
39728             return;
39729         }
39730         if(!this.metrics){
39731             this.metrics = Roo.util.TextMetrics.createInstance(this.el);
39732         }
39733         var el = this.el;
39734         var v = el.dom.value;
39735         var d = document.createElement('div');
39736         d.appendChild(document.createTextNode(v));
39737         v = d.innerHTML;
39738         d = null;
39739         v += "&#160;";
39740         var w = Math.min(this.growMax, Math.max(this.metrics.getWidth(v) + /* add extra padding */ 10, this.growMin));
39741         this.el.setWidth(w);
39742         this.fireEvent("autosize", this, w);
39743     },
39744     
39745     // private
39746     SafariOnKeyDown : function(event)
39747     {
39748         // this is a workaround for a password hang bug on chrome/ webkit.
39749         
39750         var isSelectAll = false;
39751         
39752         if(this.el.dom.selectionEnd > 0){
39753             isSelectAll = (this.el.dom.selectionEnd - this.el.dom.selectionStart - this.getValue().length == 0) ? true : false;
39754         }
39755         if(((event.getKey() == 8 || event.getKey() == 46) && this.getValue().length ==1)){ // backspace and delete key
39756             event.preventDefault();
39757             this.setValue('');
39758             return;
39759         }
39760         
39761         if(isSelectAll && event.getCharCode() > 31){ // backspace and delete key
39762             
39763             event.preventDefault();
39764             // this is very hacky as keydown always get's upper case.
39765             
39766             var cc = String.fromCharCode(event.getCharCode());
39767             
39768             
39769             this.setValue( event.shiftKey ?  cc : cc.toLowerCase());
39770             
39771         }
39772         
39773         
39774     }
39775 });/*
39776  * Based on:
39777  * Ext JS Library 1.1.1
39778  * Copyright(c) 2006-2007, Ext JS, LLC.
39779  *
39780  * Originally Released Under LGPL - original licence link has changed is not relivant.
39781  *
39782  * Fork - LGPL
39783  * <script type="text/javascript">
39784  */
39785  
39786 /**
39787  * @class Roo.form.Hidden
39788  * @extends Roo.form.TextField
39789  * Simple Hidden element used on forms 
39790  * 
39791  * usage: form.add(new Roo.form.HiddenField({ 'name' : 'test1' }));
39792  * 
39793  * @constructor
39794  * Creates a new Hidden form element.
39795  * @param {Object} config Configuration options
39796  */
39797
39798
39799
39800 // easy hidden field...
39801 Roo.form.Hidden = function(config){
39802     Roo.form.Hidden.superclass.constructor.call(this, config);
39803 };
39804   
39805 Roo.extend(Roo.form.Hidden, Roo.form.TextField, {
39806     fieldLabel:      '',
39807     inputType:      'hidden',
39808     width:          50,
39809     allowBlank:     true,
39810     labelSeparator: '',
39811     hidden:         true,
39812     itemCls :       'x-form-item-display-none'
39813
39814
39815 });
39816
39817
39818 /*
39819  * Based on:
39820  * Ext JS Library 1.1.1
39821  * Copyright(c) 2006-2007, Ext JS, LLC.
39822  *
39823  * Originally Released Under LGPL - original licence link has changed is not relivant.
39824  *
39825  * Fork - LGPL
39826  * <script type="text/javascript">
39827  */
39828  
39829 /**
39830  * @class Roo.form.TriggerField
39831  * @extends Roo.form.TextField
39832  * Provides a convenient wrapper for TextFields that adds a clickable trigger button (looks like a combobox by default).
39833  * The trigger has no default action, so you must assign a function to implement the trigger click handler by
39834  * overriding {@link #onTriggerClick}. You can create a TriggerField directly, as it renders exactly like a combobox
39835  * for which you can provide a custom implementation.  For example:
39836  * <pre><code>
39837 var trigger = new Roo.form.TriggerField();
39838 trigger.onTriggerClick = myTriggerFn;
39839 trigger.applyTo('my-field');
39840 </code></pre>
39841  *
39842  * However, in general you will most likely want to use TriggerField as the base class for a reusable component.
39843  * {@link Roo.form.DateField} and {@link Roo.form.ComboBox} are perfect examples of this.
39844  * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
39845  * class 'x-form-trigger' by default and triggerClass will be <b>appended</b> if specified.
39846  * @constructor
39847  * Create a new TriggerField.
39848  * @param {Object} config Configuration options (valid {@Roo.form.TextField} config options will also be applied
39849  * to the base TextField)
39850  */
39851 Roo.form.TriggerField = function(config){
39852     this.mimicing = false;
39853     Roo.form.TriggerField.superclass.constructor.call(this, config);
39854 };
39855
39856 Roo.extend(Roo.form.TriggerField, Roo.form.TextField,  {
39857     /**
39858      * @cfg {String} triggerClass A CSS class to apply to the trigger
39859      */
39860     /**
39861      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
39862      * {tag: "input", type: "text", size: "16", autocomplete: "off"})
39863      */
39864     defaultAutoCreate : {tag: "input", type: "text", size: "16", autocomplete: "new-password"},
39865     /**
39866      * @cfg {Boolean} hideTrigger True to hide the trigger element and display only the base text field (defaults to false)
39867      */
39868     hideTrigger:false,
39869
39870     /** @cfg {Boolean} grow @hide */
39871     /** @cfg {Number} growMin @hide */
39872     /** @cfg {Number} growMax @hide */
39873
39874     /**
39875      * @hide 
39876      * @method
39877      */
39878     autoSize: Roo.emptyFn,
39879     // private
39880     monitorTab : true,
39881     // private
39882     deferHeight : true,
39883
39884     
39885     actionMode : 'wrap',
39886     // private
39887     onResize : function(w, h){
39888         Roo.form.TriggerField.superclass.onResize.apply(this, arguments);
39889         if(typeof w == 'number'){
39890             var x = w - this.trigger.getWidth();
39891             this.el.setWidth(this.adjustWidth('input', x));
39892             this.trigger.setStyle('left', x+'px');
39893         }
39894     },
39895
39896     // private
39897     adjustSize : Roo.BoxComponent.prototype.adjustSize,
39898
39899     // private
39900     getResizeEl : function(){
39901         return this.wrap;
39902     },
39903
39904     // private
39905     getPositionEl : function(){
39906         return this.wrap;
39907     },
39908
39909     // private
39910     alignErrorIcon : function(){
39911         this.errorIcon.alignTo(this.wrap, 'tl-tr', [2, 0]);
39912     },
39913
39914     // private
39915     onRender : function(ct, position){
39916         Roo.form.TriggerField.superclass.onRender.call(this, ct, position);
39917         this.wrap = this.el.wrap({cls: "x-form-field-wrap"});
39918         this.trigger = this.wrap.createChild(this.triggerConfig ||
39919                 {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.triggerClass});
39920         if(this.hideTrigger){
39921             this.trigger.setDisplayed(false);
39922         }
39923         this.initTrigger();
39924         if(!this.width){
39925             this.wrap.setWidth(this.el.getWidth()+this.trigger.getWidth());
39926         }
39927     },
39928
39929     // private
39930     initTrigger : function(){
39931         this.trigger.on("click", this.onTriggerClick, this, {preventDefault:true});
39932         this.trigger.addClassOnOver('x-form-trigger-over');
39933         this.trigger.addClassOnClick('x-form-trigger-click');
39934     },
39935
39936     // private
39937     onDestroy : function(){
39938         if(this.trigger){
39939             this.trigger.removeAllListeners();
39940             this.trigger.remove();
39941         }
39942         if(this.wrap){
39943             this.wrap.remove();
39944         }
39945         Roo.form.TriggerField.superclass.onDestroy.call(this);
39946     },
39947
39948     // private
39949     onFocus : function(){
39950         Roo.form.TriggerField.superclass.onFocus.call(this);
39951         if(!this.mimicing){
39952             this.wrap.addClass('x-trigger-wrap-focus');
39953             this.mimicing = true;
39954             Roo.get(Roo.isIE ? document.body : document).on("mousedown", this.mimicBlur, this);
39955             if(this.monitorTab){
39956                 this.el.on("keydown", this.checkTab, this);
39957             }
39958         }
39959     },
39960
39961     // private
39962     checkTab : function(e){
39963         if(e.getKey() == e.TAB){
39964             this.triggerBlur();
39965         }
39966     },
39967
39968     // private
39969     onBlur : function(){
39970         // do nothing
39971     },
39972
39973     // private
39974     mimicBlur : function(e, t){
39975         if(!this.wrap.contains(t) && this.validateBlur()){
39976             this.triggerBlur();
39977         }
39978     },
39979
39980     // private
39981     triggerBlur : function(){
39982         this.mimicing = false;
39983         Roo.get(Roo.isIE ? document.body : document).un("mousedown", this.mimicBlur);
39984         if(this.monitorTab){
39985             this.el.un("keydown", this.checkTab, this);
39986         }
39987         this.wrap.removeClass('x-trigger-wrap-focus');
39988         Roo.form.TriggerField.superclass.onBlur.call(this);
39989     },
39990
39991     // private
39992     // This should be overriden by any subclass that needs to check whether or not the field can be blurred.
39993     validateBlur : function(e, t){
39994         return true;
39995     },
39996
39997     // private
39998     onDisable : function(){
39999         Roo.form.TriggerField.superclass.onDisable.call(this);
40000         if(this.wrap){
40001             this.wrap.addClass('x-item-disabled');
40002         }
40003     },
40004
40005     // private
40006     onEnable : function(){
40007         Roo.form.TriggerField.superclass.onEnable.call(this);
40008         if(this.wrap){
40009             this.wrap.removeClass('x-item-disabled');
40010         }
40011     },
40012
40013     // private
40014     onShow : function(){
40015         var ae = this.getActionEl();
40016         
40017         if(ae){
40018             ae.dom.style.display = '';
40019             ae.dom.style.visibility = 'visible';
40020         }
40021     },
40022
40023     // private
40024     
40025     onHide : function(){
40026         var ae = this.getActionEl();
40027         ae.dom.style.display = 'none';
40028     },
40029
40030     /**
40031      * The function that should handle the trigger's click event.  This method does nothing by default until overridden
40032      * by an implementing function.
40033      * @method
40034      * @param {EventObject} e
40035      */
40036     onTriggerClick : Roo.emptyFn
40037 });
40038
40039 // TwinTriggerField is not a public class to be used directly.  It is meant as an abstract base class
40040 // to be extended by an implementing class.  For an example of implementing this class, see the custom
40041 // SearchField implementation here: http://extjs.com/deploy/ext/examples/form/custom.html
40042 Roo.form.TwinTriggerField = Roo.extend(Roo.form.TriggerField, {
40043     initComponent : function(){
40044         Roo.form.TwinTriggerField.superclass.initComponent.call(this);
40045
40046         this.triggerConfig = {
40047             tag:'span', cls:'x-form-twin-triggers', cn:[
40048             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger1Class},
40049             {tag: "img", src: Roo.BLANK_IMAGE_URL, cls: "x-form-trigger " + this.trigger2Class}
40050         ]};
40051     },
40052
40053     getTrigger : function(index){
40054         return this.triggers[index];
40055     },
40056
40057     initTrigger : function(){
40058         var ts = this.trigger.select('.x-form-trigger', true);
40059         this.wrap.setStyle('overflow', 'hidden');
40060         var triggerField = this;
40061         ts.each(function(t, all, index){
40062             t.hide = function(){
40063                 var w = triggerField.wrap.getWidth();
40064                 this.dom.style.display = 'none';
40065                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40066             };
40067             t.show = function(){
40068                 var w = triggerField.wrap.getWidth();
40069                 this.dom.style.display = '';
40070                 triggerField.el.setWidth(w-triggerField.trigger.getWidth());
40071             };
40072             var triggerIndex = 'Trigger'+(index+1);
40073
40074             if(this['hide'+triggerIndex]){
40075                 t.dom.style.display = 'none';
40076             }
40077             t.on("click", this['on'+triggerIndex+'Click'], this, {preventDefault:true});
40078             t.addClassOnOver('x-form-trigger-over');
40079             t.addClassOnClick('x-form-trigger-click');
40080         }, this);
40081         this.triggers = ts.elements;
40082     },
40083
40084     onTrigger1Click : Roo.emptyFn,
40085     onTrigger2Click : Roo.emptyFn
40086 });/*
40087  * Based on:
40088  * Ext JS Library 1.1.1
40089  * Copyright(c) 2006-2007, Ext JS, LLC.
40090  *
40091  * Originally Released Under LGPL - original licence link has changed is not relivant.
40092  *
40093  * Fork - LGPL
40094  * <script type="text/javascript">
40095  */
40096  
40097 /**
40098  * @class Roo.form.TextArea
40099  * @extends Roo.form.TextField
40100  * Multiline text field.  Can be used as a direct replacement for traditional textarea fields, plus adds
40101  * support for auto-sizing.
40102  * @constructor
40103  * Creates a new TextArea
40104  * @param {Object} config Configuration options
40105  */
40106 Roo.form.TextArea = function(config){
40107     Roo.form.TextArea.superclass.constructor.call(this, config);
40108     // these are provided exchanges for backwards compat
40109     // minHeight/maxHeight were replaced by growMin/growMax to be
40110     // compatible with TextField growing config values
40111     if(this.minHeight !== undefined){
40112         this.growMin = this.minHeight;
40113     }
40114     if(this.maxHeight !== undefined){
40115         this.growMax = this.maxHeight;
40116     }
40117 };
40118
40119 Roo.extend(Roo.form.TextArea, Roo.form.TextField,  {
40120     /**
40121      * @cfg {Number} growMin The minimum height to allow when grow = true (defaults to 60)
40122      */
40123     growMin : 60,
40124     /**
40125      * @cfg {Number} growMax The maximum height to allow when grow = true (defaults to 1000)
40126      */
40127     growMax: 1000,
40128     /**
40129      * @cfg {Boolean} preventScrollbars True to prevent scrollbars from appearing regardless of how much text is
40130      * in the field (equivalent to setting overflow: hidden, defaults to false)
40131      */
40132     preventScrollbars: false,
40133     /**
40134      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
40135      * {tag: "textarea", style: "width:300px;height:60px;", autocomplete: "off"})
40136      */
40137
40138     // private
40139     onRender : function(ct, position){
40140         if(!this.el){
40141             this.defaultAutoCreate = {
40142                 tag: "textarea",
40143                 style:"width:300px;height:60px;",
40144                 autocomplete: "new-password"
40145             };
40146         }
40147         Roo.form.TextArea.superclass.onRender.call(this, ct, position);
40148         if(this.grow){
40149             this.textSizeEl = Roo.DomHelper.append(document.body, {
40150                 tag: "pre", cls: "x-form-grow-sizer"
40151             });
40152             if(this.preventScrollbars){
40153                 this.el.setStyle("overflow", "hidden");
40154             }
40155             this.el.setHeight(this.growMin);
40156         }
40157     },
40158
40159     onDestroy : function(){
40160         if(this.textSizeEl){
40161             this.textSizeEl.parentNode.removeChild(this.textSizeEl);
40162         }
40163         Roo.form.TextArea.superclass.onDestroy.call(this);
40164     },
40165
40166     // private
40167     onKeyUp : function(e){
40168         if(!e.isNavKeyPress() || e.getKey() == e.ENTER){
40169             this.autoSize();
40170         }
40171     },
40172
40173     /**
40174      * Automatically grows the field to accomodate the height of the text up to the maximum field height allowed.
40175      * This only takes effect if grow = true, and fires the autosize event if the height changes.
40176      */
40177     autoSize : function(){
40178         if(!this.grow || !this.textSizeEl){
40179             return;
40180         }
40181         var el = this.el;
40182         var v = el.dom.value;
40183         var ts = this.textSizeEl;
40184
40185         ts.innerHTML = '';
40186         ts.appendChild(document.createTextNode(v));
40187         v = ts.innerHTML;
40188
40189         Roo.fly(ts).setWidth(this.el.getWidth());
40190         if(v.length < 1){
40191             v = "&#160;&#160;";
40192         }else{
40193             if(Roo.isIE){
40194                 v = v.replace(/\n/g, '<p>&#160;</p>');
40195             }
40196             v += "&#160;\n&#160;";
40197         }
40198         ts.innerHTML = v;
40199         var h = Math.min(this.growMax, Math.max(ts.offsetHeight, this.growMin));
40200         if(h != this.lastHeight){
40201             this.lastHeight = h;
40202             this.el.setHeight(h);
40203             this.fireEvent("autosize", this, h);
40204         }
40205     }
40206 });/*
40207  * Based on:
40208  * Ext JS Library 1.1.1
40209  * Copyright(c) 2006-2007, Ext JS, LLC.
40210  *
40211  * Originally Released Under LGPL - original licence link has changed is not relivant.
40212  *
40213  * Fork - LGPL
40214  * <script type="text/javascript">
40215  */
40216  
40217
40218 /**
40219  * @class Roo.form.NumberField
40220  * @extends Roo.form.TextField
40221  * Numeric text field that provides automatic keystroke filtering and numeric validation.
40222  * @constructor
40223  * Creates a new NumberField
40224  * @param {Object} config Configuration options
40225  */
40226 Roo.form.NumberField = function(config){
40227     Roo.form.NumberField.superclass.constructor.call(this, config);
40228 };
40229
40230 Roo.extend(Roo.form.NumberField, Roo.form.TextField,  {
40231     /**
40232      * @cfg {String} fieldClass The default CSS class for the field (defaults to "x-form-field x-form-num-field")
40233      */
40234     fieldClass: "x-form-field x-form-num-field",
40235     /**
40236      * @cfg {Boolean} allowDecimals False to disallow decimal values (defaults to true)
40237      */
40238     allowDecimals : true,
40239     /**
40240      * @cfg {String} decimalSeparator Character(s) to allow as the decimal separator (defaults to '.')
40241      */
40242     decimalSeparator : ".",
40243     /**
40244      * @cfg {Number} decimalPrecision The maximum precision to display after the decimal separator (defaults to 2)
40245      */
40246     decimalPrecision : 2,
40247     /**
40248      * @cfg {Boolean} allowNegative False to prevent entering a negative sign (defaults to true)
40249      */
40250     allowNegative : true,
40251     /**
40252      * @cfg {Number} minValue The minimum allowed value (defaults to Number.NEGATIVE_INFINITY)
40253      */
40254     minValue : Number.NEGATIVE_INFINITY,
40255     /**
40256      * @cfg {Number} maxValue The maximum allowed value (defaults to Number.MAX_VALUE)
40257      */
40258     maxValue : Number.MAX_VALUE,
40259     /**
40260      * @cfg {String} minText Error text to display if the minimum value validation fails (defaults to "The minimum value for this field is {minValue}")
40261      */
40262     minText : "The minimum value for this field is {0}",
40263     /**
40264      * @cfg {String} maxText Error text to display if the maximum value validation fails (defaults to "The maximum value for this field is {maxValue}")
40265      */
40266     maxText : "The maximum value for this field is {0}",
40267     /**
40268      * @cfg {String} nanText Error text to display if the value is not a valid number.  For example, this can happen
40269      * if a valid character like '.' or '-' is left in the field with no number (defaults to "{value} is not a valid number")
40270      */
40271     nanText : "{0} is not a valid number",
40272
40273     // private
40274     initEvents : function(){
40275         Roo.form.NumberField.superclass.initEvents.call(this);
40276         var allowed = "0123456789";
40277         if(this.allowDecimals){
40278             allowed += this.decimalSeparator;
40279         }
40280         if(this.allowNegative){
40281             allowed += "-";
40282         }
40283         this.stripCharsRe = new RegExp('[^'+allowed+']', 'gi');
40284         var keyPress = function(e){
40285             var k = e.getKey();
40286             if(!Roo.isIE && (e.isSpecialKey() || k == e.BACKSPACE || k == e.DELETE)){
40287                 return;
40288             }
40289             var c = e.getCharCode();
40290             if(allowed.indexOf(String.fromCharCode(c)) === -1){
40291                 e.stopEvent();
40292             }
40293         };
40294         this.el.on("keypress", keyPress, this);
40295     },
40296
40297     // private
40298     validateValue : function(value){
40299         if(!Roo.form.NumberField.superclass.validateValue.call(this, value)){
40300             return false;
40301         }
40302         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40303              return true;
40304         }
40305         var num = this.parseValue(value);
40306         if(isNaN(num)){
40307             this.markInvalid(String.format(this.nanText, value));
40308             return false;
40309         }
40310         if(num < this.minValue){
40311             this.markInvalid(String.format(this.minText, this.minValue));
40312             return false;
40313         }
40314         if(num > this.maxValue){
40315             this.markInvalid(String.format(this.maxText, this.maxValue));
40316             return false;
40317         }
40318         return true;
40319     },
40320
40321     getValue : function(){
40322         return this.fixPrecision(this.parseValue(Roo.form.NumberField.superclass.getValue.call(this)));
40323     },
40324
40325     // private
40326     parseValue : function(value){
40327         value = parseFloat(String(value).replace(this.decimalSeparator, "."));
40328         return isNaN(value) ? '' : value;
40329     },
40330
40331     // private
40332     fixPrecision : function(value){
40333         var nan = isNaN(value);
40334         if(!this.allowDecimals || this.decimalPrecision == -1 || nan || !value){
40335             return nan ? '' : value;
40336         }
40337         return parseFloat(value).toFixed(this.decimalPrecision);
40338     },
40339
40340     setValue : function(v){
40341         v = this.fixPrecision(v);
40342         Roo.form.NumberField.superclass.setValue.call(this, String(v).replace(".", this.decimalSeparator));
40343     },
40344
40345     // private
40346     decimalPrecisionFcn : function(v){
40347         return Math.floor(v);
40348     },
40349
40350     beforeBlur : function(){
40351         var v = this.parseValue(this.getRawValue());
40352         if(v){
40353             this.setValue(v);
40354         }
40355     }
40356 });/*
40357  * Based on:
40358  * Ext JS Library 1.1.1
40359  * Copyright(c) 2006-2007, Ext JS, LLC.
40360  *
40361  * Originally Released Under LGPL - original licence link has changed is not relivant.
40362  *
40363  * Fork - LGPL
40364  * <script type="text/javascript">
40365  */
40366  
40367 /**
40368  * @class Roo.form.DateField
40369  * @extends Roo.form.TriggerField
40370  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40371 * @constructor
40372 * Create a new DateField
40373 * @param {Object} config
40374  */
40375 Roo.form.DateField = function(config)
40376 {
40377     Roo.form.DateField.superclass.constructor.call(this, config);
40378     
40379       this.addEvents({
40380          
40381         /**
40382          * @event select
40383          * Fires when a date is selected
40384              * @param {Roo.form.DateField} combo This combo box
40385              * @param {Date} date The date selected
40386              */
40387         'select' : true
40388          
40389     });
40390     
40391     
40392     if(typeof this.minValue == "string") {
40393         this.minValue = this.parseDate(this.minValue);
40394     }
40395     if(typeof this.maxValue == "string") {
40396         this.maxValue = this.parseDate(this.maxValue);
40397     }
40398     this.ddMatch = null;
40399     if(this.disabledDates){
40400         var dd = this.disabledDates;
40401         var re = "(?:";
40402         for(var i = 0; i < dd.length; i++){
40403             re += dd[i];
40404             if(i != dd.length-1) {
40405                 re += "|";
40406             }
40407         }
40408         this.ddMatch = new RegExp(re + ")");
40409     }
40410 };
40411
40412 Roo.extend(Roo.form.DateField, Roo.form.TriggerField,  {
40413     /**
40414      * @cfg {String} format
40415      * The default date format string which can be overriden for localization support.  The format must be
40416      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40417      */
40418     format : "m/d/y",
40419     /**
40420      * @cfg {String} altFormats
40421      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40422      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40423      */
40424     altFormats : "m/d/Y|m-d-y|m-d-Y|m/d|m-d|md|mdy|mdY|d",
40425     /**
40426      * @cfg {Array} disabledDays
40427      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40428      */
40429     disabledDays : null,
40430     /**
40431      * @cfg {String} disabledDaysText
40432      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40433      */
40434     disabledDaysText : "Disabled",
40435     /**
40436      * @cfg {Array} disabledDates
40437      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40438      * expression so they are very powerful. Some examples:
40439      * <ul>
40440      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40441      * <li>["03/08", "09/16"] would disable those days for every year</li>
40442      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40443      * <li>["03/../2006"] would disable every day in March 2006</li>
40444      * <li>["^03"] would disable every day in every March</li>
40445      * </ul>
40446      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40447      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40448      */
40449     disabledDates : null,
40450     /**
40451      * @cfg {String} disabledDatesText
40452      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40453      */
40454     disabledDatesText : "Disabled",
40455     /**
40456      * @cfg {Date/String} minValue
40457      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40458      * valid format (defaults to null).
40459      */
40460     minValue : null,
40461     /**
40462      * @cfg {Date/String} maxValue
40463      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40464      * valid format (defaults to null).
40465      */
40466     maxValue : null,
40467     /**
40468      * @cfg {String} minText
40469      * The error text to display when the date in the cell is before minValue (defaults to
40470      * 'The date in this field must be after {minValue}').
40471      */
40472     minText : "The date in this field must be equal to or after {0}",
40473     /**
40474      * @cfg {String} maxText
40475      * The error text to display when the date in the cell is after maxValue (defaults to
40476      * 'The date in this field must be before {maxValue}').
40477      */
40478     maxText : "The date in this field must be equal to or before {0}",
40479     /**
40480      * @cfg {String} invalidText
40481      * The error text to display when the date in the field is invalid (defaults to
40482      * '{value} is not a valid date - it must be in the format {format}').
40483      */
40484     invalidText : "{0} is not a valid date - it must be in the format {1}",
40485     /**
40486      * @cfg {String} triggerClass
40487      * An additional CSS class used to style the trigger button.  The trigger will always get the
40488      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40489      * which displays a calendar icon).
40490      */
40491     triggerClass : 'x-form-date-trigger',
40492     
40493
40494     /**
40495      * @cfg {Boolean} useIso
40496      * if enabled, then the date field will use a hidden field to store the 
40497      * real value as iso formated date. default (false)
40498      */ 
40499     useIso : false,
40500     /**
40501      * @cfg {String/Object} autoCreate
40502      * A DomHelper element spec, or true for a default element spec (defaults to
40503      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40504      */ 
40505     // private
40506     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "off"},
40507     
40508     // private
40509     hiddenField: false,
40510     
40511     onRender : function(ct, position)
40512     {
40513         Roo.form.DateField.superclass.onRender.call(this, ct, position);
40514         if (this.useIso) {
40515             //this.el.dom.removeAttribute('name'); 
40516             Roo.log("Changing name?");
40517             this.el.dom.setAttribute('name', this.name + '____hidden___' ); 
40518             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40519                     'before', true);
40520             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40521             // prevent input submission
40522             this.hiddenName = this.name;
40523         }
40524             
40525             
40526     },
40527     
40528     // private
40529     validateValue : function(value)
40530     {
40531         value = this.formatDate(value);
40532         if(!Roo.form.DateField.superclass.validateValue.call(this, value)){
40533             Roo.log('super failed');
40534             return false;
40535         }
40536         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40537              return true;
40538         }
40539         var svalue = value;
40540         value = this.parseDate(value);
40541         if(!value){
40542             Roo.log('parse date failed' + svalue);
40543             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40544             return false;
40545         }
40546         var time = value.getTime();
40547         if(this.minValue && time < this.minValue.getTime()){
40548             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40549             return false;
40550         }
40551         if(this.maxValue && time > this.maxValue.getTime()){
40552             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40553             return false;
40554         }
40555         if(this.disabledDays){
40556             var day = value.getDay();
40557             for(var i = 0; i < this.disabledDays.length; i++) {
40558                 if(day === this.disabledDays[i]){
40559                     this.markInvalid(this.disabledDaysText);
40560                     return false;
40561                 }
40562             }
40563         }
40564         var fvalue = this.formatDate(value);
40565         if(this.ddMatch && this.ddMatch.test(fvalue)){
40566             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40567             return false;
40568         }
40569         return true;
40570     },
40571
40572     // private
40573     // Provides logic to override the default TriggerField.validateBlur which just returns true
40574     validateBlur : function(){
40575         return !this.menu || !this.menu.isVisible();
40576     },
40577     
40578     getName: function()
40579     {
40580         // returns hidden if it's set..
40581         if (!this.rendered) {return ''};
40582         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
40583         
40584     },
40585
40586     /**
40587      * Returns the current date value of the date field.
40588      * @return {Date} The date value
40589      */
40590     getValue : function(){
40591         
40592         return  this.hiddenField ?
40593                 this.hiddenField.value :
40594                 this.parseDate(Roo.form.DateField.superclass.getValue.call(this)) || "";
40595     },
40596
40597     /**
40598      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40599      * date, using DateField.format as the date format, according to the same rules as {@link Date#parseDate}
40600      * (the default format used is "m/d/y").
40601      * <br />Usage:
40602      * <pre><code>
40603 //All of these calls set the same date value (May 4, 2006)
40604
40605 //Pass a date object:
40606 var dt = new Date('5/4/06');
40607 dateField.setValue(dt);
40608
40609 //Pass a date string (default format):
40610 dateField.setValue('5/4/06');
40611
40612 //Pass a date string (custom format):
40613 dateField.format = 'Y-m-d';
40614 dateField.setValue('2006-5-4');
40615 </code></pre>
40616      * @param {String/Date} date The date or valid date string
40617      */
40618     setValue : function(date){
40619         if (this.hiddenField) {
40620             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40621         }
40622         Roo.form.DateField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40623         // make sure the value field is always stored as a date..
40624         this.value = this.parseDate(date);
40625         
40626         
40627     },
40628
40629     // private
40630     parseDate : function(value){
40631         if(!value || value instanceof Date){
40632             return value;
40633         }
40634         var v = Date.parseDate(value, this.format);
40635          if (!v && this.useIso) {
40636             v = Date.parseDate(value, 'Y-m-d');
40637         }
40638         if(!v && this.altFormats){
40639             if(!this.altFormatsArray){
40640                 this.altFormatsArray = this.altFormats.split("|");
40641             }
40642             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
40643                 v = Date.parseDate(value, this.altFormatsArray[i]);
40644             }
40645         }
40646         return v;
40647     },
40648
40649     // private
40650     formatDate : function(date, fmt){
40651         return (!date || !(date instanceof Date)) ?
40652                date : date.dateFormat(fmt || this.format);
40653     },
40654
40655     // private
40656     menuListeners : {
40657         select: function(m, d){
40658             
40659             this.setValue(d);
40660             this.fireEvent('select', this, d);
40661         },
40662         show : function(){ // retain focus styling
40663             this.onFocus();
40664         },
40665         hide : function(){
40666             this.focus.defer(10, this);
40667             var ml = this.menuListeners;
40668             this.menu.un("select", ml.select,  this);
40669             this.menu.un("show", ml.show,  this);
40670             this.menu.un("hide", ml.hide,  this);
40671         }
40672     },
40673
40674     // private
40675     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
40676     onTriggerClick : function(){
40677         if(this.disabled){
40678             return;
40679         }
40680         if(this.menu == null){
40681             this.menu = new Roo.menu.DateMenu();
40682         }
40683         Roo.apply(this.menu.picker,  {
40684             showClear: this.allowBlank,
40685             minDate : this.minValue,
40686             maxDate : this.maxValue,
40687             disabledDatesRE : this.ddMatch,
40688             disabledDatesText : this.disabledDatesText,
40689             disabledDays : this.disabledDays,
40690             disabledDaysText : this.disabledDaysText,
40691             format : this.useIso ? 'Y-m-d' : this.format,
40692             minText : String.format(this.minText, this.formatDate(this.minValue)),
40693             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
40694         });
40695         this.menu.on(Roo.apply({}, this.menuListeners, {
40696             scope:this
40697         }));
40698         this.menu.picker.setValue(this.getValue() || new Date());
40699         this.menu.show(this.el, "tl-bl?");
40700     },
40701
40702     beforeBlur : function(){
40703         var v = this.parseDate(this.getRawValue());
40704         if(v){
40705             this.setValue(v);
40706         }
40707     },
40708
40709     /*@
40710      * overide
40711      * 
40712      */
40713     isDirty : function() {
40714         if(this.disabled) {
40715             return false;
40716         }
40717         
40718         if(typeof(this.startValue) === 'undefined'){
40719             return false;
40720         }
40721         
40722         return String(this.getValue()) !== String(this.startValue);
40723         
40724     },
40725     // @overide
40726     cleanLeadingSpace : function(e)
40727     {
40728        return;
40729     }
40730     
40731 });/*
40732  * Based on:
40733  * Ext JS Library 1.1.1
40734  * Copyright(c) 2006-2007, Ext JS, LLC.
40735  *
40736  * Originally Released Under LGPL - original licence link has changed is not relivant.
40737  *
40738  * Fork - LGPL
40739  * <script type="text/javascript">
40740  */
40741  
40742 /**
40743  * @class Roo.form.MonthField
40744  * @extends Roo.form.TriggerField
40745  * Provides a date input field with a {@link Roo.DatePicker} dropdown and automatic date validation.
40746 * @constructor
40747 * Create a new MonthField
40748 * @param {Object} config
40749  */
40750 Roo.form.MonthField = function(config){
40751     
40752     Roo.form.MonthField.superclass.constructor.call(this, config);
40753     
40754       this.addEvents({
40755          
40756         /**
40757          * @event select
40758          * Fires when a date is selected
40759              * @param {Roo.form.MonthFieeld} combo This combo box
40760              * @param {Date} date The date selected
40761              */
40762         'select' : true
40763          
40764     });
40765     
40766     
40767     if(typeof this.minValue == "string") {
40768         this.minValue = this.parseDate(this.minValue);
40769     }
40770     if(typeof this.maxValue == "string") {
40771         this.maxValue = this.parseDate(this.maxValue);
40772     }
40773     this.ddMatch = null;
40774     if(this.disabledDates){
40775         var dd = this.disabledDates;
40776         var re = "(?:";
40777         for(var i = 0; i < dd.length; i++){
40778             re += dd[i];
40779             if(i != dd.length-1) {
40780                 re += "|";
40781             }
40782         }
40783         this.ddMatch = new RegExp(re + ")");
40784     }
40785 };
40786
40787 Roo.extend(Roo.form.MonthField, Roo.form.TriggerField,  {
40788     /**
40789      * @cfg {String} format
40790      * The default date format string which can be overriden for localization support.  The format must be
40791      * valid according to {@link Date#parseDate} (defaults to 'm/d/y').
40792      */
40793     format : "M Y",
40794     /**
40795      * @cfg {String} altFormats
40796      * Multiple date formats separated by "|" to try when parsing a user input value and it doesn't match the defined
40797      * format (defaults to 'm/d/Y|m-d-y|m-d-Y|m/d|m-d|d').
40798      */
40799     altFormats : "M Y|m/Y|m-y|m-Y|my|mY",
40800     /**
40801      * @cfg {Array} disabledDays
40802      * An array of days to disable, 0 based. For example, [0, 6] disables Sunday and Saturday (defaults to null).
40803      */
40804     disabledDays : [0,1,2,3,4,5,6],
40805     /**
40806      * @cfg {String} disabledDaysText
40807      * The tooltip to display when the date falls on a disabled day (defaults to 'Disabled')
40808      */
40809     disabledDaysText : "Disabled",
40810     /**
40811      * @cfg {Array} disabledDates
40812      * An array of "dates" to disable, as strings. These strings will be used to build a dynamic regular
40813      * expression so they are very powerful. Some examples:
40814      * <ul>
40815      * <li>["03/08/2003", "09/16/2003"] would disable those exact dates</li>
40816      * <li>["03/08", "09/16"] would disable those days for every year</li>
40817      * <li>["^03/08"] would only match the beginning (useful if you are using short years)</li>
40818      * <li>["03/../2006"] would disable every day in March 2006</li>
40819      * <li>["^03"] would disable every day in every March</li>
40820      * </ul>
40821      * In order to support regular expressions, if you are using a date format that has "." in it, you will have to
40822      * escape the dot when restricting dates. For example: ["03\\.08\\.03"].
40823      */
40824     disabledDates : null,
40825     /**
40826      * @cfg {String} disabledDatesText
40827      * The tooltip text to display when the date falls on a disabled date (defaults to 'Disabled')
40828      */
40829     disabledDatesText : "Disabled",
40830     /**
40831      * @cfg {Date/String} minValue
40832      * The minimum allowed date. Can be either a Javascript date object or a string date in a
40833      * valid format (defaults to null).
40834      */
40835     minValue : null,
40836     /**
40837      * @cfg {Date/String} maxValue
40838      * The maximum allowed date. Can be either a Javascript date object or a string date in a
40839      * valid format (defaults to null).
40840      */
40841     maxValue : null,
40842     /**
40843      * @cfg {String} minText
40844      * The error text to display when the date in the cell is before minValue (defaults to
40845      * 'The date in this field must be after {minValue}').
40846      */
40847     minText : "The date in this field must be equal to or after {0}",
40848     /**
40849      * @cfg {String} maxTextf
40850      * The error text to display when the date in the cell is after maxValue (defaults to
40851      * 'The date in this field must be before {maxValue}').
40852      */
40853     maxText : "The date in this field must be equal to or before {0}",
40854     /**
40855      * @cfg {String} invalidText
40856      * The error text to display when the date in the field is invalid (defaults to
40857      * '{value} is not a valid date - it must be in the format {format}').
40858      */
40859     invalidText : "{0} is not a valid date - it must be in the format {1}",
40860     /**
40861      * @cfg {String} triggerClass
40862      * An additional CSS class used to style the trigger button.  The trigger will always get the
40863      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-date-trigger'
40864      * which displays a calendar icon).
40865      */
40866     triggerClass : 'x-form-date-trigger',
40867     
40868
40869     /**
40870      * @cfg {Boolean} useIso
40871      * if enabled, then the date field will use a hidden field to store the 
40872      * real value as iso formated date. default (true)
40873      */ 
40874     useIso : true,
40875     /**
40876      * @cfg {String/Object} autoCreate
40877      * A DomHelper element spec, or true for a default element spec (defaults to
40878      * {tag: "input", type: "text", size: "10", autocomplete: "off"})
40879      */ 
40880     // private
40881     defaultAutoCreate : {tag: "input", type: "text", size: "10", autocomplete: "new-password"},
40882     
40883     // private
40884     hiddenField: false,
40885     
40886     hideMonthPicker : false,
40887     
40888     onRender : function(ct, position)
40889     {
40890         Roo.form.MonthField.superclass.onRender.call(this, ct, position);
40891         if (this.useIso) {
40892             this.el.dom.removeAttribute('name'); 
40893             this.hiddenField = this.el.insertSibling({ tag:'input', type:'hidden', name: this.name },
40894                     'before', true);
40895             this.hiddenField.value = this.value ? this.formatDate(this.value, 'Y-m-d') : '';
40896             // prevent input submission
40897             this.hiddenName = this.name;
40898         }
40899             
40900             
40901     },
40902     
40903     // private
40904     validateValue : function(value)
40905     {
40906         value = this.formatDate(value);
40907         if(!Roo.form.MonthField.superclass.validateValue.call(this, value)){
40908             return false;
40909         }
40910         if(value.length < 1){ // if it's blank and textfield didn't flag it then it's valid
40911              return true;
40912         }
40913         var svalue = value;
40914         value = this.parseDate(value);
40915         if(!value){
40916             this.markInvalid(String.format(this.invalidText, svalue, this.format));
40917             return false;
40918         }
40919         var time = value.getTime();
40920         if(this.minValue && time < this.minValue.getTime()){
40921             this.markInvalid(String.format(this.minText, this.formatDate(this.minValue)));
40922             return false;
40923         }
40924         if(this.maxValue && time > this.maxValue.getTime()){
40925             this.markInvalid(String.format(this.maxText, this.formatDate(this.maxValue)));
40926             return false;
40927         }
40928         /*if(this.disabledDays){
40929             var day = value.getDay();
40930             for(var i = 0; i < this.disabledDays.length; i++) {
40931                 if(day === this.disabledDays[i]){
40932                     this.markInvalid(this.disabledDaysText);
40933                     return false;
40934                 }
40935             }
40936         }
40937         */
40938         var fvalue = this.formatDate(value);
40939         /*if(this.ddMatch && this.ddMatch.test(fvalue)){
40940             this.markInvalid(String.format(this.disabledDatesText, fvalue));
40941             return false;
40942         }
40943         */
40944         return true;
40945     },
40946
40947     // private
40948     // Provides logic to override the default TriggerField.validateBlur which just returns true
40949     validateBlur : function(){
40950         return !this.menu || !this.menu.isVisible();
40951     },
40952
40953     /**
40954      * Returns the current date value of the date field.
40955      * @return {Date} The date value
40956      */
40957     getValue : function(){
40958         
40959         
40960         
40961         return  this.hiddenField ?
40962                 this.hiddenField.value :
40963                 this.parseDate(Roo.form.MonthField.superclass.getValue.call(this)) || "";
40964     },
40965
40966     /**
40967      * Sets the value of the date field.  You can pass a date object or any string that can be parsed into a valid
40968      * date, using MonthField.format as the date format, according to the same rules as {@link Date#parseDate}
40969      * (the default format used is "m/d/y").
40970      * <br />Usage:
40971      * <pre><code>
40972 //All of these calls set the same date value (May 4, 2006)
40973
40974 //Pass a date object:
40975 var dt = new Date('5/4/06');
40976 monthField.setValue(dt);
40977
40978 //Pass a date string (default format):
40979 monthField.setValue('5/4/06');
40980
40981 //Pass a date string (custom format):
40982 monthField.format = 'Y-m-d';
40983 monthField.setValue('2006-5-4');
40984 </code></pre>
40985      * @param {String/Date} date The date or valid date string
40986      */
40987     setValue : function(date){
40988         Roo.log('month setValue' + date);
40989         // can only be first of month..
40990         
40991         var val = this.parseDate(date);
40992         
40993         if (this.hiddenField) {
40994             this.hiddenField.value = this.formatDate(this.parseDate(date), 'Y-m-d');
40995         }
40996         Roo.form.MonthField.superclass.setValue.call(this, this.formatDate(this.parseDate(date)));
40997         this.value = this.parseDate(date);
40998     },
40999
41000     // private
41001     parseDate : function(value){
41002         if(!value || value instanceof Date){
41003             value = value ? Date.parseDate(value.format('Y-m') + '-01', 'Y-m-d') : null;
41004             return value;
41005         }
41006         var v = Date.parseDate(value, this.format);
41007         if (!v && this.useIso) {
41008             v = Date.parseDate(value, 'Y-m-d');
41009         }
41010         if (v) {
41011             // 
41012             v = Date.parseDate(v.format('Y-m') +'-01', 'Y-m-d');
41013         }
41014         
41015         
41016         if(!v && this.altFormats){
41017             if(!this.altFormatsArray){
41018                 this.altFormatsArray = this.altFormats.split("|");
41019             }
41020             for(var i = 0, len = this.altFormatsArray.length; i < len && !v; i++){
41021                 v = Date.parseDate(value, this.altFormatsArray[i]);
41022             }
41023         }
41024         return v;
41025     },
41026
41027     // private
41028     formatDate : function(date, fmt){
41029         return (!date || !(date instanceof Date)) ?
41030                date : date.dateFormat(fmt || this.format);
41031     },
41032
41033     // private
41034     menuListeners : {
41035         select: function(m, d){
41036             this.setValue(d);
41037             this.fireEvent('select', this, d);
41038         },
41039         show : function(){ // retain focus styling
41040             this.onFocus();
41041         },
41042         hide : function(){
41043             this.focus.defer(10, this);
41044             var ml = this.menuListeners;
41045             this.menu.un("select", ml.select,  this);
41046             this.menu.un("show", ml.show,  this);
41047             this.menu.un("hide", ml.hide,  this);
41048         }
41049     },
41050     // private
41051     // Implements the default empty TriggerField.onTriggerClick function to display the DatePicker
41052     onTriggerClick : function(){
41053         if(this.disabled){
41054             return;
41055         }
41056         if(this.menu == null){
41057             this.menu = new Roo.menu.DateMenu();
41058            
41059         }
41060         
41061         Roo.apply(this.menu.picker,  {
41062             
41063             showClear: this.allowBlank,
41064             minDate : this.minValue,
41065             maxDate : this.maxValue,
41066             disabledDatesRE : this.ddMatch,
41067             disabledDatesText : this.disabledDatesText,
41068             
41069             format : this.useIso ? 'Y-m-d' : this.format,
41070             minText : String.format(this.minText, this.formatDate(this.minValue)),
41071             maxText : String.format(this.maxText, this.formatDate(this.maxValue))
41072             
41073         });
41074          this.menu.on(Roo.apply({}, this.menuListeners, {
41075             scope:this
41076         }));
41077        
41078         
41079         var m = this.menu;
41080         var p = m.picker;
41081         
41082         // hide month picker get's called when we called by 'before hide';
41083         
41084         var ignorehide = true;
41085         p.hideMonthPicker  = function(disableAnim){
41086             if (ignorehide) {
41087                 return;
41088             }
41089              if(this.monthPicker){
41090                 Roo.log("hideMonthPicker called");
41091                 if(disableAnim === true){
41092                     this.monthPicker.hide();
41093                 }else{
41094                     this.monthPicker.slideOut('t', {duration:.2});
41095                     p.setValue(new Date(m.picker.mpSelYear, m.picker.mpSelMonth, 1));
41096                     p.fireEvent("select", this, this.value);
41097                     m.hide();
41098                 }
41099             }
41100         }
41101         
41102         Roo.log('picker set value');
41103         Roo.log(this.getValue());
41104         p.setValue(this.getValue() ? this.parseDate(this.getValue()) : new Date());
41105         m.show(this.el, 'tl-bl?');
41106         ignorehide  = false;
41107         // this will trigger hideMonthPicker..
41108         
41109         
41110         // hidden the day picker
41111         Roo.select('.x-date-picker table', true).first().dom.style.visibility = "hidden";
41112         
41113         
41114         
41115       
41116         
41117         p.showMonthPicker.defer(100, p);
41118     
41119         
41120        
41121     },
41122
41123     beforeBlur : function(){
41124         var v = this.parseDate(this.getRawValue());
41125         if(v){
41126             this.setValue(v);
41127         }
41128     }
41129
41130     /** @cfg {Boolean} grow @hide */
41131     /** @cfg {Number} growMin @hide */
41132     /** @cfg {Number} growMax @hide */
41133     /**
41134      * @hide
41135      * @method autoSize
41136      */
41137 });/*
41138  * Based on:
41139  * Ext JS Library 1.1.1
41140  * Copyright(c) 2006-2007, Ext JS, LLC.
41141  *
41142  * Originally Released Under LGPL - original licence link has changed is not relivant.
41143  *
41144  * Fork - LGPL
41145  * <script type="text/javascript">
41146  */
41147  
41148
41149 /**
41150  * @class Roo.form.ComboBox
41151  * @extends Roo.form.TriggerField
41152  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
41153  * @constructor
41154  * Create a new ComboBox.
41155  * @param {Object} config Configuration options
41156  */
41157 Roo.form.ComboBox = function(config){
41158     Roo.form.ComboBox.superclass.constructor.call(this, config);
41159     this.addEvents({
41160         /**
41161          * @event expand
41162          * Fires when the dropdown list is expanded
41163              * @param {Roo.form.ComboBox} combo This combo box
41164              */
41165         'expand' : true,
41166         /**
41167          * @event collapse
41168          * Fires when the dropdown list is collapsed
41169              * @param {Roo.form.ComboBox} combo This combo box
41170              */
41171         'collapse' : true,
41172         /**
41173          * @event beforeselect
41174          * Fires before a list item is selected. Return false to cancel the selection.
41175              * @param {Roo.form.ComboBox} combo This combo box
41176              * @param {Roo.data.Record} record The data record returned from the underlying store
41177              * @param {Number} index The index of the selected item in the dropdown list
41178              */
41179         'beforeselect' : true,
41180         /**
41181          * @event select
41182          * Fires when a list item is selected
41183              * @param {Roo.form.ComboBox} combo This combo box
41184              * @param {Roo.data.Record} record The data record returned from the underlying store (or false on clear)
41185              * @param {Number} index The index of the selected item in the dropdown list
41186              */
41187         'select' : true,
41188         /**
41189          * @event beforequery
41190          * Fires before all queries are processed. Return false to cancel the query or set cancel to true.
41191          * The event object passed has these properties:
41192              * @param {Roo.form.ComboBox} combo This combo box
41193              * @param {String} query The query
41194              * @param {Boolean} forceAll true to force "all" query
41195              * @param {Boolean} cancel true to cancel the query
41196              * @param {Object} e The query event object
41197              */
41198         'beforequery': true,
41199          /**
41200          * @event add
41201          * Fires when the 'add' icon is pressed (add a listener to enable add button)
41202              * @param {Roo.form.ComboBox} combo This combo box
41203              */
41204         'add' : true,
41205         /**
41206          * @event edit
41207          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
41208              * @param {Roo.form.ComboBox} combo This combo box
41209              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
41210              */
41211         'edit' : true
41212         
41213         
41214     });
41215     if(this.transform){
41216         this.allowDomMove = false;
41217         var s = Roo.getDom(this.transform);
41218         if(!this.hiddenName){
41219             this.hiddenName = s.name;
41220         }
41221         if(!this.store){
41222             this.mode = 'local';
41223             var d = [], opts = s.options;
41224             for(var i = 0, len = opts.length;i < len; i++){
41225                 var o = opts[i];
41226                 var value = (Roo.isIE ? o.getAttributeNode('value').specified : o.hasAttribute('value')) ? o.value : o.text;
41227                 if(o.selected) {
41228                     this.value = value;
41229                 }
41230                 d.push([value, o.text]);
41231             }
41232             this.store = new Roo.data.SimpleStore({
41233                 'id': 0,
41234                 fields: ['value', 'text'],
41235                 data : d
41236             });
41237             this.valueField = 'value';
41238             this.displayField = 'text';
41239         }
41240         s.name = Roo.id(); // wipe out the name in case somewhere else they have a reference
41241         if(!this.lazyRender){
41242             this.target = true;
41243             this.el = Roo.DomHelper.insertBefore(s, this.autoCreate || this.defaultAutoCreate);
41244             s.parentNode.removeChild(s); // remove it
41245             this.render(this.el.parentNode);
41246         }else{
41247             s.parentNode.removeChild(s); // remove it
41248         }
41249
41250     }
41251     if (this.store) {
41252         this.store = Roo.factory(this.store, Roo.data);
41253     }
41254     
41255     this.selectedIndex = -1;
41256     if(this.mode == 'local'){
41257         if(config.queryDelay === undefined){
41258             this.queryDelay = 10;
41259         }
41260         if(config.minChars === undefined){
41261             this.minChars = 0;
41262         }
41263     }
41264 };
41265
41266 Roo.extend(Roo.form.ComboBox, Roo.form.TriggerField, {
41267     /**
41268      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
41269      */
41270     /**
41271      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
41272      * rendering into an Roo.Editor, defaults to false)
41273      */
41274     /**
41275      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
41276      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
41277      */
41278     /**
41279      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
41280      */
41281     /**
41282      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
41283      * the dropdown list (defaults to undefined, with no header element)
41284      */
41285
41286      /**
41287      * @cfg {String/Roo.Template} tpl The template to use to render the output
41288      */
41289      
41290     // private
41291     defaultAutoCreate : {tag: "input", type: "text", size: "24", autocomplete: "off"},
41292     /**
41293      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
41294      */
41295     listWidth: undefined,
41296     /**
41297      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
41298      * mode = 'remote' or 'text' if mode = 'local')
41299      */
41300     displayField: undefined,
41301     /**
41302      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
41303      * mode = 'remote' or 'value' if mode = 'local'). 
41304      * Note: use of a valueField requires the user make a selection
41305      * in order for a value to be mapped.
41306      */
41307     valueField: undefined,
41308     
41309     
41310     /**
41311      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
41312      * field's data value (defaults to the underlying DOM element's name)
41313      */
41314     hiddenName: undefined,
41315     /**
41316      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
41317      */
41318     listClass: '',
41319     /**
41320      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
41321      */
41322     selectedClass: 'x-combo-selected',
41323     /**
41324      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
41325      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
41326      * which displays a downward arrow icon).
41327      */
41328     triggerClass : 'x-form-arrow-trigger',
41329     /**
41330      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
41331      */
41332     shadow:'sides',
41333     /**
41334      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
41335      * anchor positions (defaults to 'tl-bl')
41336      */
41337     listAlign: 'tl-bl?',
41338     /**
41339      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
41340      */
41341     maxHeight: 300,
41342     /**
41343      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
41344      * query specified by the allQuery config option (defaults to 'query')
41345      */
41346     triggerAction: 'query',
41347     /**
41348      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
41349      * (defaults to 4, does not apply if editable = false)
41350      */
41351     minChars : 4,
41352     /**
41353      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
41354      * delay (typeAheadDelay) if it matches a known value (defaults to false)
41355      */
41356     typeAhead: false,
41357     /**
41358      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
41359      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
41360      */
41361     queryDelay: 500,
41362     /**
41363      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
41364      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
41365      */
41366     pageSize: 0,
41367     /**
41368      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
41369      * when editable = true (defaults to false)
41370      */
41371     selectOnFocus:false,
41372     /**
41373      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
41374      */
41375     queryParam: 'query',
41376     /**
41377      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
41378      * when mode = 'remote' (defaults to 'Loading...')
41379      */
41380     loadingText: 'Loading...',
41381     /**
41382      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
41383      */
41384     resizable: false,
41385     /**
41386      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
41387      */
41388     handleHeight : 8,
41389     /**
41390      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
41391      * traditional select (defaults to true)
41392      */
41393     editable: true,
41394     /**
41395      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
41396      */
41397     allQuery: '',
41398     /**
41399      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
41400      */
41401     mode: 'remote',
41402     /**
41403      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
41404      * listWidth has a higher value)
41405      */
41406     minListWidth : 70,
41407     /**
41408      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
41409      * allow the user to set arbitrary text into the field (defaults to false)
41410      */
41411     forceSelection:false,
41412     /**
41413      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
41414      * if typeAhead = true (defaults to 250)
41415      */
41416     typeAheadDelay : 250,
41417     /**
41418      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
41419      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
41420      */
41421     valueNotFoundText : undefined,
41422     /**
41423      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
41424      */
41425     blockFocus : false,
41426     
41427     /**
41428      * @cfg {Boolean} disableClear Disable showing of clear button.
41429      */
41430     disableClear : false,
41431     /**
41432      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
41433      */
41434     alwaysQuery : false,
41435     
41436     //private
41437     addicon : false,
41438     editicon: false,
41439     
41440     // element that contains real text value.. (when hidden is used..)
41441      
41442     // private
41443     onRender : function(ct, position)
41444     {
41445         Roo.form.ComboBox.superclass.onRender.call(this, ct, position);
41446         
41447         if(this.hiddenName){
41448             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
41449                     'before', true);
41450             this.hiddenField.value =
41451                 this.hiddenValue !== undefined ? this.hiddenValue :
41452                 this.value !== undefined ? this.value : '';
41453
41454             // prevent input submission
41455             this.el.dom.removeAttribute('name');
41456              
41457              
41458         }
41459         
41460         if(Roo.isGecko){
41461             this.el.dom.setAttribute('autocomplete', 'off');
41462         }
41463
41464         var cls = 'x-combo-list';
41465
41466         this.list = new Roo.Layer({
41467             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
41468         });
41469
41470         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
41471         this.list.setWidth(lw);
41472         this.list.swallowEvent('mousewheel');
41473         this.assetHeight = 0;
41474
41475         if(this.title){
41476             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
41477             this.assetHeight += this.header.getHeight();
41478         }
41479
41480         this.innerList = this.list.createChild({cls:cls+'-inner'});
41481         this.innerList.on('mouseover', this.onViewOver, this);
41482         this.innerList.on('mousemove', this.onViewMove, this);
41483         this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41484         
41485         if(this.allowBlank && !this.pageSize && !this.disableClear){
41486             this.footer = this.list.createChild({cls:cls+'-ft'});
41487             this.pageTb = new Roo.Toolbar(this.footer);
41488            
41489         }
41490         if(this.pageSize){
41491             this.footer = this.list.createChild({cls:cls+'-ft'});
41492             this.pageTb = new Roo.PagingToolbar(this.footer, this.store,
41493                     {pageSize: this.pageSize});
41494             
41495         }
41496         
41497         if (this.pageTb && this.allowBlank && !this.disableClear) {
41498             var _this = this;
41499             this.pageTb.add(new Roo.Toolbar.Fill(), {
41500                 cls: 'x-btn-icon x-btn-clear',
41501                 text: '&#160;',
41502                 handler: function()
41503                 {
41504                     _this.collapse();
41505                     _this.clearValue();
41506                     _this.onSelect(false, -1);
41507                 }
41508             });
41509         }
41510         if (this.footer) {
41511             this.assetHeight += this.footer.getHeight();
41512         }
41513         
41514
41515         if(!this.tpl){
41516             this.tpl = '<div class="'+cls+'-item">{' + this.displayField + '}</div>';
41517         }
41518
41519         this.view = new Roo.View(this.innerList, this.tpl, {
41520             singleSelect:true,
41521             store: this.store,
41522             selectedClass: this.selectedClass
41523         });
41524
41525         this.view.on('click', this.onViewClick, this);
41526
41527         this.store.on('beforeload', this.onBeforeLoad, this);
41528         this.store.on('load', this.onLoad, this);
41529         this.store.on('loadexception', this.onLoadException, this);
41530
41531         if(this.resizable){
41532             this.resizer = new Roo.Resizable(this.list,  {
41533                pinned:true, handles:'se'
41534             });
41535             this.resizer.on('resize', function(r, w, h){
41536                 this.maxHeight = h-this.handleHeight-this.list.getFrameWidth('tb')-this.assetHeight;
41537                 this.listWidth = w;
41538                 this.innerList.setWidth(w - this.list.getFrameWidth('lr'));
41539                 this.restrictHeight();
41540             }, this);
41541             this[this.pageSize?'footer':'innerList'].setStyle('margin-bottom', this.handleHeight+'px');
41542         }
41543         if(!this.editable){
41544             this.editable = true;
41545             this.setEditable(false);
41546         }  
41547         
41548         
41549         if (typeof(this.events.add.listeners) != 'undefined') {
41550             
41551             this.addicon = this.wrap.createChild(
41552                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-add' });  
41553        
41554             this.addicon.on('click', function(e) {
41555                 this.fireEvent('add', this);
41556             }, this);
41557         }
41558         if (typeof(this.events.edit.listeners) != 'undefined') {
41559             
41560             this.editicon = this.wrap.createChild(
41561                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-combo-edit' });  
41562             if (this.addicon) {
41563                 this.editicon.setStyle('margin-left', '40px');
41564             }
41565             this.editicon.on('click', function(e) {
41566                 
41567                 // we fire even  if inothing is selected..
41568                 this.fireEvent('edit', this, this.lastData );
41569                 
41570             }, this);
41571         }
41572         
41573         
41574         
41575     },
41576
41577     // private
41578     initEvents : function(){
41579         Roo.form.ComboBox.superclass.initEvents.call(this);
41580
41581         this.keyNav = new Roo.KeyNav(this.el, {
41582             "up" : function(e){
41583                 this.inKeyMode = true;
41584                 this.selectPrev();
41585             },
41586
41587             "down" : function(e){
41588                 if(!this.isExpanded()){
41589                     this.onTriggerClick();
41590                 }else{
41591                     this.inKeyMode = true;
41592                     this.selectNext();
41593                 }
41594             },
41595
41596             "enter" : function(e){
41597                 this.onViewClick();
41598                 //return true;
41599             },
41600
41601             "esc" : function(e){
41602                 this.collapse();
41603             },
41604
41605             "tab" : function(e){
41606                 this.onViewClick(false);
41607                 this.fireEvent("specialkey", this, e);
41608                 return true;
41609             },
41610
41611             scope : this,
41612
41613             doRelay : function(foo, bar, hname){
41614                 if(hname == 'down' || this.scope.isExpanded()){
41615                    return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
41616                 }
41617                 return true;
41618             },
41619
41620             forceKeyDown: true
41621         });
41622         this.queryDelay = Math.max(this.queryDelay || 10,
41623                 this.mode == 'local' ? 10 : 250);
41624         this.dqTask = new Roo.util.DelayedTask(this.initQuery, this);
41625         if(this.typeAhead){
41626             this.taTask = new Roo.util.DelayedTask(this.onTypeAhead, this);
41627         }
41628         if(this.editable !== false){
41629             this.el.on("keyup", this.onKeyUp, this);
41630         }
41631         if(this.forceSelection){
41632             this.on('blur', this.doForce, this);
41633         }
41634     },
41635
41636     onDestroy : function(){
41637         if(this.view){
41638             this.view.setStore(null);
41639             this.view.el.removeAllListeners();
41640             this.view.el.remove();
41641             this.view.purgeListeners();
41642         }
41643         if(this.list){
41644             this.list.destroy();
41645         }
41646         if(this.store){
41647             this.store.un('beforeload', this.onBeforeLoad, this);
41648             this.store.un('load', this.onLoad, this);
41649             this.store.un('loadexception', this.onLoadException, this);
41650         }
41651         Roo.form.ComboBox.superclass.onDestroy.call(this);
41652     },
41653
41654     // private
41655     fireKey : function(e){
41656         if(e.isNavKeyPress() && !this.list.isVisible()){
41657             this.fireEvent("specialkey", this, e);
41658         }
41659     },
41660
41661     // private
41662     onResize: function(w, h){
41663         Roo.form.ComboBox.superclass.onResize.apply(this, arguments);
41664         
41665         if(typeof w != 'number'){
41666             // we do not handle it!?!?
41667             return;
41668         }
41669         var tw = this.trigger.getWidth();
41670         tw += this.addicon ? this.addicon.getWidth() : 0;
41671         tw += this.editicon ? this.editicon.getWidth() : 0;
41672         var x = w - tw;
41673         this.el.setWidth( this.adjustWidth('input', x));
41674             
41675         this.trigger.setStyle('left', x+'px');
41676         
41677         if(this.list && this.listWidth === undefined){
41678             var lw = Math.max(x + this.trigger.getWidth(), this.minListWidth);
41679             this.list.setWidth(lw);
41680             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
41681         }
41682         
41683     
41684         
41685     },
41686
41687     /**
41688      * Allow or prevent the user from directly editing the field text.  If false is passed,
41689      * the user will only be able to select from the items defined in the dropdown list.  This method
41690      * is the runtime equivalent of setting the 'editable' config option at config time.
41691      * @param {Boolean} value True to allow the user to directly edit the field text
41692      */
41693     setEditable : function(value){
41694         if(value == this.editable){
41695             return;
41696         }
41697         this.editable = value;
41698         if(!value){
41699             this.el.dom.setAttribute('readOnly', true);
41700             this.el.on('mousedown', this.onTriggerClick,  this);
41701             this.el.addClass('x-combo-noedit');
41702         }else{
41703             this.el.dom.setAttribute('readOnly', false);
41704             this.el.un('mousedown', this.onTriggerClick,  this);
41705             this.el.removeClass('x-combo-noedit');
41706         }
41707     },
41708
41709     // private
41710     onBeforeLoad : function(){
41711         if(!this.hasFocus){
41712             return;
41713         }
41714         this.innerList.update(this.loadingText ?
41715                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
41716         this.restrictHeight();
41717         this.selectedIndex = -1;
41718     },
41719
41720     // private
41721     onLoad : function(){
41722         if(!this.hasFocus){
41723             return;
41724         }
41725         if(this.store.getCount() > 0){
41726             this.expand();
41727             this.restrictHeight();
41728             if(this.lastQuery == this.allQuery){
41729                 if(this.editable){
41730                     this.el.dom.select();
41731                 }
41732                 if(!this.selectByValue(this.value, true)){
41733                     this.select(0, true);
41734                 }
41735             }else{
41736                 this.selectNext();
41737                 if(this.typeAhead && this.lastKey != Roo.EventObject.BACKSPACE && this.lastKey != Roo.EventObject.DELETE){
41738                     this.taTask.delay(this.typeAheadDelay);
41739                 }
41740             }
41741         }else{
41742             this.onEmptyResults();
41743         }
41744         //this.el.focus();
41745     },
41746     // private
41747     onLoadException : function()
41748     {
41749         this.collapse();
41750         Roo.log(this.store.reader.jsonData);
41751         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
41752             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
41753         }
41754         
41755         
41756     },
41757     // private
41758     onTypeAhead : function(){
41759         if(this.store.getCount() > 0){
41760             var r = this.store.getAt(0);
41761             var newValue = r.data[this.displayField];
41762             var len = newValue.length;
41763             var selStart = this.getRawValue().length;
41764             if(selStart != len){
41765                 this.setRawValue(newValue);
41766                 this.selectText(selStart, newValue.length);
41767             }
41768         }
41769     },
41770
41771     // private
41772     onSelect : function(record, index){
41773         if(this.fireEvent('beforeselect', this, record, index) !== false){
41774             this.setFromData(index > -1 ? record.data : false);
41775             this.collapse();
41776             this.fireEvent('select', this, record, index);
41777         }
41778     },
41779
41780     /**
41781      * Returns the currently selected field value or empty string if no value is set.
41782      * @return {String} value The selected value
41783      */
41784     getValue : function(){
41785         if(this.valueField){
41786             return typeof this.value != 'undefined' ? this.value : '';
41787         }
41788         return Roo.form.ComboBox.superclass.getValue.call(this);
41789     },
41790
41791     /**
41792      * Clears any text/value currently set in the field
41793      */
41794     clearValue : function(){
41795         if(this.hiddenField){
41796             this.hiddenField.value = '';
41797         }
41798         this.value = '';
41799         this.setRawValue('');
41800         this.lastSelectionText = '';
41801         
41802     },
41803
41804     /**
41805      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
41806      * will be displayed in the field.  If the value does not match the data value of an existing item,
41807      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
41808      * Otherwise the field will be blank (although the value will still be set).
41809      * @param {String} value The value to match
41810      */
41811     setValue : function(v){
41812         var text = v;
41813         if(this.valueField){
41814             var r = this.findRecord(this.valueField, v);
41815             if(r){
41816                 text = r.data[this.displayField];
41817             }else if(this.valueNotFoundText !== undefined){
41818                 text = this.valueNotFoundText;
41819             }
41820         }
41821         this.lastSelectionText = text;
41822         if(this.hiddenField){
41823             this.hiddenField.value = v;
41824         }
41825         Roo.form.ComboBox.superclass.setValue.call(this, text);
41826         this.value = v;
41827     },
41828     /**
41829      * @property {Object} the last set data for the element
41830      */
41831     
41832     lastData : false,
41833     /**
41834      * Sets the value of the field based on a object which is related to the record format for the store.
41835      * @param {Object} value the value to set as. or false on reset?
41836      */
41837     setFromData : function(o){
41838         var dv = ''; // display value
41839         var vv = ''; // value value..
41840         this.lastData = o;
41841         if (this.displayField) {
41842             dv = !o || typeof(o[this.displayField]) == 'undefined' ? '' : o[this.displayField];
41843         } else {
41844             // this is an error condition!!!
41845             Roo.log('no  displayField value set for '+ (this.name ? this.name : this.id));
41846         }
41847         
41848         if(this.valueField){
41849             vv = !o || typeof(o[this.valueField]) == 'undefined' ? dv : o[this.valueField];
41850         }
41851         if(this.hiddenField){
41852             this.hiddenField.value = vv;
41853             
41854             this.lastSelectionText = dv;
41855             Roo.form.ComboBox.superclass.setValue.call(this, dv);
41856             this.value = vv;
41857             return;
41858         }
41859         // no hidden field.. - we store the value in 'value', but still display
41860         // display field!!!!
41861         this.lastSelectionText = dv;
41862         Roo.form.ComboBox.superclass.setValue.call(this, dv);
41863         this.value = vv;
41864         
41865         
41866     },
41867     // private
41868     reset : function(){
41869         // overridden so that last data is reset..
41870         this.setValue(this.resetValue);
41871         this.originalValue = this.getValue();
41872         this.clearInvalid();
41873         this.lastData = false;
41874         if (this.view) {
41875             this.view.clearSelections();
41876         }
41877     },
41878     // private
41879     findRecord : function(prop, value){
41880         var record;
41881         if(this.store.getCount() > 0){
41882             this.store.each(function(r){
41883                 if(r.data[prop] == value){
41884                     record = r;
41885                     return false;
41886                 }
41887                 return true;
41888             });
41889         }
41890         return record;
41891     },
41892     
41893     getName: function()
41894     {
41895         // returns hidden if it's set..
41896         if (!this.rendered) {return ''};
41897         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
41898         
41899     },
41900     // private
41901     onViewMove : function(e, t){
41902         this.inKeyMode = false;
41903     },
41904
41905     // private
41906     onViewOver : function(e, t){
41907         if(this.inKeyMode){ // prevent key nav and mouse over conflicts
41908             return;
41909         }
41910         var item = this.view.findItemFromChild(t);
41911         if(item){
41912             var index = this.view.indexOf(item);
41913             this.select(index, false);
41914         }
41915     },
41916
41917     // private
41918     onViewClick : function(doFocus)
41919     {
41920         var index = this.view.getSelectedIndexes()[0];
41921         var r = this.store.getAt(index);
41922         if(r){
41923             this.onSelect(r, index);
41924         }
41925         if(doFocus !== false && !this.blockFocus){
41926             this.el.focus();
41927         }
41928     },
41929
41930     // private
41931     restrictHeight : function(){
41932         this.innerList.dom.style.height = '';
41933         var inner = this.innerList.dom;
41934         var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
41935         this.innerList.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
41936         this.list.beginUpdate();
41937         this.list.setHeight(this.innerList.getHeight()+this.list.getFrameWidth('tb')+(this.resizable?this.handleHeight:0)+this.assetHeight);
41938         this.list.alignTo(this.el, this.listAlign);
41939         this.list.endUpdate();
41940     },
41941
41942     // private
41943     onEmptyResults : function(){
41944         this.collapse();
41945     },
41946
41947     /**
41948      * Returns true if the dropdown list is expanded, else false.
41949      */
41950     isExpanded : function(){
41951         return this.list.isVisible();
41952     },
41953
41954     /**
41955      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
41956      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41957      * @param {String} value The data value of the item to select
41958      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41959      * selected item if it is not currently in view (defaults to true)
41960      * @return {Boolean} True if the value matched an item in the list, else false
41961      */
41962     selectByValue : function(v, scrollIntoView){
41963         if(v !== undefined && v !== null){
41964             var r = this.findRecord(this.valueField || this.displayField, v);
41965             if(r){
41966                 this.select(this.store.indexOf(r), scrollIntoView);
41967                 return true;
41968             }
41969         }
41970         return false;
41971     },
41972
41973     /**
41974      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
41975      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
41976      * @param {Number} index The zero-based index of the list item to select
41977      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
41978      * selected item if it is not currently in view (defaults to true)
41979      */
41980     select : function(index, scrollIntoView){
41981         this.selectedIndex = index;
41982         this.view.select(index);
41983         if(scrollIntoView !== false){
41984             var el = this.view.getNode(index);
41985             if(el){
41986                 this.innerList.scrollChildIntoView(el, false);
41987             }
41988         }
41989     },
41990
41991     // private
41992     selectNext : function(){
41993         var ct = this.store.getCount();
41994         if(ct > 0){
41995             if(this.selectedIndex == -1){
41996                 this.select(0);
41997             }else if(this.selectedIndex < ct-1){
41998                 this.select(this.selectedIndex+1);
41999             }
42000         }
42001     },
42002
42003     // private
42004     selectPrev : function(){
42005         var ct = this.store.getCount();
42006         if(ct > 0){
42007             if(this.selectedIndex == -1){
42008                 this.select(0);
42009             }else if(this.selectedIndex != 0){
42010                 this.select(this.selectedIndex-1);
42011             }
42012         }
42013     },
42014
42015     // private
42016     onKeyUp : function(e){
42017         if(this.editable !== false && !e.isSpecialKey()){
42018             this.lastKey = e.getKey();
42019             this.dqTask.delay(this.queryDelay);
42020         }
42021     },
42022
42023     // private
42024     validateBlur : function(){
42025         return !this.list || !this.list.isVisible();   
42026     },
42027
42028     // private
42029     initQuery : function(){
42030         this.doQuery(this.getRawValue());
42031     },
42032
42033     // private
42034     doForce : function(){
42035         if(this.el.dom.value.length > 0){
42036             this.el.dom.value =
42037                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
42038              
42039         }
42040     },
42041
42042     /**
42043      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
42044      * query allowing the query action to be canceled if needed.
42045      * @param {String} query The SQL query to execute
42046      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
42047      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
42048      * saved in the current store (defaults to false)
42049      */
42050     doQuery : function(q, forceAll){
42051         if(q === undefined || q === null){
42052             q = '';
42053         }
42054         var qe = {
42055             query: q,
42056             forceAll: forceAll,
42057             combo: this,
42058             cancel:false
42059         };
42060         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
42061             return false;
42062         }
42063         q = qe.query;
42064         forceAll = qe.forceAll;
42065         if(forceAll === true || (q.length >= this.minChars)){
42066             if(this.lastQuery != q || this.alwaysQuery){
42067                 this.lastQuery = q;
42068                 if(this.mode == 'local'){
42069                     this.selectedIndex = -1;
42070                     if(forceAll){
42071                         this.store.clearFilter();
42072                     }else{
42073                         this.store.filter(this.displayField, q);
42074                     }
42075                     this.onLoad();
42076                 }else{
42077                     this.store.baseParams[this.queryParam] = q;
42078                     this.store.load({
42079                         params: this.getParams(q)
42080                     });
42081                     this.expand();
42082                 }
42083             }else{
42084                 this.selectedIndex = -1;
42085                 this.onLoad();   
42086             }
42087         }
42088     },
42089
42090     // private
42091     getParams : function(q){
42092         var p = {};
42093         //p[this.queryParam] = q;
42094         if(this.pageSize){
42095             p.start = 0;
42096             p.limit = this.pageSize;
42097         }
42098         return p;
42099     },
42100
42101     /**
42102      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
42103      */
42104     collapse : function(){
42105         if(!this.isExpanded()){
42106             return;
42107         }
42108         this.list.hide();
42109         Roo.get(document).un('mousedown', this.collapseIf, this);
42110         Roo.get(document).un('mousewheel', this.collapseIf, this);
42111         if (!this.editable) {
42112             Roo.get(document).un('keydown', this.listKeyPress, this);
42113         }
42114         this.fireEvent('collapse', this);
42115     },
42116
42117     // private
42118     collapseIf : function(e){
42119         if(!e.within(this.wrap) && !e.within(this.list)){
42120             this.collapse();
42121         }
42122     },
42123
42124     /**
42125      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
42126      */
42127     expand : function(){
42128         if(this.isExpanded() || !this.hasFocus){
42129             return;
42130         }
42131         this.list.alignTo(this.el, this.listAlign);
42132         this.list.show();
42133         Roo.get(document).on('mousedown', this.collapseIf, this);
42134         Roo.get(document).on('mousewheel', this.collapseIf, this);
42135         if (!this.editable) {
42136             Roo.get(document).on('keydown', this.listKeyPress, this);
42137         }
42138         
42139         this.fireEvent('expand', this);
42140     },
42141
42142     // private
42143     // Implements the default empty TriggerField.onTriggerClick function
42144     onTriggerClick : function(){
42145         if(this.disabled){
42146             return;
42147         }
42148         if(this.isExpanded()){
42149             this.collapse();
42150             if (!this.blockFocus) {
42151                 this.el.focus();
42152             }
42153             
42154         }else {
42155             this.hasFocus = true;
42156             if(this.triggerAction == 'all') {
42157                 this.doQuery(this.allQuery, true);
42158             } else {
42159                 this.doQuery(this.getRawValue());
42160             }
42161             if (!this.blockFocus) {
42162                 this.el.focus();
42163             }
42164         }
42165     },
42166     listKeyPress : function(e)
42167     {
42168         //Roo.log('listkeypress');
42169         // scroll to first matching element based on key pres..
42170         if (e.isSpecialKey()) {
42171             return false;
42172         }
42173         var k = String.fromCharCode(e.getKey()).toUpperCase();
42174         //Roo.log(k);
42175         var match  = false;
42176         var csel = this.view.getSelectedNodes();
42177         var cselitem = false;
42178         if (csel.length) {
42179             var ix = this.view.indexOf(csel[0]);
42180             cselitem  = this.store.getAt(ix);
42181             if (!cselitem.get(this.displayField) || cselitem.get(this.displayField).substring(0,1).toUpperCase() != k) {
42182                 cselitem = false;
42183             }
42184             
42185         }
42186         
42187         this.store.each(function(v) { 
42188             if (cselitem) {
42189                 // start at existing selection.
42190                 if (cselitem.id == v.id) {
42191                     cselitem = false;
42192                 }
42193                 return;
42194             }
42195                 
42196             if (v.get(this.displayField) && v.get(this.displayField).substring(0,1).toUpperCase() == k) {
42197                 match = this.store.indexOf(v);
42198                 return false;
42199             }
42200         }, this);
42201         
42202         if (match === false) {
42203             return true; // no more action?
42204         }
42205         // scroll to?
42206         this.view.select(match);
42207         var sn = Roo.get(this.view.getSelectedNodes()[0]);
42208         sn.scrollIntoView(sn.dom.parentNode, false);
42209     } 
42210
42211     /** 
42212     * @cfg {Boolean} grow 
42213     * @hide 
42214     */
42215     /** 
42216     * @cfg {Number} growMin 
42217     * @hide 
42218     */
42219     /** 
42220     * @cfg {Number} growMax 
42221     * @hide 
42222     */
42223     /**
42224      * @hide
42225      * @method autoSize
42226      */
42227 });/*
42228  * Copyright(c) 2010-2012, Roo J Solutions Limited
42229  *
42230  * Licence LGPL
42231  *
42232  */
42233
42234 /**
42235  * @class Roo.form.ComboBoxArray
42236  * @extends Roo.form.TextField
42237  * A facebook style adder... for lists of email / people / countries  etc...
42238  * pick multiple items from a combo box, and shows each one.
42239  *
42240  *  Fred [x]  Brian [x]  [Pick another |v]
42241  *
42242  *
42243  *  For this to work: it needs various extra information
42244  *    - normal combo problay has
42245  *      name, hiddenName
42246  *    + displayField, valueField
42247  *
42248  *    For our purpose...
42249  *
42250  *
42251  *   If we change from 'extends' to wrapping...
42252  *   
42253  *  
42254  *
42255  
42256  
42257  * @constructor
42258  * Create a new ComboBoxArray.
42259  * @param {Object} config Configuration options
42260  */
42261  
42262
42263 Roo.form.ComboBoxArray = function(config)
42264 {
42265     this.addEvents({
42266         /**
42267          * @event beforeremove
42268          * Fires before remove the value from the list
42269              * @param {Roo.form.ComboBoxArray} _self This combo box array
42270              * @param {Roo.form.ComboBoxArray.Item} item removed item
42271              */
42272         'beforeremove' : true,
42273         /**
42274          * @event remove
42275          * Fires when remove the value from the list
42276              * @param {Roo.form.ComboBoxArray} _self This combo box array
42277              * @param {Roo.form.ComboBoxArray.Item} item removed item
42278              */
42279         'remove' : true
42280         
42281         
42282     });
42283     
42284     Roo.form.ComboBoxArray.superclass.constructor.call(this, config);
42285     
42286     this.items = new Roo.util.MixedCollection(false);
42287     
42288     // construct the child combo...
42289     
42290     
42291     
42292     
42293    
42294     
42295 }
42296
42297  
42298 Roo.extend(Roo.form.ComboBoxArray, Roo.form.TextField,
42299
42300     /**
42301      * @cfg {Roo.form.Combo} combo The combo box that is wrapped
42302      */
42303     
42304     lastData : false,
42305     
42306     // behavies liek a hiddne field
42307     inputType:      'hidden',
42308     /**
42309      * @cfg {Number} width The width of the box that displays the selected element
42310      */ 
42311     width:          300,
42312
42313     
42314     
42315     /**
42316      * @cfg {String} name    The name of the visable items on this form (eg. titles not ids)
42317      */
42318     name : false,
42319     /**
42320      * @cfg {String} hiddenName    The hidden name of the field, often contains an comma seperated list of names
42321      */
42322     hiddenName : false,
42323     
42324     
42325     // private the array of items that are displayed..
42326     items  : false,
42327     // private - the hidden field el.
42328     hiddenEl : false,
42329     // private - the filed el..
42330     el : false,
42331     
42332     //validateValue : function() { return true; }, // all values are ok!
42333     //onAddClick: function() { },
42334     
42335     onRender : function(ct, position) 
42336     {
42337         
42338         // create the standard hidden element
42339         //Roo.form.ComboBoxArray.superclass.onRender.call(this, ct, position);
42340         
42341         
42342         // give fake names to child combo;
42343         this.combo.hiddenName = this.hiddenName ? (this.hiddenName+'-subcombo') : this.hiddenName;
42344         this.combo.name = this.name ? (this.name+'-subcombo') : this.name;
42345         
42346         this.combo = Roo.factory(this.combo, Roo.form);
42347         this.combo.onRender(ct, position);
42348         if (typeof(this.combo.width) != 'undefined') {
42349             this.combo.onResize(this.combo.width,0);
42350         }
42351         
42352         this.combo.initEvents();
42353         
42354         // assigned so form know we need to do this..
42355         this.store          = this.combo.store;
42356         this.valueField     = this.combo.valueField;
42357         this.displayField   = this.combo.displayField ;
42358         
42359         
42360         this.combo.wrap.addClass('x-cbarray-grp');
42361         
42362         var cbwrap = this.combo.wrap.createChild(
42363             {tag: 'div', cls: 'x-cbarray-cb'},
42364             this.combo.el.dom
42365         );
42366         
42367              
42368         this.hiddenEl = this.combo.wrap.createChild({
42369             tag: 'input',  type:'hidden' , name: this.hiddenName, value : ''
42370         });
42371         this.el = this.combo.wrap.createChild({
42372             tag: 'input',  type:'hidden' , name: this.name, value : ''
42373         });
42374          //   this.el.dom.removeAttribute("name");
42375         
42376         
42377         this.outerWrap = this.combo.wrap;
42378         this.wrap = cbwrap;
42379         
42380         this.outerWrap.setWidth(this.width);
42381         this.outerWrap.dom.removeChild(this.el.dom);
42382         
42383         this.wrap.dom.appendChild(this.el.dom);
42384         this.outerWrap.dom.removeChild(this.combo.trigger.dom);
42385         this.combo.wrap.dom.appendChild(this.combo.trigger.dom);
42386         
42387         this.combo.trigger.setStyle('position','relative');
42388         this.combo.trigger.setStyle('left', '0px');
42389         this.combo.trigger.setStyle('top', '2px');
42390         
42391         this.combo.el.setStyle('vertical-align', 'text-bottom');
42392         
42393         //this.trigger.setStyle('vertical-align', 'top');
42394         
42395         // this should use the code from combo really... on('add' ....)
42396         if (this.adder) {
42397             
42398         
42399             this.adder = this.outerWrap.createChild(
42400                 {tag: 'img', src: Roo.BLANK_IMAGE_URL, cls: 'x-form-adder', style: 'margin-left:2px'});  
42401             var _t = this;
42402             this.adder.on('click', function(e) {
42403                 _t.fireEvent('adderclick', this, e);
42404             }, _t);
42405         }
42406         //var _t = this;
42407         //this.adder.on('click', this.onAddClick, _t);
42408         
42409         
42410         this.combo.on('select', function(cb, rec, ix) {
42411             this.addItem(rec.data);
42412             
42413             cb.setValue('');
42414             cb.el.dom.value = '';
42415             //cb.lastData = rec.data;
42416             // add to list
42417             
42418         }, this);
42419         
42420         
42421     },
42422     
42423     
42424     getName: function()
42425     {
42426         // returns hidden if it's set..
42427         if (!this.rendered) {return ''};
42428         return  this.hiddenName ? this.hiddenName : this.name;
42429         
42430     },
42431     
42432     
42433     onResize: function(w, h){
42434         
42435         return;
42436         // not sure if this is needed..
42437         //this.combo.onResize(w,h);
42438         
42439         if(typeof w != 'number'){
42440             // we do not handle it!?!?
42441             return;
42442         }
42443         var tw = this.combo.trigger.getWidth();
42444         tw += this.addicon ? this.addicon.getWidth() : 0;
42445         tw += this.editicon ? this.editicon.getWidth() : 0;
42446         var x = w - tw;
42447         this.combo.el.setWidth( this.combo.adjustWidth('input', x));
42448             
42449         this.combo.trigger.setStyle('left', '0px');
42450         
42451         if(this.list && this.listWidth === undefined){
42452             var lw = Math.max(x + this.combo.trigger.getWidth(), this.combo.minListWidth);
42453             this.list.setWidth(lw);
42454             this.innerList.setWidth(lw - this.list.getFrameWidth('lr'));
42455         }
42456         
42457     
42458         
42459     },
42460     
42461     addItem: function(rec)
42462     {
42463         var valueField = this.combo.valueField;
42464         var displayField = this.combo.displayField;
42465         
42466         if (this.items.indexOfKey(rec[valueField]) > -1) {
42467             //console.log("GOT " + rec.data.id);
42468             return;
42469         }
42470         
42471         var x = new Roo.form.ComboBoxArray.Item({
42472             //id : rec[this.idField],
42473             data : rec,
42474             displayField : displayField ,
42475             tipField : displayField ,
42476             cb : this
42477         });
42478         // use the 
42479         this.items.add(rec[valueField],x);
42480         // add it before the element..
42481         this.updateHiddenEl();
42482         x.render(this.outerWrap, this.wrap.dom);
42483         // add the image handler..
42484     },
42485     
42486     updateHiddenEl : function()
42487     {
42488         this.validate();
42489         if (!this.hiddenEl) {
42490             return;
42491         }
42492         var ar = [];
42493         var idField = this.combo.valueField;
42494         
42495         this.items.each(function(f) {
42496             ar.push(f.data[idField]);
42497         });
42498         this.hiddenEl.dom.value = ar.join(',');
42499         this.validate();
42500     },
42501     
42502     reset : function()
42503     {
42504         this.items.clear();
42505         
42506         Roo.each(this.outerWrap.select('.x-cbarray-item', true).elements, function(el){
42507            el.remove();
42508         });
42509         
42510         this.el.dom.value = '';
42511         if (this.hiddenEl) {
42512             this.hiddenEl.dom.value = '';
42513         }
42514         
42515     },
42516     getValue: function()
42517     {
42518         return this.hiddenEl ? this.hiddenEl.dom.value : '';
42519     },
42520     setValue: function(v) // not a valid action - must use addItems..
42521     {
42522         
42523         this.reset();
42524          
42525         if (this.store.isLocal && (typeof(v) == 'string')) {
42526             // then we can use the store to find the values..
42527             // comma seperated at present.. this needs to allow JSON based encoding..
42528             this.hiddenEl.value  = v;
42529             var v_ar = [];
42530             Roo.each(v.split(','), function(k) {
42531                 Roo.log("CHECK " + this.valueField + ',' + k);
42532                 var li = this.store.query(this.valueField, k);
42533                 if (!li.length) {
42534                     return;
42535                 }
42536                 var add = {};
42537                 add[this.valueField] = k;
42538                 add[this.displayField] = li.item(0).data[this.displayField];
42539                 
42540                 this.addItem(add);
42541             }, this) 
42542              
42543         }
42544         if (typeof(v) == 'object' ) {
42545             // then let's assume it's an array of objects..
42546             Roo.each(v, function(l) {
42547                 this.addItem(l);
42548             }, this);
42549              
42550         }
42551         
42552         
42553     },
42554     setFromData: function(v)
42555     {
42556         // this recieves an object, if setValues is called.
42557         this.reset();
42558         this.el.dom.value = v[this.displayField];
42559         this.hiddenEl.dom.value = v[this.valueField];
42560         if (typeof(v[this.valueField]) != 'string' || !v[this.valueField].length) {
42561             return;
42562         }
42563         var kv = v[this.valueField];
42564         var dv = v[this.displayField];
42565         kv = typeof(kv) != 'string' ? '' : kv;
42566         dv = typeof(dv) != 'string' ? '' : dv;
42567         
42568         
42569         var keys = kv.split(',');
42570         var display = dv.split(',');
42571         for (var i = 0 ; i < keys.length; i++) {
42572             
42573             add = {};
42574             add[this.valueField] = keys[i];
42575             add[this.displayField] = display[i];
42576             this.addItem(add);
42577         }
42578       
42579         
42580     },
42581     
42582     /**
42583      * Validates the combox array value
42584      * @return {Boolean} True if the value is valid, else false
42585      */
42586     validate : function(){
42587         if(this.disabled || this.validateValue(this.processValue(this.getValue()))){
42588             this.clearInvalid();
42589             return true;
42590         }
42591         return false;
42592     },
42593     
42594     validateValue : function(value){
42595         return Roo.form.ComboBoxArray.superclass.validateValue.call(this, this.getValue());
42596         
42597     },
42598     
42599     /*@
42600      * overide
42601      * 
42602      */
42603     isDirty : function() {
42604         if(this.disabled) {
42605             return false;
42606         }
42607         
42608         try {
42609             var d = Roo.decode(String(this.originalValue));
42610         } catch (e) {
42611             return String(this.getValue()) !== String(this.originalValue);
42612         }
42613         
42614         var originalValue = [];
42615         
42616         for (var i = 0; i < d.length; i++){
42617             originalValue.push(d[i][this.valueField]);
42618         }
42619         
42620         return String(this.getValue()) !== String(originalValue.join(','));
42621         
42622     }
42623     
42624 });
42625
42626
42627
42628 /**
42629  * @class Roo.form.ComboBoxArray.Item
42630  * @extends Roo.BoxComponent
42631  * A selected item in the list
42632  *  Fred [x]  Brian [x]  [Pick another |v]
42633  * 
42634  * @constructor
42635  * Create a new item.
42636  * @param {Object} config Configuration options
42637  */
42638  
42639 Roo.form.ComboBoxArray.Item = function(config) {
42640     config.id = Roo.id();
42641     Roo.form.ComboBoxArray.Item.superclass.constructor.call(this, config);
42642 }
42643
42644 Roo.extend(Roo.form.ComboBoxArray.Item, Roo.BoxComponent, {
42645     data : {},
42646     cb: false,
42647     displayField : false,
42648     tipField : false,
42649     
42650     
42651     defaultAutoCreate : {
42652         tag: 'div',
42653         cls: 'x-cbarray-item',
42654         cn : [ 
42655             { tag: 'div' },
42656             {
42657                 tag: 'img',
42658                 width:16,
42659                 height : 16,
42660                 src : Roo.BLANK_IMAGE_URL ,
42661                 align: 'center'
42662             }
42663         ]
42664         
42665     },
42666     
42667  
42668     onRender : function(ct, position)
42669     {
42670         Roo.form.Field.superclass.onRender.call(this, ct, position);
42671         
42672         if(!this.el){
42673             var cfg = this.getAutoCreate();
42674             this.el = ct.createChild(cfg, position);
42675         }
42676         
42677         this.el.child('img').dom.setAttribute('src', Roo.BLANK_IMAGE_URL);
42678         
42679         this.el.child('div').dom.innerHTML = this.cb.renderer ? 
42680             this.cb.renderer(this.data) :
42681             String.format('{0}',this.data[this.displayField]);
42682         
42683             
42684         this.el.child('div').dom.setAttribute('qtip',
42685                         String.format('{0}',this.data[this.tipField])
42686         );
42687         
42688         this.el.child('img').on('click', this.remove, this);
42689         
42690     },
42691    
42692     remove : function()
42693     {
42694         if(this.cb.disabled){
42695             return;
42696         }
42697         
42698         if(false !== this.cb.fireEvent('beforeremove', this.cb, this)){
42699             this.cb.items.remove(this);
42700             this.el.child('img').un('click', this.remove, this);
42701             this.el.remove();
42702             this.cb.updateHiddenEl();
42703
42704             this.cb.fireEvent('remove', this.cb, this);
42705         }
42706         
42707     }
42708 });/*
42709  * RooJS Library 1.1.1
42710  * Copyright(c) 2008-2011  Alan Knowles
42711  *
42712  * License - LGPL
42713  */
42714  
42715
42716 /**
42717  * @class Roo.form.ComboNested
42718  * @extends Roo.form.ComboBox
42719  * A combobox for that allows selection of nested items in a list,
42720  * eg.
42721  *
42722  *  Book
42723  *    -> red
42724  *    -> green
42725  *  Table
42726  *    -> square
42727  *      ->red
42728  *      ->green
42729  *    -> rectangle
42730  *      ->green
42731  *      
42732  * 
42733  * @constructor
42734  * Create a new ComboNested
42735  * @param {Object} config Configuration options
42736  */
42737 Roo.form.ComboNested = function(config){
42738     Roo.form.ComboCheck.superclass.constructor.call(this, config);
42739     // should verify some data...
42740     // like
42741     // hiddenName = required..
42742     // displayField = required
42743     // valudField == required
42744     var req= [ 'hiddenName', 'displayField', 'valueField' ];
42745     var _t = this;
42746     Roo.each(req, function(e) {
42747         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
42748             throw "Roo.form.ComboNested : missing value for: " + e;
42749         }
42750     });
42751      
42752     
42753 };
42754
42755 Roo.extend(Roo.form.ComboNested, Roo.form.ComboBox, {
42756    
42757     /*
42758      * @config {Number} max Number of columns to show
42759      */
42760     
42761     maxColumns : 3,
42762    
42763     list : null, // the outermost div..
42764     innerLists : null, // the
42765     views : null,
42766     stores : null,
42767     // private
42768     onRender : function(ct, position)
42769     {
42770         Roo.form.ComboBox.superclass.onRender.call(this, ct, position); // skip parent call - got to above..
42771         
42772         if(this.hiddenName){
42773             this.hiddenField = this.el.insertSibling({tag:'input', type:'hidden', name: this.hiddenName, id:  (this.hiddenId||this.hiddenName)},
42774                     'before', true);
42775             this.hiddenField.value =
42776                 this.hiddenValue !== undefined ? this.hiddenValue :
42777                 this.value !== undefined ? this.value : '';
42778
42779             // prevent input submission
42780             this.el.dom.removeAttribute('name');
42781              
42782              
42783         }
42784         
42785         if(Roo.isGecko){
42786             this.el.dom.setAttribute('autocomplete', 'off');
42787         }
42788
42789         var cls = 'x-combo-list';
42790
42791         this.list = new Roo.Layer({
42792             shadow: this.shadow, cls: [cls, this.listClass].join(' '), constrain:false
42793         });
42794
42795         var lw = this.listWidth || Math.max(this.wrap.getWidth(), this.minListWidth);
42796         this.list.setWidth(lw);
42797         this.list.swallowEvent('mousewheel');
42798         this.assetHeight = 0;
42799
42800         if(this.title){
42801             this.header = this.list.createChild({cls:cls+'-hd', html: this.title});
42802             this.assetHeight += this.header.getHeight();
42803         }
42804         this.innerLists = [];
42805         this.views = [];
42806         this.stores = [];
42807         for (var i =0 ; i < this.maxColumns; i++) {
42808             this.onRenderList( cls, i);
42809         }
42810         
42811         // always needs footer, as we are going to have an 'OK' button.
42812         this.footer = this.list.createChild({cls:cls+'-ft'});
42813         this.pageTb = new Roo.Toolbar(this.footer);  
42814         var _this = this;
42815         this.pageTb.add(  {
42816             
42817             text: 'Done',
42818             handler: function()
42819             {
42820                 _this.collapse();
42821             }
42822         });
42823         
42824         if ( this.allowBlank && !this.disableClear) {
42825             
42826             this.pageTb.add(new Roo.Toolbar.Fill(), {
42827                 cls: 'x-btn-icon x-btn-clear',
42828                 text: '&#160;',
42829                 handler: function()
42830                 {
42831                     _this.collapse();
42832                     _this.clearValue();
42833                     _this.onSelect(false, -1);
42834                 }
42835             });
42836         }
42837         if (this.footer) {
42838             this.assetHeight += this.footer.getHeight();
42839         }
42840         
42841     },
42842     onRenderList : function (  cls, i)
42843     {
42844         
42845         var lw = Math.floor(
42846                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
42847         );
42848         
42849         this.list.setWidth(lw); // default to '1'
42850
42851         var il = this.innerLists[i] = this.list.createChild({cls:cls+'-inner'});
42852         //il.on('mouseover', this.onViewOver, this, { list:  i });
42853         //il.on('mousemove', this.onViewMove, this, { list:  i });
42854         il.setWidth(lw);
42855         il.setStyle({ 'overflow-x' : 'hidden'});
42856
42857         if(!this.tpl){
42858             this.tpl = new Roo.Template({
42859                 html :  '<div class="'+cls+'-item '+cls+'-item-{cn:this.isEmpty}">{' + this.displayField + '}</div>',
42860                 isEmpty: function (value, allValues) {
42861                     //Roo.log(value);
42862                     var dl = typeof(value.data) != 'undefined' ? value.data.length : value.length; ///json is a nested response..
42863                     return dl ? 'has-children' : 'no-children'
42864                 }
42865             });
42866         }
42867         
42868         var store  = this.store;
42869         if (i > 0) {
42870             store  = new Roo.data.SimpleStore({
42871                 //fields : this.store.reader.meta.fields,
42872                 reader : this.store.reader,
42873                 data : [ ]
42874             });
42875         }
42876         this.stores[i]  = store;
42877                 
42878         
42879         
42880         var view = this.views[i] = new Roo.View(
42881             il,
42882             this.tpl,
42883             {
42884                 singleSelect:true,
42885                 store: store,
42886                 selectedClass: this.selectedClass
42887             }
42888         );
42889         view.getEl().setWidth(lw);
42890         view.getEl().setStyle({
42891             position: i < 1 ? 'relative' : 'absolute',
42892             top: 0,
42893             left: (i * lw ) + 'px',
42894             display : i > 0 ? 'none' : 'block'
42895         });
42896         view.on('selectionchange', this.onSelectChange, this, {list : i });
42897         view.on('dblclick', this.onDoubleClick, this, {list : i });
42898         //view.on('click', this.onViewClick, this, { list : i });
42899
42900         store.on('beforeload', this.onBeforeLoad, this);
42901         store.on('load',  this.onLoad, this, { list  : i});
42902         store.on('loadexception', this.onLoadException, this);
42903
42904         // hide the other vies..
42905         
42906         
42907         
42908     },
42909     onResize : function()  {},
42910     
42911     restrictHeight : function()
42912     {
42913         var mh = 0;
42914         Roo.each(this.innerLists, function(il,i) {
42915             var el = this.views[i].getEl();
42916             el.dom.style.height = '';
42917             var inner = el.dom;
42918             var h = Math.max(inner.clientHeight, inner.offsetHeight, inner.scrollHeight);
42919             // only adjust heights on other ones..
42920             if (i < 1) {
42921                 
42922                 el.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42923                 il.setHeight(h < this.maxHeight ? 'auto' : this.maxHeight);
42924                 mh = Math.max(el.getHeight(), mh);
42925             }
42926             
42927             
42928         }, this);
42929         
42930         this.list.beginUpdate();
42931         this.list.setHeight(mh+this.list.getFrameWidth('tb')+this.assetHeight);
42932         this.list.alignTo(this.el, this.listAlign);
42933         this.list.endUpdate();
42934         
42935     },
42936      
42937     
42938     // -- store handlers..
42939     // private
42940     onBeforeLoad : function()
42941     {
42942         if(!this.hasFocus){
42943             return;
42944         }
42945         this.innerLists[0].update(this.loadingText ?
42946                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
42947         this.restrictHeight();
42948         this.selectedIndex = -1;
42949     },
42950     // private
42951     onLoad : function(a,b,c,d)
42952     {
42953         
42954         if(!this.hasFocus){
42955             return;
42956         }
42957         
42958         if(this.store.getCount() > 0) {
42959             this.expand();
42960             this.restrictHeight();   
42961         } else {
42962             this.onEmptyResults();
42963         }
42964         /*
42965         this.stores[1].loadData([]);
42966         this.stores[2].loadData([]);
42967         this.views
42968         */    
42969     
42970         //this.el.focus();
42971     },
42972     
42973     
42974     // private
42975     onLoadException : function()
42976     {
42977         this.collapse();
42978         Roo.log(this.store.reader.jsonData);
42979         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
42980             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
42981         }
42982         
42983         
42984     } ,
42985      
42986      
42987
42988     onSelectChange : function (view, sels, opts )
42989     {
42990         var ix = view.getSelectedIndexes();
42991         
42992         
42993         if (opts.list > this.maxColumns - 2) {
42994              
42995             this.setFromData(ix.length ? view.store.getAt(ix[0]).data : {});
42996             return;
42997         }
42998         
42999         if (!ix.length) {
43000             this.setFromData({});
43001             var str = this.stores[opts.list+1];
43002             str.removeAll();
43003             return;
43004         }
43005         
43006         var rec = view.store.getAt(ix[0]);
43007         this.setFromData(rec.data);
43008         
43009         var lw = Math.floor(
43010                 ((this.listWidth * this.maxColumns || Math.max(this.wrap.getWidth(), this.minListWidth)) - this.list.getFrameWidth('lr')) / this.maxColumns
43011         );
43012         
43013         this.stores[opts.list+1].loadDataFromChildren( rec );
43014         var dl = this.stores[opts.list+1]. getTotalCount();
43015         this.views[opts.list+1].getEl().setHeight( this.innerLists[0].getHeight());
43016         this.views[opts.list+1].getEl().setStyle({ display : dl ? 'block' : 'none' });
43017         this.innerLists[opts.list+1].setHeight( this.innerLists[0].getHeight());
43018         this.list.setWidth(lw * (opts.list + (dl ? 2 : 1))); 
43019     },
43020     onDoubleClick : function()
43021     {
43022         this.collapse(); //??
43023     },
43024     
43025      
43026     
43027     findRecord : function (prop,value)
43028     {
43029         return this.findRecordInStore(this.store, prop,value);
43030     },
43031     
43032     // private
43033     findRecordInStore : function(store, prop, value)
43034     {
43035         var cstore = new Roo.data.SimpleStore({
43036             //fields : this.store.reader.meta.fields, // we need array reader.. for
43037             reader : this.store.reader,
43038             data : [ ]
43039         });
43040         var _this = this;
43041         var record  = false;
43042         if(store.getCount() > 0){
43043            store.each(function(r){
43044                 if(r.data[prop] == value){
43045                     record = r;
43046                     return false;
43047                 }
43048                 if (r.data.cn && r.data.cn.length) {
43049                     cstore.loadDataFromChildren( r);
43050                     var cret = _this.findRecordInStore(cstore, prop, value);
43051                     if (cret !== false) {
43052                         record = cret;
43053                         return false;
43054                     }
43055                 }
43056                 
43057                 return true;
43058             });
43059         }
43060         return record;
43061     }
43062     
43063     
43064     
43065     
43066 });/*
43067  * Based on:
43068  * Ext JS Library 1.1.1
43069  * Copyright(c) 2006-2007, Ext JS, LLC.
43070  *
43071  * Originally Released Under LGPL - original licence link has changed is not relivant.
43072  *
43073  * Fork - LGPL
43074  * <script type="text/javascript">
43075  */
43076 /**
43077  * @class Roo.form.Checkbox
43078  * @extends Roo.form.Field
43079  * Single checkbox field.  Can be used as a direct replacement for traditional checkbox fields.
43080  * @constructor
43081  * Creates a new Checkbox
43082  * @param {Object} config Configuration options
43083  */
43084 Roo.form.Checkbox = function(config){
43085     Roo.form.Checkbox.superclass.constructor.call(this, config);
43086     this.addEvents({
43087         /**
43088          * @event check
43089          * Fires when the checkbox is checked or unchecked.
43090              * @param {Roo.form.Checkbox} this This checkbox
43091              * @param {Boolean} checked The new checked value
43092              */
43093         check : true
43094     });
43095 };
43096
43097 Roo.extend(Roo.form.Checkbox, Roo.form.Field,  {
43098     /**
43099      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
43100      */
43101     focusClass : undefined,
43102     /**
43103      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
43104      */
43105     fieldClass: "x-form-field",
43106     /**
43107      * @cfg {Boolean} checked True if the the checkbox should render already checked (defaults to false)
43108      */
43109     checked: false,
43110     /**
43111      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
43112      * {tag: "input", type: "checkbox", autocomplete: "off"})
43113      */
43114     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "off"},
43115     /**
43116      * @cfg {String} boxLabel The text that appears beside the checkbox
43117      */
43118     boxLabel : "",
43119     /**
43120      * @cfg {String} inputValue The value that should go into the generated input element's value attribute
43121      */  
43122     inputValue : '1',
43123     /**
43124      * @cfg {String} valueOff The value that should go into the generated input element's value when unchecked.
43125      */
43126      valueOff: '0', // value when not checked..
43127
43128     actionMode : 'viewEl', 
43129     //
43130     // private
43131     itemCls : 'x-menu-check-item x-form-item',
43132     groupClass : 'x-menu-group-item',
43133     inputType : 'hidden',
43134     
43135     
43136     inSetChecked: false, // check that we are not calling self...
43137     
43138     inputElement: false, // real input element?
43139     basedOn: false, // ????
43140     
43141     isFormField: true, // not sure where this is needed!!!!
43142
43143     onResize : function(){
43144         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
43145         if(!this.boxLabel){
43146             this.el.alignTo(this.wrap, 'c-c');
43147         }
43148     },
43149
43150     initEvents : function(){
43151         Roo.form.Checkbox.superclass.initEvents.call(this);
43152         this.el.on("click", this.onClick,  this);
43153         this.el.on("change", this.onClick,  this);
43154     },
43155
43156
43157     getResizeEl : function(){
43158         return this.wrap;
43159     },
43160
43161     getPositionEl : function(){
43162         return this.wrap;
43163     },
43164
43165     // private
43166     onRender : function(ct, position){
43167         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43168         /*
43169         if(this.inputValue !== undefined){
43170             this.el.dom.value = this.inputValue;
43171         }
43172         */
43173         //this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43174         this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43175         var viewEl = this.wrap.createChild({ 
43176             tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43177         this.viewEl = viewEl;   
43178         this.wrap.on('click', this.onClick,  this); 
43179         
43180         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43181         this.el.on('propertychange', this.setFromHidden,  this);  //ie
43182         
43183         
43184         
43185         if(this.boxLabel){
43186             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43187         //    viewEl.on('click', this.onClick,  this); 
43188         }
43189         //if(this.checked){
43190             this.setChecked(this.checked);
43191         //}else{
43192             //this.checked = this.el.dom;
43193         //}
43194
43195     },
43196
43197     // private
43198     initValue : Roo.emptyFn,
43199
43200     /**
43201      * Returns the checked state of the checkbox.
43202      * @return {Boolean} True if checked, else false
43203      */
43204     getValue : function(){
43205         if(this.el){
43206             return String(this.el.dom.value) == String(this.inputValue ) ? this.inputValue : this.valueOff;
43207         }
43208         return this.valueOff;
43209         
43210     },
43211
43212         // private
43213     onClick : function(){ 
43214         if (this.disabled) {
43215             return;
43216         }
43217         this.setChecked(!this.checked);
43218
43219         //if(this.el.dom.checked != this.checked){
43220         //    this.setValue(this.el.dom.checked);
43221        // }
43222     },
43223
43224     /**
43225      * Sets the checked state of the checkbox.
43226      * On is always based on a string comparison between inputValue and the param.
43227      * @param {Boolean/String} value - the value to set 
43228      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
43229      */
43230     setValue : function(v,suppressEvent){
43231         
43232         
43233         //this.checked = (v === true || v === 'true' || v == '1' || String(v).toLowerCase() == 'on');
43234         //if(this.el && this.el.dom){
43235         //    this.el.dom.checked = this.checked;
43236         //    this.el.dom.defaultChecked = this.checked;
43237         //}
43238         this.setChecked(String(v) === String(this.inputValue), suppressEvent);
43239         //this.fireEvent("check", this, this.checked);
43240     },
43241     // private..
43242     setChecked : function(state,suppressEvent)
43243     {
43244         if (this.inSetChecked) {
43245             this.checked = state;
43246             return;
43247         }
43248         
43249     
43250         if(this.wrap){
43251             this.wrap[state ? 'addClass' : 'removeClass']('x-menu-item-checked');
43252         }
43253         this.checked = state;
43254         if(suppressEvent !== true){
43255             this.fireEvent('check', this, state);
43256         }
43257         this.inSetChecked = true;
43258         this.el.dom.value = state ? this.inputValue : this.valueOff;
43259         this.inSetChecked = false;
43260         
43261     },
43262     // handle setting of hidden value by some other method!!?!?
43263     setFromHidden: function()
43264     {
43265         if(!this.el){
43266             return;
43267         }
43268         //console.log("SET FROM HIDDEN");
43269         //alert('setFrom hidden');
43270         this.setValue(this.el.dom.value);
43271     },
43272     
43273     onDestroy : function()
43274     {
43275         if(this.viewEl){
43276             Roo.get(this.viewEl).remove();
43277         }
43278          
43279         Roo.form.Checkbox.superclass.onDestroy.call(this);
43280     },
43281     
43282     setBoxLabel : function(str)
43283     {
43284         this.wrap.select('.x-form-cb-label', true).first().dom.innerHTML = str;
43285     }
43286
43287 });/*
43288  * Based on:
43289  * Ext JS Library 1.1.1
43290  * Copyright(c) 2006-2007, Ext JS, LLC.
43291  *
43292  * Originally Released Under LGPL - original licence link has changed is not relivant.
43293  *
43294  * Fork - LGPL
43295  * <script type="text/javascript">
43296  */
43297  
43298 /**
43299  * @class Roo.form.Radio
43300  * @extends Roo.form.Checkbox
43301  * Single radio field.  Same as Checkbox, but provided as a convenience for automatically setting the input type.
43302  * Radio grouping is handled automatically by the browser if you give each radio in a group the same name.
43303  * @constructor
43304  * Creates a new Radio
43305  * @param {Object} config Configuration options
43306  */
43307 Roo.form.Radio = function(){
43308     Roo.form.Radio.superclass.constructor.apply(this, arguments);
43309 };
43310 Roo.extend(Roo.form.Radio, Roo.form.Checkbox, {
43311     inputType: 'radio',
43312
43313     /**
43314      * If this radio is part of a group, it will return the selected value
43315      * @return {String}
43316      */
43317     getGroupValue : function(){
43318         return this.el.up('form').child('input[name='+this.el.dom.name+']:checked', true).value;
43319     },
43320     
43321     
43322     onRender : function(ct, position){
43323         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
43324         
43325         if(this.inputValue !== undefined){
43326             this.el.dom.value = this.inputValue;
43327         }
43328          
43329         this.wrap = this.el.wrap({cls: "x-form-check-wrap"});
43330         //this.wrap = this.el.wrap({cls: 'x-menu-check-item '});
43331         //var viewEl = this.wrap.createChild({ 
43332         //    tag: 'img', cls: 'x-menu-item-icon', style: 'margin: 0px;' ,src : Roo.BLANK_IMAGE_URL });
43333         //this.viewEl = viewEl;   
43334         //this.wrap.on('click', this.onClick,  this); 
43335         
43336         //this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
43337         //this.el.on('propertychange', this.setFromHidden,  this);  //ie
43338         
43339         
43340         
43341         if(this.boxLabel){
43342             this.wrap.createChild({tag: 'label', htmlFor: this.el.id, cls: 'x-form-cb-label', html: this.boxLabel});
43343         //    viewEl.on('click', this.onClick,  this); 
43344         }
43345          if(this.checked){
43346             this.el.dom.checked =   'checked' ;
43347         }
43348          
43349     } 
43350     
43351     
43352 });//<script type="text/javascript">
43353
43354 /*
43355  * Based  Ext JS Library 1.1.1
43356  * Copyright(c) 2006-2007, Ext JS, LLC.
43357  * LGPL
43358  *
43359  */
43360  
43361 /**
43362  * @class Roo.HtmlEditorCore
43363  * @extends Roo.Component
43364  * Provides a the editing component for the HTML editors in Roo. (bootstrap and Roo.form)
43365  *
43366  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
43367  */
43368
43369 Roo.HtmlEditorCore = function(config){
43370     
43371     
43372     Roo.HtmlEditorCore.superclass.constructor.call(this, config);
43373     
43374     
43375     this.addEvents({
43376         /**
43377          * @event initialize
43378          * Fires when the editor is fully initialized (including the iframe)
43379          * @param {Roo.HtmlEditorCore} this
43380          */
43381         initialize: true,
43382         /**
43383          * @event activate
43384          * Fires when the editor is first receives the focus. Any insertion must wait
43385          * until after this event.
43386          * @param {Roo.HtmlEditorCore} this
43387          */
43388         activate: true,
43389          /**
43390          * @event beforesync
43391          * Fires before the textarea is updated with content from the editor iframe. Return false
43392          * to cancel the sync.
43393          * @param {Roo.HtmlEditorCore} this
43394          * @param {String} html
43395          */
43396         beforesync: true,
43397          /**
43398          * @event beforepush
43399          * Fires before the iframe editor is updated with content from the textarea. Return false
43400          * to cancel the push.
43401          * @param {Roo.HtmlEditorCore} this
43402          * @param {String} html
43403          */
43404         beforepush: true,
43405          /**
43406          * @event sync
43407          * Fires when the textarea is updated with content from the editor iframe.
43408          * @param {Roo.HtmlEditorCore} this
43409          * @param {String} html
43410          */
43411         sync: true,
43412          /**
43413          * @event push
43414          * Fires when the iframe editor is updated with content from the textarea.
43415          * @param {Roo.HtmlEditorCore} this
43416          * @param {String} html
43417          */
43418         push: true,
43419         
43420         /**
43421          * @event editorevent
43422          * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
43423          * @param {Roo.HtmlEditorCore} this
43424          */
43425         editorevent: true
43426         
43427     });
43428     
43429     // at this point this.owner is set, so we can start working out the whitelisted / blacklisted elements
43430     
43431     // defaults : white / black...
43432     this.applyBlacklists();
43433     
43434     
43435     
43436 };
43437
43438
43439 Roo.extend(Roo.HtmlEditorCore, Roo.Component,  {
43440
43441
43442      /**
43443      * @cfg {Roo.form.HtmlEditor|Roo.bootstrap.HtmlEditor} the owner field 
43444      */
43445     
43446     owner : false,
43447     
43448      /**
43449      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
43450      *                        Roo.resizable.
43451      */
43452     resizable : false,
43453      /**
43454      * @cfg {Number} height (in pixels)
43455      */   
43456     height: 300,
43457    /**
43458      * @cfg {Number} width (in pixels)
43459      */   
43460     width: 500,
43461     
43462     /**
43463      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
43464      * 
43465      */
43466     stylesheets: false,
43467     
43468     // id of frame..
43469     frameId: false,
43470     
43471     // private properties
43472     validationEvent : false,
43473     deferHeight: true,
43474     initialized : false,
43475     activated : false,
43476     sourceEditMode : false,
43477     onFocus : Roo.emptyFn,
43478     iframePad:3,
43479     hideMode:'offsets',
43480     
43481     clearUp: true,
43482     
43483     // blacklist + whitelisted elements..
43484     black: false,
43485     white: false,
43486      
43487     bodyCls : '',
43488
43489     /**
43490      * Protected method that will not generally be called directly. It
43491      * is called when the editor initializes the iframe with HTML contents. Override this method if you
43492      * want to change the initialization markup of the iframe (e.g. to add stylesheets).
43493      */
43494     getDocMarkup : function(){
43495         // body styles..
43496         var st = '';
43497         
43498         // inherit styels from page...?? 
43499         if (this.stylesheets === false) {
43500             
43501             Roo.get(document.head).select('style').each(function(node) {
43502                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43503             });
43504             
43505             Roo.get(document.head).select('link').each(function(node) { 
43506                 st += node.dom.outerHTML || new XMLSerializer().serializeToString(node.dom);
43507             });
43508             
43509         } else if (!this.stylesheets.length) {
43510                 // simple..
43511                 st = '<style type="text/css">' +
43512                     'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43513                    '</style>';
43514         } else { 
43515             st = '<style type="text/css">' +
43516                     this.stylesheets +
43517                 '</style>';
43518         }
43519         
43520         st +=  '<style type="text/css">' +
43521             'IMG { cursor: pointer } ' +
43522         '</style>';
43523
43524         var cls = 'roo-htmleditor-body';
43525         
43526         if(this.bodyCls.length){
43527             cls += ' ' + this.bodyCls;
43528         }
43529         
43530         return '<html><head>' + st  +
43531             //<style type="text/css">' +
43532             //'body{border:0;margin:0;padding:3px;height:98%;cursor:text;}' +
43533             //'</style>' +
43534             ' </head><body class="' +  cls + '"></body></html>';
43535     },
43536
43537     // private
43538     onRender : function(ct, position)
43539     {
43540         var _t = this;
43541         //Roo.HtmlEditorCore.superclass.onRender.call(this, ct, position);
43542         this.el = this.owner.inputEl ? this.owner.inputEl() : this.owner.el;
43543         
43544         
43545         this.el.dom.style.border = '0 none';
43546         this.el.dom.setAttribute('tabIndex', -1);
43547         this.el.addClass('x-hidden hide');
43548         
43549         
43550         
43551         if(Roo.isIE){ // fix IE 1px bogus margin
43552             this.el.applyStyles('margin-top:-1px;margin-bottom:-1px;')
43553         }
43554        
43555         
43556         this.frameId = Roo.id();
43557         
43558          
43559         
43560         var iframe = this.owner.wrap.createChild({
43561             tag: 'iframe',
43562             cls: 'form-control', // bootstrap..
43563             id: this.frameId,
43564             name: this.frameId,
43565             frameBorder : 'no',
43566             'src' : Roo.SSL_SECURE_URL ? Roo.SSL_SECURE_URL  :  "javascript:false"
43567         }, this.el
43568         );
43569         
43570         
43571         this.iframe = iframe.dom;
43572
43573          this.assignDocWin();
43574         
43575         this.doc.designMode = 'on';
43576        
43577         this.doc.open();
43578         this.doc.write(this.getDocMarkup());
43579         this.doc.close();
43580
43581         
43582         var task = { // must defer to wait for browser to be ready
43583             run : function(){
43584                 //console.log("run task?" + this.doc.readyState);
43585                 this.assignDocWin();
43586                 if(this.doc.body || this.doc.readyState == 'complete'){
43587                     try {
43588                         this.doc.designMode="on";
43589                     } catch (e) {
43590                         return;
43591                     }
43592                     Roo.TaskMgr.stop(task);
43593                     this.initEditor.defer(10, this);
43594                 }
43595             },
43596             interval : 10,
43597             duration: 10000,
43598             scope: this
43599         };
43600         Roo.TaskMgr.start(task);
43601
43602     },
43603
43604     // private
43605     onResize : function(w, h)
43606     {
43607          Roo.log('resize: ' +w + ',' + h );
43608         //Roo.HtmlEditorCore.superclass.onResize.apply(this, arguments);
43609         if(!this.iframe){
43610             return;
43611         }
43612         if(typeof w == 'number'){
43613             
43614             this.iframe.style.width = w + 'px';
43615         }
43616         if(typeof h == 'number'){
43617             
43618             this.iframe.style.height = h + 'px';
43619             if(this.doc){
43620                 (this.doc.body || this.doc.documentElement).style.height = (h - (this.iframePad*2)) + 'px';
43621             }
43622         }
43623         
43624     },
43625
43626     /**
43627      * Toggles the editor between standard and source edit mode.
43628      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
43629      */
43630     toggleSourceEdit : function(sourceEditMode){
43631         
43632         this.sourceEditMode = sourceEditMode === true;
43633         
43634         if(this.sourceEditMode){
43635  
43636             Roo.get(this.iframe).addClass(['x-hidden','hide']);     //FIXME - what's the BS styles for these
43637             
43638         }else{
43639             Roo.get(this.iframe).removeClass(['x-hidden','hide']);
43640             //this.iframe.className = '';
43641             this.deferFocus();
43642         }
43643         //this.setSize(this.owner.wrap.getSize());
43644         //this.fireEvent('editmodechange', this, this.sourceEditMode);
43645     },
43646
43647     
43648   
43649
43650     /**
43651      * Protected method that will not generally be called directly. If you need/want
43652      * custom HTML cleanup, this is the method you should override.
43653      * @param {String} html The HTML to be cleaned
43654      * return {String} The cleaned HTML
43655      */
43656     cleanHtml : function(html){
43657         html = String(html);
43658         if(html.length > 5){
43659             if(Roo.isSafari){ // strip safari nonsense
43660                 html = html.replace(/\sclass="(?:Apple-style-span|khtml-block-placeholder)"/gi, '');
43661             }
43662         }
43663         if(html == '&nbsp;'){
43664             html = '';
43665         }
43666         return html;
43667     },
43668
43669     /**
43670      * HTML Editor -> Textarea
43671      * Protected method that will not generally be called directly. Syncs the contents
43672      * of the editor iframe with the textarea.
43673      */
43674     syncValue : function(){
43675         if(this.initialized){
43676             var bd = (this.doc.body || this.doc.documentElement);
43677             //this.cleanUpPaste(); -- this is done else where and causes havoc..
43678             var html = bd.innerHTML;
43679             if(Roo.isSafari){
43680                 var bs = bd.getAttribute('style'); // Safari puts text-align styles on the body element!
43681                 var m = bs ? bs.match(/text-align:(.*?);/i) : false;
43682                 if(m && m[1]){
43683                     html = '<div style="'+m[0]+'">' + html + '</div>';
43684                 }
43685             }
43686             html = this.cleanHtml(html);
43687             // fix up the special chars.. normaly like back quotes in word...
43688             // however we do not want to do this with chinese..
43689             html = html.replace(/[\uD800-\uDBFF][\uDC00-\uDFFF]|[\u0080-\uFFFF]/g, function(match) {
43690                 
43691                 var cc = match.charCodeAt();
43692
43693                 // Get the character value, handling surrogate pairs
43694                 if (match.length == 2) {
43695                     // It's a surrogate pair, calculate the Unicode code point
43696                     var high = match.charCodeAt(0) - 0xD800;
43697                     var low  = match.charCodeAt(1) - 0xDC00;
43698                     cc = (high * 0x400) + low + 0x10000;
43699                 }  else if (
43700                     (cc >= 0x4E00 && cc < 0xA000 ) ||
43701                     (cc >= 0x3400 && cc < 0x4E00 ) ||
43702                     (cc >= 0xf900 && cc < 0xfb00 )
43703                 ) {
43704                         return match;
43705                 }  
43706          
43707                 // No, use a numeric entity. Here we brazenly (and possibly mistakenly)
43708                 return "&#" + cc + ";";
43709                 
43710                 
43711             });
43712             
43713             
43714              
43715             if(this.owner.fireEvent('beforesync', this, html) !== false){
43716                 this.el.dom.value = html;
43717                 this.owner.fireEvent('sync', this, html);
43718             }
43719         }
43720     },
43721
43722     /**
43723      * Protected method that will not generally be called directly. Pushes the value of the textarea
43724      * into the iframe editor.
43725      */
43726     pushValue : function(){
43727         if(this.initialized){
43728             var v = this.el.dom.value.trim();
43729             
43730 //            if(v.length < 1){
43731 //                v = '&#160;';
43732 //            }
43733             
43734             if(this.owner.fireEvent('beforepush', this, v) !== false){
43735                 var d = (this.doc.body || this.doc.documentElement);
43736                 d.innerHTML = v;
43737                 this.cleanUpPaste();
43738                 this.el.dom.value = d.innerHTML;
43739                 this.owner.fireEvent('push', this, v);
43740             }
43741         }
43742     },
43743
43744     // private
43745     deferFocus : function(){
43746         this.focus.defer(10, this);
43747     },
43748
43749     // doc'ed in Field
43750     focus : function(){
43751         if(this.win && !this.sourceEditMode){
43752             this.win.focus();
43753         }else{
43754             this.el.focus();
43755         }
43756     },
43757     
43758     assignDocWin: function()
43759     {
43760         var iframe = this.iframe;
43761         
43762          if(Roo.isIE){
43763             this.doc = iframe.contentWindow.document;
43764             this.win = iframe.contentWindow;
43765         } else {
43766 //            if (!Roo.get(this.frameId)) {
43767 //                return;
43768 //            }
43769 //            this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43770 //            this.win = Roo.get(this.frameId).dom.contentWindow;
43771             
43772             if (!Roo.get(this.frameId) && !iframe.contentDocument) {
43773                 return;
43774             }
43775             
43776             this.doc = (iframe.contentDocument || Roo.get(this.frameId).dom.document);
43777             this.win = (iframe.contentWindow || Roo.get(this.frameId).dom.contentWindow);
43778         }
43779     },
43780     
43781     // private
43782     initEditor : function(){
43783         //console.log("INIT EDITOR");
43784         this.assignDocWin();
43785         
43786         
43787         
43788         this.doc.designMode="on";
43789         this.doc.open();
43790         this.doc.write(this.getDocMarkup());
43791         this.doc.close();
43792         
43793         var dbody = (this.doc.body || this.doc.documentElement);
43794         //var ss = this.el.getStyles('font-size', 'font-family', 'background-image', 'background-repeat');
43795         // this copies styles from the containing element into thsi one..
43796         // not sure why we need all of this..
43797         //var ss = this.el.getStyles('font-size', 'background-image', 'background-repeat');
43798         
43799         //var ss = this.el.getStyles( 'background-image', 'background-repeat');
43800         //ss['background-attachment'] = 'fixed'; // w3c
43801         dbody.bgProperties = 'fixed'; // ie
43802         //Roo.DomHelper.applyStyles(dbody, ss);
43803         Roo.EventManager.on(this.doc, {
43804             //'mousedown': this.onEditorEvent,
43805             'mouseup': this.onEditorEvent,
43806             'dblclick': this.onEditorEvent,
43807             'click': this.onEditorEvent,
43808             'keyup': this.onEditorEvent,
43809             buffer:100,
43810             scope: this
43811         });
43812         if(Roo.isGecko){
43813             Roo.EventManager.on(this.doc, 'keypress', this.mozKeyPress, this);
43814         }
43815         if(Roo.isIE || Roo.isSafari || Roo.isOpera){
43816             Roo.EventManager.on(this.doc, 'keydown', this.fixKeys, this);
43817         }
43818         this.initialized = true;
43819
43820         this.owner.fireEvent('initialize', this);
43821         this.pushValue();
43822     },
43823
43824     // private
43825     onDestroy : function(){
43826         
43827         
43828         
43829         if(this.rendered){
43830             
43831             //for (var i =0; i < this.toolbars.length;i++) {
43832             //    // fixme - ask toolbars for heights?
43833             //    this.toolbars[i].onDestroy();
43834            // }
43835             
43836             //this.wrap.dom.innerHTML = '';
43837             //this.wrap.remove();
43838         }
43839     },
43840
43841     // private
43842     onFirstFocus : function(){
43843         
43844         this.assignDocWin();
43845         
43846         
43847         this.activated = true;
43848          
43849     
43850         if(Roo.isGecko){ // prevent silly gecko errors
43851             this.win.focus();
43852             var s = this.win.getSelection();
43853             if(!s.focusNode || s.focusNode.nodeType != 3){
43854                 var r = s.getRangeAt(0);
43855                 r.selectNodeContents((this.doc.body || this.doc.documentElement));
43856                 r.collapse(true);
43857                 this.deferFocus();
43858             }
43859             try{
43860                 this.execCmd('useCSS', true);
43861                 this.execCmd('styleWithCSS', false);
43862             }catch(e){}
43863         }
43864         this.owner.fireEvent('activate', this);
43865     },
43866
43867     // private
43868     adjustFont: function(btn){
43869         var adjust = btn.cmd == 'increasefontsize' ? 1 : -1;
43870         //if(Roo.isSafari){ // safari
43871         //    adjust *= 2;
43872        // }
43873         var v = parseInt(this.doc.queryCommandValue('FontSize')|| 3, 10);
43874         if(Roo.isSafari){ // safari
43875             var sm = { 10 : 1, 13: 2, 16:3, 18:4, 24: 5, 32:6, 48: 7 };
43876             v =  (v < 10) ? 10 : v;
43877             v =  (v > 48) ? 48 : v;
43878             v = typeof(sm[v]) == 'undefined' ? 1 : sm[v];
43879             
43880         }
43881         
43882         
43883         v = Math.max(1, v+adjust);
43884         
43885         this.execCmd('FontSize', v  );
43886     },
43887
43888     onEditorEvent : function(e)
43889     {
43890         this.owner.fireEvent('editorevent', this, e);
43891       //  this.updateToolbar();
43892         this.syncValue(); //we can not sync so often.. sync cleans, so this breaks stuff
43893     },
43894
43895     insertTag : function(tg)
43896     {
43897         // could be a bit smarter... -> wrap the current selected tRoo..
43898         if (tg.toLowerCase() == 'span' ||
43899             tg.toLowerCase() == 'code' ||
43900             tg.toLowerCase() == 'sup' ||
43901             tg.toLowerCase() == 'sub' 
43902             ) {
43903             
43904             range = this.createRange(this.getSelection());
43905             var wrappingNode = this.doc.createElement(tg.toLowerCase());
43906             wrappingNode.appendChild(range.extractContents());
43907             range.insertNode(wrappingNode);
43908
43909             return;
43910             
43911             
43912             
43913         }
43914         this.execCmd("formatblock",   tg);
43915         
43916     },
43917     
43918     insertText : function(txt)
43919     {
43920         
43921         
43922         var range = this.createRange();
43923         range.deleteContents();
43924                //alert(Sender.getAttribute('label'));
43925                
43926         range.insertNode(this.doc.createTextNode(txt));
43927     } ,
43928     
43929      
43930
43931     /**
43932      * Executes a Midas editor command on the editor document and performs necessary focus and
43933      * toolbar updates. <b>This should only be called after the editor is initialized.</b>
43934      * @param {String} cmd The Midas command
43935      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43936      */
43937     relayCmd : function(cmd, value){
43938         this.win.focus();
43939         this.execCmd(cmd, value);
43940         this.owner.fireEvent('editorevent', this);
43941         //this.updateToolbar();
43942         this.owner.deferFocus();
43943     },
43944
43945     /**
43946      * Executes a Midas editor command directly on the editor document.
43947      * For visual commands, you should use {@link #relayCmd} instead.
43948      * <b>This should only be called after the editor is initialized.</b>
43949      * @param {String} cmd The Midas command
43950      * @param {String/Boolean} value (optional) The value to pass to the command (defaults to null)
43951      */
43952     execCmd : function(cmd, value){
43953         this.doc.execCommand(cmd, false, value === undefined ? null : value);
43954         this.syncValue();
43955     },
43956  
43957  
43958    
43959     /**
43960      * Inserts the passed text at the current cursor position. Note: the editor must be initialized and activated
43961      * to insert tRoo.
43962      * @param {String} text | dom node.. 
43963      */
43964     insertAtCursor : function(text)
43965     {
43966         
43967         if(!this.activated){
43968             return;
43969         }
43970         /*
43971         if(Roo.isIE){
43972             this.win.focus();
43973             var r = this.doc.selection.createRange();
43974             if(r){
43975                 r.collapse(true);
43976                 r.pasteHTML(text);
43977                 this.syncValue();
43978                 this.deferFocus();
43979             
43980             }
43981             return;
43982         }
43983         */
43984         if(Roo.isGecko || Roo.isOpera || Roo.isSafari){
43985             this.win.focus();
43986             
43987             
43988             // from jquery ui (MIT licenced)
43989             var range, node;
43990             var win = this.win;
43991             
43992             if (win.getSelection && win.getSelection().getRangeAt) {
43993                 range = win.getSelection().getRangeAt(0);
43994                 node = typeof(text) == 'string' ? range.createContextualFragment(text) : text;
43995                 range.insertNode(node);
43996             } else if (win.document.selection && win.document.selection.createRange) {
43997                 // no firefox support
43998                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
43999                 win.document.selection.createRange().pasteHTML(txt);
44000             } else {
44001                 // no firefox support
44002                 var txt = typeof(text) == 'string' ? text : text.outerHTML;
44003                 this.execCmd('InsertHTML', txt);
44004             } 
44005             
44006             this.syncValue();
44007             
44008             this.deferFocus();
44009         }
44010     },
44011  // private
44012     mozKeyPress : function(e){
44013         if(e.ctrlKey){
44014             var c = e.getCharCode(), cmd;
44015           
44016             if(c > 0){
44017                 c = String.fromCharCode(c).toLowerCase();
44018                 switch(c){
44019                     case 'b':
44020                         cmd = 'bold';
44021                         break;
44022                     case 'i':
44023                         cmd = 'italic';
44024                         break;
44025                     
44026                     case 'u':
44027                         cmd = 'underline';
44028                         break;
44029                     
44030                     case 'v':
44031                         this.cleanUpPaste.defer(100, this);
44032                         return;
44033                         
44034                 }
44035                 if(cmd){
44036                     this.win.focus();
44037                     this.execCmd(cmd);
44038                     this.deferFocus();
44039                     e.preventDefault();
44040                 }
44041                 
44042             }
44043         }
44044     },
44045
44046     // private
44047     fixKeys : function(){ // load time branching for fastest keydown performance
44048         if(Roo.isIE){
44049             return function(e){
44050                 var k = e.getKey(), r;
44051                 if(k == e.TAB){
44052                     e.stopEvent();
44053                     r = this.doc.selection.createRange();
44054                     if(r){
44055                         r.collapse(true);
44056                         r.pasteHTML('&#160;&#160;&#160;&#160;');
44057                         this.deferFocus();
44058                     }
44059                     return;
44060                 }
44061                 
44062                 if(k == e.ENTER){
44063                     r = this.doc.selection.createRange();
44064                     if(r){
44065                         var target = r.parentElement();
44066                         if(!target || target.tagName.toLowerCase() != 'li'){
44067                             e.stopEvent();
44068                             r.pasteHTML('<br />');
44069                             r.collapse(false);
44070                             r.select();
44071                         }
44072                     }
44073                 }
44074                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44075                     this.cleanUpPaste.defer(100, this);
44076                     return;
44077                 }
44078                 
44079                 
44080             };
44081         }else if(Roo.isOpera){
44082             return function(e){
44083                 var k = e.getKey();
44084                 if(k == e.TAB){
44085                     e.stopEvent();
44086                     this.win.focus();
44087                     this.execCmd('InsertHTML','&#160;&#160;&#160;&#160;');
44088                     this.deferFocus();
44089                 }
44090                 if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44091                     this.cleanUpPaste.defer(100, this);
44092                     return;
44093                 }
44094                 
44095             };
44096         }else if(Roo.isSafari){
44097             return function(e){
44098                 var k = e.getKey();
44099                 
44100                 if(k == e.TAB){
44101                     e.stopEvent();
44102                     this.execCmd('InsertText','\t');
44103                     this.deferFocus();
44104                     return;
44105                 }
44106                if (String.fromCharCode(k).toLowerCase() == 'v') { // paste
44107                     this.cleanUpPaste.defer(100, this);
44108                     return;
44109                 }
44110                 
44111              };
44112         }
44113     }(),
44114     
44115     getAllAncestors: function()
44116     {
44117         var p = this.getSelectedNode();
44118         var a = [];
44119         if (!p) {
44120             a.push(p); // push blank onto stack..
44121             p = this.getParentElement();
44122         }
44123         
44124         
44125         while (p && (p.nodeType == 1) && (p.tagName.toLowerCase() != 'body')) {
44126             a.push(p);
44127             p = p.parentNode;
44128         }
44129         a.push(this.doc.body);
44130         return a;
44131     },
44132     lastSel : false,
44133     lastSelNode : false,
44134     
44135     
44136     getSelection : function() 
44137     {
44138         this.assignDocWin();
44139         return Roo.isIE ? this.doc.selection : this.win.getSelection();
44140     },
44141     
44142     getSelectedNode: function() 
44143     {
44144         // this may only work on Gecko!!!
44145         
44146         // should we cache this!!!!
44147         
44148         
44149         
44150          
44151         var range = this.createRange(this.getSelection()).cloneRange();
44152         
44153         if (Roo.isIE) {
44154             var parent = range.parentElement();
44155             while (true) {
44156                 var testRange = range.duplicate();
44157                 testRange.moveToElementText(parent);
44158                 if (testRange.inRange(range)) {
44159                     break;
44160                 }
44161                 if ((parent.nodeType != 1) || (parent.tagName.toLowerCase() == 'body')) {
44162                     break;
44163                 }
44164                 parent = parent.parentElement;
44165             }
44166             return parent;
44167         }
44168         
44169         // is ancestor a text element.
44170         var ac =  range.commonAncestorContainer;
44171         if (ac.nodeType == 3) {
44172             ac = ac.parentNode;
44173         }
44174         
44175         var ar = ac.childNodes;
44176          
44177         var nodes = [];
44178         var other_nodes = [];
44179         var has_other_nodes = false;
44180         for (var i=0;i<ar.length;i++) {
44181             if ((ar[i].nodeType == 3) && (!ar[i].data.length)) { // empty text ? 
44182                 continue;
44183             }
44184             // fullly contained node.
44185             
44186             if (this.rangeIntersectsNode(range,ar[i]) && this.rangeCompareNode(range,ar[i]) == 3) {
44187                 nodes.push(ar[i]);
44188                 continue;
44189             }
44190             
44191             // probably selected..
44192             if ((ar[i].nodeType == 1) && this.rangeIntersectsNode(range,ar[i]) && (this.rangeCompareNode(range,ar[i]) > 0)) {
44193                 other_nodes.push(ar[i]);
44194                 continue;
44195             }
44196             // outer..
44197             if (!this.rangeIntersectsNode(range,ar[i])|| (this.rangeCompareNode(range,ar[i]) == 0))  {
44198                 continue;
44199             }
44200             
44201             
44202             has_other_nodes = true;
44203         }
44204         if (!nodes.length && other_nodes.length) {
44205             nodes= other_nodes;
44206         }
44207         if (has_other_nodes || !nodes.length || (nodes.length > 1)) {
44208             return false;
44209         }
44210         
44211         return nodes[0];
44212     },
44213     createRange: function(sel)
44214     {
44215         // this has strange effects when using with 
44216         // top toolbar - not sure if it's a great idea.
44217         //this.editor.contentWindow.focus();
44218         if (typeof sel != "undefined") {
44219             try {
44220                 return sel.getRangeAt ? sel.getRangeAt(0) : sel.createRange();
44221             } catch(e) {
44222                 return this.doc.createRange();
44223             }
44224         } else {
44225             return this.doc.createRange();
44226         }
44227     },
44228     getParentElement: function()
44229     {
44230         
44231         this.assignDocWin();
44232         var sel = Roo.isIE ? this.doc.selection : this.win.getSelection();
44233         
44234         var range = this.createRange(sel);
44235          
44236         try {
44237             var p = range.commonAncestorContainer;
44238             while (p.nodeType == 3) { // text node
44239                 p = p.parentNode;
44240             }
44241             return p;
44242         } catch (e) {
44243             return null;
44244         }
44245     
44246     },
44247     /***
44248      *
44249      * Range intersection.. the hard stuff...
44250      *  '-1' = before
44251      *  '0' = hits..
44252      *  '1' = after.
44253      *         [ -- selected range --- ]
44254      *   [fail]                        [fail]
44255      *
44256      *    basically..
44257      *      if end is before start or  hits it. fail.
44258      *      if start is after end or hits it fail.
44259      *
44260      *   if either hits (but other is outside. - then it's not 
44261      *   
44262      *    
44263      **/
44264     
44265     
44266     // @see http://www.thismuchiknow.co.uk/?p=64.
44267     rangeIntersectsNode : function(range, node)
44268     {
44269         var nodeRange = node.ownerDocument.createRange();
44270         try {
44271             nodeRange.selectNode(node);
44272         } catch (e) {
44273             nodeRange.selectNodeContents(node);
44274         }
44275     
44276         var rangeStartRange = range.cloneRange();
44277         rangeStartRange.collapse(true);
44278     
44279         var rangeEndRange = range.cloneRange();
44280         rangeEndRange.collapse(false);
44281     
44282         var nodeStartRange = nodeRange.cloneRange();
44283         nodeStartRange.collapse(true);
44284     
44285         var nodeEndRange = nodeRange.cloneRange();
44286         nodeEndRange.collapse(false);
44287     
44288         return rangeStartRange.compareBoundaryPoints(
44289                  Range.START_TO_START, nodeEndRange) == -1 &&
44290                rangeEndRange.compareBoundaryPoints(
44291                  Range.START_TO_START, nodeStartRange) == 1;
44292         
44293          
44294     },
44295     rangeCompareNode : function(range, node)
44296     {
44297         var nodeRange = node.ownerDocument.createRange();
44298         try {
44299             nodeRange.selectNode(node);
44300         } catch (e) {
44301             nodeRange.selectNodeContents(node);
44302         }
44303         
44304         
44305         range.collapse(true);
44306     
44307         nodeRange.collapse(true);
44308      
44309         var ss = range.compareBoundaryPoints( Range.START_TO_START, nodeRange);
44310         var ee = range.compareBoundaryPoints(  Range.END_TO_END, nodeRange);
44311          
44312         //Roo.log(node.tagName + ': ss='+ss +', ee='+ee)
44313         
44314         var nodeIsBefore   =  ss == 1;
44315         var nodeIsAfter    = ee == -1;
44316         
44317         if (nodeIsBefore && nodeIsAfter) {
44318             return 0; // outer
44319         }
44320         if (!nodeIsBefore && nodeIsAfter) {
44321             return 1; //right trailed.
44322         }
44323         
44324         if (nodeIsBefore && !nodeIsAfter) {
44325             return 2;  // left trailed.
44326         }
44327         // fully contined.
44328         return 3;
44329     },
44330
44331     // private? - in a new class?
44332     cleanUpPaste :  function()
44333     {
44334         // cleans up the whole document..
44335         Roo.log('cleanuppaste');
44336         
44337         this.cleanUpChildren(this.doc.body);
44338         var clean = this.cleanWordChars(this.doc.body.innerHTML);
44339         if (clean != this.doc.body.innerHTML) {
44340             this.doc.body.innerHTML = clean;
44341         }
44342         
44343     },
44344     
44345     cleanWordChars : function(input) {// change the chars to hex code
44346         var he = Roo.HtmlEditorCore;
44347         
44348         var output = input;
44349         Roo.each(he.swapCodes, function(sw) { 
44350             var swapper = new RegExp("\\u" + sw[0].toString(16), "g"); // hex codes
44351             
44352             output = output.replace(swapper, sw[1]);
44353         });
44354         
44355         return output;
44356     },
44357     
44358     
44359     cleanUpChildren : function (n)
44360     {
44361         if (!n.childNodes.length) {
44362             return;
44363         }
44364         for (var i = n.childNodes.length-1; i > -1 ; i--) {
44365            this.cleanUpChild(n.childNodes[i]);
44366         }
44367     },
44368     
44369     
44370         
44371     
44372     cleanUpChild : function (node)
44373     {
44374         var ed = this;
44375         //console.log(node);
44376         if (node.nodeName == "#text") {
44377             // clean up silly Windows -- stuff?
44378             return; 
44379         }
44380         if (node.nodeName == "#comment") {
44381             node.parentNode.removeChild(node);
44382             // clean up silly Windows -- stuff?
44383             return; 
44384         }
44385         var lcname = node.tagName.toLowerCase();
44386         // we ignore whitelists... ?? = not really the way to go, but we probably have not got a full
44387         // whitelist of tags..
44388         
44389         if (this.black.indexOf(lcname) > -1 && this.clearUp ) {
44390             // remove node.
44391             node.parentNode.removeChild(node);
44392             return;
44393             
44394         }
44395         
44396         var remove_keep_children= Roo.HtmlEditorCore.remove.indexOf(node.tagName.toLowerCase()) > -1;
44397         
44398         // spans with no attributes - just remove them..
44399         if ((!node.attributes || !node.attributes.length) && lcname == 'span') { 
44400             remove_keep_children = true;
44401         }
44402         
44403         // remove <a name=....> as rendering on yahoo mailer is borked with this.
44404         // this will have to be flaged elsewhere - perhaps ablack=name... on the mailer..
44405         
44406         //if (node.tagName.toLowerCase() == 'a' && !node.hasAttribute('href')) {
44407         //    remove_keep_children = true;
44408         //}
44409         
44410         if (remove_keep_children) {
44411             this.cleanUpChildren(node);
44412             // inserts everything just before this node...
44413             while (node.childNodes.length) {
44414                 var cn = node.childNodes[0];
44415                 node.removeChild(cn);
44416                 node.parentNode.insertBefore(cn, node);
44417             }
44418             node.parentNode.removeChild(node);
44419             return;
44420         }
44421         
44422         if (!node.attributes || !node.attributes.length) {
44423             
44424           
44425             
44426             
44427             this.cleanUpChildren(node);
44428             return;
44429         }
44430         
44431         function cleanAttr(n,v)
44432         {
44433             
44434             if (v.match(/^\./) || v.match(/^\//)) {
44435                 return;
44436             }
44437             if (v.match(/^(http|https):\/\//) || v.match(/^mailto:/) || v.match(/^ftp:/)) {
44438                 return;
44439             }
44440             if (v.match(/^#/)) {
44441                 return;
44442             }
44443 //            Roo.log("(REMOVE TAG)"+ node.tagName +'.' + n + '=' + v);
44444             node.removeAttribute(n);
44445             
44446         }
44447         
44448         var cwhite = this.cwhite;
44449         var cblack = this.cblack;
44450             
44451         function cleanStyle(n,v)
44452         {
44453             if (v.match(/expression/)) { //XSS?? should we even bother..
44454                 node.removeAttribute(n);
44455                 return;
44456             }
44457             
44458             var parts = v.split(/;/);
44459             var clean = [];
44460             
44461             Roo.each(parts, function(p) {
44462                 p = p.replace(/^\s+/g,'').replace(/\s+$/g,'');
44463                 if (!p.length) {
44464                     return true;
44465                 }
44466                 var l = p.split(':').shift().replace(/\s+/g,'');
44467                 l = l.replace(/^\s+/g,'').replace(/\s+$/g,'');
44468                 
44469                 if ( cwhite.length && cblack.indexOf(l) > -1) {
44470 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44471                     //node.removeAttribute(n);
44472                     return true;
44473                 }
44474                 //Roo.log()
44475                 // only allow 'c whitelisted system attributes'
44476                 if ( cwhite.length &&  cwhite.indexOf(l) < 0) {
44477 //                    Roo.log('(REMOVE CSS)' + node.tagName +'.' + n + ':'+l + '=' + v);
44478                     //node.removeAttribute(n);
44479                     return true;
44480                 }
44481                 
44482                 
44483                  
44484                 
44485                 clean.push(p);
44486                 return true;
44487             });
44488             if (clean.length) { 
44489                 node.setAttribute(n, clean.join(';'));
44490             } else {
44491                 node.removeAttribute(n);
44492             }
44493             
44494         }
44495         
44496         
44497         for (var i = node.attributes.length-1; i > -1 ; i--) {
44498             var a = node.attributes[i];
44499             //console.log(a);
44500             
44501             if (a.name.toLowerCase().substr(0,2)=='on')  {
44502                 node.removeAttribute(a.name);
44503                 continue;
44504             }
44505             if (Roo.HtmlEditorCore.ablack.indexOf(a.name.toLowerCase()) > -1) {
44506                 node.removeAttribute(a.name);
44507                 continue;
44508             }
44509             if (Roo.HtmlEditorCore.aclean.indexOf(a.name.toLowerCase()) > -1) {
44510                 cleanAttr(a.name,a.value); // fixme..
44511                 continue;
44512             }
44513             if (a.name == 'style') {
44514                 cleanStyle(a.name,a.value);
44515                 continue;
44516             }
44517             /// clean up MS crap..
44518             // tecnically this should be a list of valid class'es..
44519             
44520             
44521             if (a.name == 'class') {
44522                 if (a.value.match(/^Mso/)) {
44523                     node.removeAttribute('class');
44524                 }
44525                 
44526                 if (a.value.match(/^body$/)) {
44527                     node.removeAttribute('class');
44528                 }
44529                 continue;
44530             }
44531             
44532             // style cleanup!?
44533             // class cleanup?
44534             
44535         }
44536         
44537         
44538         this.cleanUpChildren(node);
44539         
44540         
44541     },
44542     
44543     /**
44544      * Clean up MS wordisms...
44545      */
44546     cleanWord : function(node)
44547     {
44548         if (!node) {
44549             this.cleanWord(this.doc.body);
44550             return;
44551         }
44552         
44553         if(
44554                 node.nodeName == 'SPAN' &&
44555                 !node.hasAttributes() &&
44556                 node.childNodes.length == 1 &&
44557                 node.firstChild.nodeName == "#text"  
44558         ) {
44559             var textNode = node.firstChild;
44560             node.removeChild(textNode);
44561             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44562                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" "), node);
44563             }
44564             node.parentNode.insertBefore(textNode, node);
44565             if (node.getAttribute('lang') != 'zh-CN') {   // do not space pad on chinese characters..
44566                 node.parentNode.insertBefore(node.ownerDocument.createTextNode(" ") , node);
44567             }
44568             node.parentNode.removeChild(node);
44569         }
44570         
44571         if (node.nodeName == "#text") {
44572             // clean up silly Windows -- stuff?
44573             return; 
44574         }
44575         if (node.nodeName == "#comment") {
44576             node.parentNode.removeChild(node);
44577             // clean up silly Windows -- stuff?
44578             return; 
44579         }
44580         
44581         if (node.tagName.toLowerCase().match(/^(style|script|applet|embed|noframes|noscript)$/)) {
44582             node.parentNode.removeChild(node);
44583             return;
44584         }
44585         //Roo.log(node.tagName);
44586         // remove - but keep children..
44587         if (node.tagName.toLowerCase().match(/^(meta|link|\\?xml:|st1:|o:|v:|font)/)) {
44588             //Roo.log('-- removed');
44589             while (node.childNodes.length) {
44590                 var cn = node.childNodes[0];
44591                 node.removeChild(cn);
44592                 node.parentNode.insertBefore(cn, node);
44593                 // move node to parent - and clean it..
44594                 this.cleanWord(cn);
44595             }
44596             node.parentNode.removeChild(node);
44597             /// no need to iterate chidlren = it's got none..
44598             //this.iterateChildren(node, this.cleanWord);
44599             return;
44600         }
44601         // clean styles
44602         if (node.className.length) {
44603             
44604             var cn = node.className.split(/\W+/);
44605             var cna = [];
44606             Roo.each(cn, function(cls) {
44607                 if (cls.match(/Mso[a-zA-Z]+/)) {
44608                     return;
44609                 }
44610                 cna.push(cls);
44611             });
44612             node.className = cna.length ? cna.join(' ') : '';
44613             if (!cna.length) {
44614                 node.removeAttribute("class");
44615             }
44616         }
44617         
44618         if (node.hasAttribute("lang")) {
44619             node.removeAttribute("lang");
44620         }
44621         
44622         if (node.hasAttribute("style")) {
44623             
44624             var styles = node.getAttribute("style").split(";");
44625             var nstyle = [];
44626             Roo.each(styles, function(s) {
44627                 if (!s.match(/:/)) {
44628                     return;
44629                 }
44630                 var kv = s.split(":");
44631                 if (kv[0].match(/^(mso-|line|font|background|margin|padding|color)/)) {
44632                     return;
44633                 }
44634                 // what ever is left... we allow.
44635                 nstyle.push(s);
44636             });
44637             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44638             if (!nstyle.length) {
44639                 node.removeAttribute('style');
44640             }
44641         }
44642         this.iterateChildren(node, this.cleanWord);
44643         
44644         
44645         
44646     },
44647     /**
44648      * iterateChildren of a Node, calling fn each time, using this as the scole..
44649      * @param {DomNode} node node to iterate children of.
44650      * @param {Function} fn method of this class to call on each item.
44651      */
44652     iterateChildren : function(node, fn)
44653     {
44654         if (!node.childNodes.length) {
44655                 return;
44656         }
44657         for (var i = node.childNodes.length-1; i > -1 ; i--) {
44658            fn.call(this, node.childNodes[i])
44659         }
44660     },
44661     
44662     
44663     /**
44664      * cleanTableWidths.
44665      *
44666      * Quite often pasting from word etc.. results in tables with column and widths.
44667      * This does not work well on fluid HTML layouts - like emails. - so this code should hunt an destroy them..
44668      *
44669      */
44670     cleanTableWidths : function(node)
44671     {
44672          
44673          
44674         if (!node) {
44675             this.cleanTableWidths(this.doc.body);
44676             return;
44677         }
44678         
44679         // ignore list...
44680         if (node.nodeName == "#text" || node.nodeName == "#comment") {
44681             return; 
44682         }
44683         Roo.log(node.tagName);
44684         if (!node.tagName.toLowerCase().match(/^(table|td|tr)$/)) {
44685             this.iterateChildren(node, this.cleanTableWidths);
44686             return;
44687         }
44688         if (node.hasAttribute('width')) {
44689             node.removeAttribute('width');
44690         }
44691         
44692          
44693         if (node.hasAttribute("style")) {
44694             // pretty basic...
44695             
44696             var styles = node.getAttribute("style").split(";");
44697             var nstyle = [];
44698             Roo.each(styles, function(s) {
44699                 if (!s.match(/:/)) {
44700                     return;
44701                 }
44702                 var kv = s.split(":");
44703                 if (kv[0].match(/^\s*(width|min-width)\s*$/)) {
44704                     return;
44705                 }
44706                 // what ever is left... we allow.
44707                 nstyle.push(s);
44708             });
44709             node.setAttribute("style", nstyle.length ? nstyle.join(';') : '');
44710             if (!nstyle.length) {
44711                 node.removeAttribute('style');
44712             }
44713         }
44714         
44715         this.iterateChildren(node, this.cleanTableWidths);
44716         
44717         
44718     },
44719     
44720     
44721     
44722     
44723     domToHTML : function(currentElement, depth, nopadtext) {
44724         
44725         depth = depth || 0;
44726         nopadtext = nopadtext || false;
44727     
44728         if (!currentElement) {
44729             return this.domToHTML(this.doc.body);
44730         }
44731         
44732         //Roo.log(currentElement);
44733         var j;
44734         var allText = false;
44735         var nodeName = currentElement.nodeName;
44736         var tagName = Roo.util.Format.htmlEncode(currentElement.tagName);
44737         
44738         if  (nodeName == '#text') {
44739             
44740             return nopadtext ? currentElement.nodeValue : currentElement.nodeValue.trim();
44741         }
44742         
44743         
44744         var ret = '';
44745         if (nodeName != 'BODY') {
44746              
44747             var i = 0;
44748             // Prints the node tagName, such as <A>, <IMG>, etc
44749             if (tagName) {
44750                 var attr = [];
44751                 for(i = 0; i < currentElement.attributes.length;i++) {
44752                     // quoting?
44753                     var aname = currentElement.attributes.item(i).name;
44754                     if (!currentElement.attributes.item(i).value.length) {
44755                         continue;
44756                     }
44757                     attr.push(aname + '="' + Roo.util.Format.htmlEncode(currentElement.attributes.item(i).value) + '"' );
44758                 }
44759                 
44760                 ret = "<"+currentElement.tagName+ ( attr.length ? (' ' + attr.join(' ') ) : '') + ">";
44761             } 
44762             else {
44763                 
44764                 // eack
44765             }
44766         } else {
44767             tagName = false;
44768         }
44769         if (['IMG', 'BR', 'HR', 'INPUT'].indexOf(tagName) > -1) {
44770             return ret;
44771         }
44772         if (['PRE', 'TEXTAREA', 'TD', 'A', 'SPAN'].indexOf(tagName) > -1) { // or code?
44773             nopadtext = true;
44774         }
44775         
44776         
44777         // Traverse the tree
44778         i = 0;
44779         var currentElementChild = currentElement.childNodes.item(i);
44780         var allText = true;
44781         var innerHTML  = '';
44782         lastnode = '';
44783         while (currentElementChild) {
44784             // Formatting code (indent the tree so it looks nice on the screen)
44785             var nopad = nopadtext;
44786             if (lastnode == 'SPAN') {
44787                 nopad  = true;
44788             }
44789             // text
44790             if  (currentElementChild.nodeName == '#text') {
44791                 var toadd = Roo.util.Format.htmlEncode(currentElementChild.nodeValue);
44792                 toadd = nopadtext ? toadd : toadd.trim();
44793                 if (!nopad && toadd.length > 80) {
44794                     innerHTML  += "\n" + (new Array( depth + 1 )).join( "  "  );
44795                 }
44796                 innerHTML  += toadd;
44797                 
44798                 i++;
44799                 currentElementChild = currentElement.childNodes.item(i);
44800                 lastNode = '';
44801                 continue;
44802             }
44803             allText = false;
44804             
44805             innerHTML  += nopad ? '' : "\n" + (new Array( depth + 1 )).join( "  "  );
44806                 
44807             // Recursively traverse the tree structure of the child node
44808             innerHTML   += this.domToHTML(currentElementChild, depth+1, nopadtext);
44809             lastnode = currentElementChild.nodeName;
44810             i++;
44811             currentElementChild=currentElement.childNodes.item(i);
44812         }
44813         
44814         ret += innerHTML;
44815         
44816         if (!allText) {
44817                 // The remaining code is mostly for formatting the tree
44818             ret+= nopadtext ? '' : "\n" + (new Array( depth  )).join( "  "  );
44819         }
44820         
44821         
44822         if (tagName) {
44823             ret+= "</"+tagName+">";
44824         }
44825         return ret;
44826         
44827     },
44828         
44829     applyBlacklists : function()
44830     {
44831         var w = typeof(this.owner.white) != 'undefined' && this.owner.white ? this.owner.white  : [];
44832         var b = typeof(this.owner.black) != 'undefined' && this.owner.black ? this.owner.black :  [];
44833         
44834         this.white = [];
44835         this.black = [];
44836         Roo.each(Roo.HtmlEditorCore.white, function(tag) {
44837             if (b.indexOf(tag) > -1) {
44838                 return;
44839             }
44840             this.white.push(tag);
44841             
44842         }, this);
44843         
44844         Roo.each(w, function(tag) {
44845             if (b.indexOf(tag) > -1) {
44846                 return;
44847             }
44848             if (this.white.indexOf(tag) > -1) {
44849                 return;
44850             }
44851             this.white.push(tag);
44852             
44853         }, this);
44854         
44855         
44856         Roo.each(Roo.HtmlEditorCore.black, function(tag) {
44857             if (w.indexOf(tag) > -1) {
44858                 return;
44859             }
44860             this.black.push(tag);
44861             
44862         }, this);
44863         
44864         Roo.each(b, function(tag) {
44865             if (w.indexOf(tag) > -1) {
44866                 return;
44867             }
44868             if (this.black.indexOf(tag) > -1) {
44869                 return;
44870             }
44871             this.black.push(tag);
44872             
44873         }, this);
44874         
44875         
44876         w = typeof(this.owner.cwhite) != 'undefined' && this.owner.cwhite ? this.owner.cwhite  : [];
44877         b = typeof(this.owner.cblack) != 'undefined' && this.owner.cblack ? this.owner.cblack :  [];
44878         
44879         this.cwhite = [];
44880         this.cblack = [];
44881         Roo.each(Roo.HtmlEditorCore.cwhite, function(tag) {
44882             if (b.indexOf(tag) > -1) {
44883                 return;
44884             }
44885             this.cwhite.push(tag);
44886             
44887         }, this);
44888         
44889         Roo.each(w, function(tag) {
44890             if (b.indexOf(tag) > -1) {
44891                 return;
44892             }
44893             if (this.cwhite.indexOf(tag) > -1) {
44894                 return;
44895             }
44896             this.cwhite.push(tag);
44897             
44898         }, this);
44899         
44900         
44901         Roo.each(Roo.HtmlEditorCore.cblack, function(tag) {
44902             if (w.indexOf(tag) > -1) {
44903                 return;
44904             }
44905             this.cblack.push(tag);
44906             
44907         }, this);
44908         
44909         Roo.each(b, function(tag) {
44910             if (w.indexOf(tag) > -1) {
44911                 return;
44912             }
44913             if (this.cblack.indexOf(tag) > -1) {
44914                 return;
44915             }
44916             this.cblack.push(tag);
44917             
44918         }, this);
44919     },
44920     
44921     setStylesheets : function(stylesheets)
44922     {
44923         if(typeof(stylesheets) == 'string'){
44924             Roo.get(this.iframe.contentDocument.head).createChild({
44925                 tag : 'link',
44926                 rel : 'stylesheet',
44927                 type : 'text/css',
44928                 href : stylesheets
44929             });
44930             
44931             return;
44932         }
44933         var _this = this;
44934      
44935         Roo.each(stylesheets, function(s) {
44936             if(!s.length){
44937                 return;
44938             }
44939             
44940             Roo.get(_this.iframe.contentDocument.head).createChild({
44941                 tag : 'link',
44942                 rel : 'stylesheet',
44943                 type : 'text/css',
44944                 href : s
44945             });
44946         });
44947
44948         
44949     },
44950     
44951     removeStylesheets : function()
44952     {
44953         var _this = this;
44954         
44955         Roo.each(Roo.get(_this.iframe.contentDocument.head).select('link[rel=stylesheet]', true).elements, function(s){
44956             s.remove();
44957         });
44958     },
44959     
44960     setStyle : function(style)
44961     {
44962         Roo.get(this.iframe.contentDocument.head).createChild({
44963             tag : 'style',
44964             type : 'text/css',
44965             html : style
44966         });
44967
44968         return;
44969     }
44970     
44971     // hide stuff that is not compatible
44972     /**
44973      * @event blur
44974      * @hide
44975      */
44976     /**
44977      * @event change
44978      * @hide
44979      */
44980     /**
44981      * @event focus
44982      * @hide
44983      */
44984     /**
44985      * @event specialkey
44986      * @hide
44987      */
44988     /**
44989      * @cfg {String} fieldClass @hide
44990      */
44991     /**
44992      * @cfg {String} focusClass @hide
44993      */
44994     /**
44995      * @cfg {String} autoCreate @hide
44996      */
44997     /**
44998      * @cfg {String} inputType @hide
44999      */
45000     /**
45001      * @cfg {String} invalidClass @hide
45002      */
45003     /**
45004      * @cfg {String} invalidText @hide
45005      */
45006     /**
45007      * @cfg {String} msgFx @hide
45008      */
45009     /**
45010      * @cfg {String} validateOnBlur @hide
45011      */
45012 });
45013
45014 Roo.HtmlEditorCore.white = [
45015         'area', 'br', 'img', 'input', 'hr', 'wbr',
45016         
45017        'address', 'blockquote', 'center', 'dd',      'dir',       'div', 
45018        'dl',      'dt',         'h1',     'h2',      'h3',        'h4', 
45019        'h5',      'h6',         'hr',     'isindex', 'listing',   'marquee', 
45020        'menu',    'multicol',   'ol',     'p',       'plaintext', 'pre', 
45021        'table',   'ul',         'xmp', 
45022        
45023        'caption', 'col', 'colgroup', 'tbody', 'td', 'tfoot', 'th', 
45024       'thead',   'tr', 
45025      
45026       'dir', 'menu', 'ol', 'ul', 'dl',
45027        
45028       'embed',  'object'
45029 ];
45030
45031
45032 Roo.HtmlEditorCore.black = [
45033     //    'embed',  'object', // enable - backend responsiblity to clean thiese
45034         'applet', // 
45035         'base',   'basefont', 'bgsound', 'blink',  'body', 
45036         'frame',  'frameset', 'head',    'html',   'ilayer', 
45037         'iframe', 'layer',  'link',     'meta',    'object',   
45038         'script', 'style' ,'title',  'xml' // clean later..
45039 ];
45040 Roo.HtmlEditorCore.clean = [
45041     'script', 'style', 'title', 'xml'
45042 ];
45043 Roo.HtmlEditorCore.remove = [
45044     'font'
45045 ];
45046 // attributes..
45047
45048 Roo.HtmlEditorCore.ablack = [
45049     'on'
45050 ];
45051     
45052 Roo.HtmlEditorCore.aclean = [ 
45053     'action', 'background', 'codebase', 'dynsrc', 'href', 'lowsrc' 
45054 ];
45055
45056 // protocols..
45057 Roo.HtmlEditorCore.pwhite= [
45058         'http',  'https',  'mailto'
45059 ];
45060
45061 // white listed style attributes.
45062 Roo.HtmlEditorCore.cwhite= [
45063       //  'text-align', /// default is to allow most things..
45064       
45065          
45066 //        'font-size'//??
45067 ];
45068
45069 // black listed style attributes.
45070 Roo.HtmlEditorCore.cblack= [
45071       //  'font-size' -- this can be set by the project 
45072 ];
45073
45074
45075 Roo.HtmlEditorCore.swapCodes   =[ 
45076     [    8211, "--" ], 
45077     [    8212, "--" ], 
45078     [    8216,  "'" ],  
45079     [    8217, "'" ],  
45080     [    8220, '"' ],  
45081     [    8221, '"' ],  
45082     [    8226, "*" ],  
45083     [    8230, "..." ]
45084 ]; 
45085
45086     //<script type="text/javascript">
45087
45088 /*
45089  * Ext JS Library 1.1.1
45090  * Copyright(c) 2006-2007, Ext JS, LLC.
45091  * Licence LGPL
45092  * 
45093  */
45094  
45095  
45096 Roo.form.HtmlEditor = function(config){
45097     
45098     
45099     
45100     Roo.form.HtmlEditor.superclass.constructor.call(this, config);
45101     
45102     if (!this.toolbars) {
45103         this.toolbars = [];
45104     }
45105     this.editorcore = new Roo.HtmlEditorCore(Roo.apply({ owner : this} , config));
45106     
45107     
45108 };
45109
45110 /**
45111  * @class Roo.form.HtmlEditor
45112  * @extends Roo.form.Field
45113  * Provides a lightweight HTML Editor component.
45114  *
45115  * This has been tested on Fireforx / Chrome.. IE may not be so great..
45116  * 
45117  * <br><br><b>Note: The focus/blur and validation marking functionality inherited from Ext.form.Field is NOT
45118  * supported by this editor.</b><br/><br/>
45119  * An Editor is a sensitive component that can't be used in all spots standard fields can be used. Putting an Editor within
45120  * any element that has display set to 'none' can cause problems in Safari and Firefox.<br/><br/>
45121  */
45122 Roo.extend(Roo.form.HtmlEditor, Roo.form.Field, {
45123     /**
45124      * @cfg {Boolean} clearUp
45125      */
45126     clearUp : true,
45127       /**
45128      * @cfg {Array} toolbars Array of toolbars. - defaults to just the Standard one
45129      */
45130     toolbars : false,
45131    
45132      /**
45133      * @cfg {String} resizable  's' or 'se' or 'e' - wrapps the element in a
45134      *                        Roo.resizable.
45135      */
45136     resizable : false,
45137      /**
45138      * @cfg {Number} height (in pixels)
45139      */   
45140     height: 300,
45141    /**
45142      * @cfg {Number} width (in pixels)
45143      */   
45144     width: 500,
45145     
45146     /**
45147      * @cfg {Array} stylesheets url of stylesheets. set to [] to disable stylesheets.
45148      * 
45149      */
45150     stylesheets: false,
45151     
45152     
45153      /**
45154      * @cfg {Array} blacklist of css styles style attributes (blacklist overrides whitelist)
45155      * 
45156      */
45157     cblack: false,
45158     /**
45159      * @cfg {Array} whitelist of css styles style attributes (blacklist overrides whitelist)
45160      * 
45161      */
45162     cwhite: false,
45163     
45164      /**
45165      * @cfg {Array} blacklist of html tags - in addition to standard blacklist.
45166      * 
45167      */
45168     black: false,
45169     /**
45170      * @cfg {Array} whitelist of html tags - in addition to statndard whitelist
45171      * 
45172      */
45173     white: false,
45174     
45175     // id of frame..
45176     frameId: false,
45177     
45178     // private properties
45179     validationEvent : false,
45180     deferHeight: true,
45181     initialized : false,
45182     activated : false,
45183     
45184     onFocus : Roo.emptyFn,
45185     iframePad:3,
45186     hideMode:'offsets',
45187     
45188     actionMode : 'container', // defaults to hiding it...
45189     
45190     defaultAutoCreate : { // modified by initCompnoent..
45191         tag: "textarea",
45192         style:"width:500px;height:300px;",
45193         autocomplete: "new-password"
45194     },
45195
45196     // private
45197     initComponent : function(){
45198         this.addEvents({
45199             /**
45200              * @event initialize
45201              * Fires when the editor is fully initialized (including the iframe)
45202              * @param {HtmlEditor} this
45203              */
45204             initialize: true,
45205             /**
45206              * @event activate
45207              * Fires when the editor is first receives the focus. Any insertion must wait
45208              * until after this event.
45209              * @param {HtmlEditor} this
45210              */
45211             activate: true,
45212              /**
45213              * @event beforesync
45214              * Fires before the textarea is updated with content from the editor iframe. Return false
45215              * to cancel the sync.
45216              * @param {HtmlEditor} this
45217              * @param {String} html
45218              */
45219             beforesync: true,
45220              /**
45221              * @event beforepush
45222              * Fires before the iframe editor is updated with content from the textarea. Return false
45223              * to cancel the push.
45224              * @param {HtmlEditor} this
45225              * @param {String} html
45226              */
45227             beforepush: true,
45228              /**
45229              * @event sync
45230              * Fires when the textarea is updated with content from the editor iframe.
45231              * @param {HtmlEditor} this
45232              * @param {String} html
45233              */
45234             sync: true,
45235              /**
45236              * @event push
45237              * Fires when the iframe editor is updated with content from the textarea.
45238              * @param {HtmlEditor} this
45239              * @param {String} html
45240              */
45241             push: true,
45242              /**
45243              * @event editmodechange
45244              * Fires when the editor switches edit modes
45245              * @param {HtmlEditor} this
45246              * @param {Boolean} sourceEdit True if source edit, false if standard editing.
45247              */
45248             editmodechange: true,
45249             /**
45250              * @event editorevent
45251              * Fires when on any editor (mouse up/down cursor movement etc.) - used for toolbar hooks.
45252              * @param {HtmlEditor} this
45253              */
45254             editorevent: true,
45255             /**
45256              * @event firstfocus
45257              * Fires when on first focus - needed by toolbars..
45258              * @param {HtmlEditor} this
45259              */
45260             firstfocus: true,
45261             /**
45262              * @event autosave
45263              * Auto save the htmlEditor value as a file into Events
45264              * @param {HtmlEditor} this
45265              */
45266             autosave: true,
45267             /**
45268              * @event savedpreview
45269              * preview the saved version of htmlEditor
45270              * @param {HtmlEditor} this
45271              */
45272             savedpreview: true,
45273             
45274             /**
45275             * @event stylesheetsclick
45276             * Fires when press the Sytlesheets button
45277             * @param {Roo.HtmlEditorCore} this
45278             */
45279             stylesheetsclick: true
45280         });
45281         this.defaultAutoCreate =  {
45282             tag: "textarea",
45283             style:'width: ' + this.width + 'px;height: ' + this.height + 'px;',
45284             autocomplete: "new-password"
45285         };
45286     },
45287
45288     /**
45289      * Protected method that will not generally be called directly. It
45290      * is called when the editor creates its toolbar. Override this method if you need to
45291      * add custom toolbar buttons.
45292      * @param {HtmlEditor} editor
45293      */
45294     createToolbar : function(editor){
45295         Roo.log("create toolbars");
45296         if (!editor.toolbars || !editor.toolbars.length) {
45297             editor.toolbars = [ new Roo.form.HtmlEditor.ToolbarStandard() ]; // can be empty?
45298         }
45299         
45300         for (var i =0 ; i < editor.toolbars.length;i++) {
45301             editor.toolbars[i] = Roo.factory(
45302                     typeof(editor.toolbars[i]) == 'string' ?
45303                         { xtype: editor.toolbars[i]} : editor.toolbars[i],
45304                 Roo.form.HtmlEditor);
45305             editor.toolbars[i].init(editor);
45306         }
45307          
45308         
45309     },
45310
45311      
45312     // private
45313     onRender : function(ct, position)
45314     {
45315         var _t = this;
45316         Roo.form.HtmlEditor.superclass.onRender.call(this, ct, position);
45317         
45318         this.wrap = this.el.wrap({
45319             cls:'x-html-editor-wrap', cn:{cls:'x-html-editor-tb'}
45320         });
45321         
45322         this.editorcore.onRender(ct, position);
45323          
45324         if (this.resizable) {
45325             this.resizeEl = new Roo.Resizable(this.wrap, {
45326                 pinned : true,
45327                 wrap: true,
45328                 dynamic : true,
45329                 minHeight : this.height,
45330                 height: this.height,
45331                 handles : this.resizable,
45332                 width: this.width,
45333                 listeners : {
45334                     resize : function(r, w, h) {
45335                         _t.onResize(w,h); // -something
45336                     }
45337                 }
45338             });
45339             
45340         }
45341         this.createToolbar(this);
45342        
45343         
45344         if(!this.width){
45345             this.setSize(this.wrap.getSize());
45346         }
45347         if (this.resizeEl) {
45348             this.resizeEl.resizeTo.defer(100, this.resizeEl,[ this.width,this.height ] );
45349             // should trigger onReize..
45350         }
45351         
45352         this.keyNav = new Roo.KeyNav(this.el, {
45353             
45354             "tab" : function(e){
45355                 e.preventDefault();
45356                 
45357                 var value = this.getValue();
45358                 
45359                 var start = this.el.dom.selectionStart;
45360                 var end = this.el.dom.selectionEnd;
45361                 
45362                 if(!e.shiftKey){
45363                     
45364                     this.setValue(value.substring(0, start) + "\t" + value.substring(end));
45365                     this.el.dom.setSelectionRange(end + 1, end + 1);
45366                     return;
45367                 }
45368                 
45369                 var f = value.substring(0, start).split("\t");
45370                 
45371                 if(f.pop().length != 0){
45372                     return;
45373                 }
45374                 
45375                 this.setValue(f.join("\t") + value.substring(end));
45376                 this.el.dom.setSelectionRange(start - 1, start - 1);
45377                 
45378             },
45379             
45380             "home" : function(e){
45381                 e.preventDefault();
45382                 
45383                 var curr = this.el.dom.selectionStart;
45384                 var lines = this.getValue().split("\n");
45385                 
45386                 if(!lines.length){
45387                     return;
45388                 }
45389                 
45390                 if(e.ctrlKey){
45391                     this.el.dom.setSelectionRange(0, 0);
45392                     return;
45393                 }
45394                 
45395                 var pos = 0;
45396                 
45397                 for (var i = 0; i < lines.length;i++) {
45398                     pos += lines[i].length;
45399                     
45400                     if(i != 0){
45401                         pos += 1;
45402                     }
45403                     
45404                     if(pos < curr){
45405                         continue;
45406                     }
45407                     
45408                     pos -= lines[i].length;
45409                     
45410                     break;
45411                 }
45412                 
45413                 if(!e.shiftKey){
45414                     this.el.dom.setSelectionRange(pos, pos);
45415                     return;
45416                 }
45417                 
45418                 this.el.dom.selectionStart = pos;
45419                 this.el.dom.selectionEnd = curr;
45420             },
45421             
45422             "end" : function(e){
45423                 e.preventDefault();
45424                 
45425                 var curr = this.el.dom.selectionStart;
45426                 var lines = this.getValue().split("\n");
45427                 
45428                 if(!lines.length){
45429                     return;
45430                 }
45431                 
45432                 if(e.ctrlKey){
45433                     this.el.dom.setSelectionRange(this.getValue().length, this.getValue().length);
45434                     return;
45435                 }
45436                 
45437                 var pos = 0;
45438                 
45439                 for (var i = 0; i < lines.length;i++) {
45440                     
45441                     pos += lines[i].length;
45442                     
45443                     if(i != 0){
45444                         pos += 1;
45445                     }
45446                     
45447                     if(pos < curr){
45448                         continue;
45449                     }
45450                     
45451                     break;
45452                 }
45453                 
45454                 if(!e.shiftKey){
45455                     this.el.dom.setSelectionRange(pos, pos);
45456                     return;
45457                 }
45458                 
45459                 this.el.dom.selectionStart = curr;
45460                 this.el.dom.selectionEnd = pos;
45461             },
45462
45463             scope : this,
45464
45465             doRelay : function(foo, bar, hname){
45466                 return Roo.KeyNav.prototype.doRelay.apply(this, arguments);
45467             },
45468
45469             forceKeyDown: true
45470         });
45471         
45472 //        if(this.autosave && this.w){
45473 //            this.autoSaveFn = setInterval(this.autosave, 1000);
45474 //        }
45475     },
45476
45477     // private
45478     onResize : function(w, h)
45479     {
45480         Roo.form.HtmlEditor.superclass.onResize.apply(this, arguments);
45481         var ew = false;
45482         var eh = false;
45483         
45484         if(this.el ){
45485             if(typeof w == 'number'){
45486                 var aw = w - this.wrap.getFrameWidth('lr');
45487                 this.el.setWidth(this.adjustWidth('textarea', aw));
45488                 ew = aw;
45489             }
45490             if(typeof h == 'number'){
45491                 var tbh = 0;
45492                 for (var i =0; i < this.toolbars.length;i++) {
45493                     // fixme - ask toolbars for heights?
45494                     tbh += this.toolbars[i].tb.el.getHeight();
45495                     if (this.toolbars[i].footer) {
45496                         tbh += this.toolbars[i].footer.el.getHeight();
45497                     }
45498                 }
45499                 
45500                 
45501                 
45502                 
45503                 var ah = h - this.wrap.getFrameWidth('tb') - tbh;// this.tb.el.getHeight();
45504                 ah -= 5; // knock a few pixes off for look..
45505 //                Roo.log(ah);
45506                 this.el.setHeight(this.adjustWidth('textarea', ah));
45507                 var eh = ah;
45508             }
45509         }
45510         Roo.log('onResize:' + [w,h,ew,eh].join(',') );
45511         this.editorcore.onResize(ew,eh);
45512         
45513     },
45514
45515     /**
45516      * Toggles the editor between standard and source edit mode.
45517      * @param {Boolean} sourceEdit (optional) True for source edit, false for standard
45518      */
45519     toggleSourceEdit : function(sourceEditMode)
45520     {
45521         this.editorcore.toggleSourceEdit(sourceEditMode);
45522         
45523         if(this.editorcore.sourceEditMode){
45524             Roo.log('editor - showing textarea');
45525             
45526 //            Roo.log('in');
45527 //            Roo.log(this.syncValue());
45528             this.editorcore.syncValue();
45529             this.el.removeClass('x-hidden');
45530             this.el.dom.removeAttribute('tabIndex');
45531             this.el.focus();
45532             
45533             for (var i = 0; i < this.toolbars.length; i++) {
45534                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45535                     this.toolbars[i].tb.hide();
45536                     this.toolbars[i].footer.hide();
45537                 }
45538             }
45539             
45540         }else{
45541             Roo.log('editor - hiding textarea');
45542 //            Roo.log('out')
45543 //            Roo.log(this.pushValue()); 
45544             this.editorcore.pushValue();
45545             
45546             this.el.addClass('x-hidden');
45547             this.el.dom.setAttribute('tabIndex', -1);
45548             
45549             for (var i = 0; i < this.toolbars.length; i++) {
45550                 if(this.toolbars[i] instanceof Roo.form.HtmlEditor.ToolbarContext){
45551                     this.toolbars[i].tb.show();
45552                     this.toolbars[i].footer.show();
45553                 }
45554             }
45555             
45556             //this.deferFocus();
45557         }
45558         
45559         this.setSize(this.wrap.getSize());
45560         this.onResize(this.wrap.getSize().width, this.wrap.getSize().height);
45561         
45562         this.fireEvent('editmodechange', this, this.editorcore.sourceEditMode);
45563     },
45564  
45565     // private (for BoxComponent)
45566     adjustSize : Roo.BoxComponent.prototype.adjustSize,
45567
45568     // private (for BoxComponent)
45569     getResizeEl : function(){
45570         return this.wrap;
45571     },
45572
45573     // private (for BoxComponent)
45574     getPositionEl : function(){
45575         return this.wrap;
45576     },
45577
45578     // private
45579     initEvents : function(){
45580         this.originalValue = this.getValue();
45581     },
45582
45583     /**
45584      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45585      * @method
45586      */
45587     markInvalid : Roo.emptyFn,
45588     /**
45589      * Overridden and disabled. The editor element does not support standard valid/invalid marking. @hide
45590      * @method
45591      */
45592     clearInvalid : Roo.emptyFn,
45593
45594     setValue : function(v){
45595         Roo.form.HtmlEditor.superclass.setValue.call(this, v);
45596         this.editorcore.pushValue();
45597     },
45598
45599      
45600     // private
45601     deferFocus : function(){
45602         this.focus.defer(10, this);
45603     },
45604
45605     // doc'ed in Field
45606     focus : function(){
45607         this.editorcore.focus();
45608         
45609     },
45610       
45611
45612     // private
45613     onDestroy : function(){
45614         
45615         
45616         
45617         if(this.rendered){
45618             
45619             for (var i =0; i < this.toolbars.length;i++) {
45620                 // fixme - ask toolbars for heights?
45621                 this.toolbars[i].onDestroy();
45622             }
45623             
45624             this.wrap.dom.innerHTML = '';
45625             this.wrap.remove();
45626         }
45627     },
45628
45629     // private
45630     onFirstFocus : function(){
45631         //Roo.log("onFirstFocus");
45632         this.editorcore.onFirstFocus();
45633          for (var i =0; i < this.toolbars.length;i++) {
45634             this.toolbars[i].onFirstFocus();
45635         }
45636         
45637     },
45638     
45639     // private
45640     syncValue : function()
45641     {
45642         this.editorcore.syncValue();
45643     },
45644     
45645     pushValue : function()
45646     {
45647         this.editorcore.pushValue();
45648     },
45649     
45650     setStylesheets : function(stylesheets)
45651     {
45652         this.editorcore.setStylesheets(stylesheets);
45653     },
45654     
45655     removeStylesheets : function()
45656     {
45657         this.editorcore.removeStylesheets();
45658     }
45659      
45660     
45661     // hide stuff that is not compatible
45662     /**
45663      * @event blur
45664      * @hide
45665      */
45666     /**
45667      * @event change
45668      * @hide
45669      */
45670     /**
45671      * @event focus
45672      * @hide
45673      */
45674     /**
45675      * @event specialkey
45676      * @hide
45677      */
45678     /**
45679      * @cfg {String} fieldClass @hide
45680      */
45681     /**
45682      * @cfg {String} focusClass @hide
45683      */
45684     /**
45685      * @cfg {String} autoCreate @hide
45686      */
45687     /**
45688      * @cfg {String} inputType @hide
45689      */
45690     /**
45691      * @cfg {String} invalidClass @hide
45692      */
45693     /**
45694      * @cfg {String} invalidText @hide
45695      */
45696     /**
45697      * @cfg {String} msgFx @hide
45698      */
45699     /**
45700      * @cfg {String} validateOnBlur @hide
45701      */
45702 });
45703  
45704     // <script type="text/javascript">
45705 /*
45706  * Based on
45707  * Ext JS Library 1.1.1
45708  * Copyright(c) 2006-2007, Ext JS, LLC.
45709  *  
45710  
45711  */
45712
45713 /**
45714  * @class Roo.form.HtmlEditorToolbar1
45715  * Basic Toolbar
45716  * 
45717  * Usage:
45718  *
45719  new Roo.form.HtmlEditor({
45720     ....
45721     toolbars : [
45722         new Roo.form.HtmlEditorToolbar1({
45723             disable : { fonts: 1 , format: 1, ..., ... , ...],
45724             btns : [ .... ]
45725         })
45726     }
45727      
45728  * 
45729  * @cfg {Object} disable List of elements to disable..
45730  * @cfg {Array} btns List of additional buttons.
45731  * 
45732  * 
45733  * NEEDS Extra CSS? 
45734  * .x-html-editor-tb .x-edit-none .x-btn-text { background: none; }
45735  */
45736  
45737 Roo.form.HtmlEditor.ToolbarStandard = function(config)
45738 {
45739     
45740     Roo.apply(this, config);
45741     
45742     // default disabled, based on 'good practice'..
45743     this.disable = this.disable || {};
45744     Roo.applyIf(this.disable, {
45745         fontSize : true,
45746         colors : true,
45747         specialElements : true
45748     });
45749     
45750     
45751     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
45752     // dont call parent... till later.
45753 }
45754
45755 Roo.apply(Roo.form.HtmlEditor.ToolbarStandard.prototype,  {
45756     
45757     tb: false,
45758     
45759     rendered: false,
45760     
45761     editor : false,
45762     editorcore : false,
45763     /**
45764      * @cfg {Object} disable  List of toolbar elements to disable
45765          
45766      */
45767     disable : false,
45768     
45769     
45770      /**
45771      * @cfg {String} createLinkText The default text for the create link prompt
45772      */
45773     createLinkText : 'Please enter the URL for the link:',
45774     /**
45775      * @cfg {String} defaultLinkValue The default value for the create link prompt (defaults to http:/ /)
45776      */
45777     defaultLinkValue : 'http:/'+'/',
45778    
45779     
45780       /**
45781      * @cfg {Array} fontFamilies An array of available font families
45782      */
45783     fontFamilies : [
45784         'Arial',
45785         'Courier New',
45786         'Tahoma',
45787         'Times New Roman',
45788         'Verdana'
45789     ],
45790     
45791     specialChars : [
45792            "&#169;",
45793           "&#174;",     
45794           "&#8482;",    
45795           "&#163;" ,    
45796          // "&#8212;",    
45797           "&#8230;",    
45798           "&#247;" ,    
45799         //  "&#225;" ,     ?? a acute?
45800            "&#8364;"    , //Euro
45801        //   "&#8220;"    ,
45802         //  "&#8221;"    ,
45803         //  "&#8226;"    ,
45804           "&#176;"  //   , // degrees
45805
45806          // "&#233;"     , // e ecute
45807          // "&#250;"     , // u ecute?
45808     ],
45809     
45810     specialElements : [
45811         {
45812             text: "Insert Table",
45813             xtype: 'MenuItem',
45814             xns : Roo.Menu,
45815             ihtml :  '<table><tr><td>Cell</td></tr></table>' 
45816                 
45817         },
45818         {    
45819             text: "Insert Image",
45820             xtype: 'MenuItem',
45821             xns : Roo.Menu,
45822             ihtml : '<img src="about:blank"/>'
45823             
45824         }
45825         
45826          
45827     ],
45828     
45829     
45830     inputElements : [ 
45831             "form", "input:text", "input:hidden", "input:checkbox", "input:radio", "input:password", 
45832             "input:submit", "input:button", "select", "textarea", "label" ],
45833     formats : [
45834         ["p"] ,  
45835         ["h1"],["h2"],["h3"],["h4"],["h5"],["h6"], 
45836         ["pre"],[ "code"], 
45837         ["abbr"],[ "acronym"],[ "address"],[ "cite"],[ "samp"],[ "var"],
45838         ['div'],['span'],
45839         ['sup'],['sub']
45840     ],
45841     
45842     cleanStyles : [
45843         "font-size"
45844     ],
45845      /**
45846      * @cfg {String} defaultFont default font to use.
45847      */
45848     defaultFont: 'tahoma',
45849    
45850     fontSelect : false,
45851     
45852     
45853     formatCombo : false,
45854     
45855     init : function(editor)
45856     {
45857         this.editor = editor;
45858         this.editorcore = editor.editorcore ? editor.editorcore : editor;
45859         var editorcore = this.editorcore;
45860         
45861         var _t = this;
45862         
45863         var fid = editorcore.frameId;
45864         var etb = this;
45865         function btn(id, toggle, handler){
45866             var xid = fid + '-'+ id ;
45867             return {
45868                 id : xid,
45869                 cmd : id,
45870                 cls : 'x-btn-icon x-edit-'+id,
45871                 enableToggle:toggle !== false,
45872                 scope: _t, // was editor...
45873                 handler:handler||_t.relayBtnCmd,
45874                 clickEvent:'mousedown',
45875                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
45876                 tabIndex:-1
45877             };
45878         }
45879         
45880         
45881         
45882         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
45883         this.tb = tb;
45884          // stop form submits
45885         tb.el.on('click', function(e){
45886             e.preventDefault(); // what does this do?
45887         });
45888
45889         if(!this.disable.font) { // && !Roo.isSafari){
45890             /* why no safari for fonts 
45891             editor.fontSelect = tb.el.createChild({
45892                 tag:'select',
45893                 tabIndex: -1,
45894                 cls:'x-font-select',
45895                 html: this.createFontOptions()
45896             });
45897             
45898             editor.fontSelect.on('change', function(){
45899                 var font = editor.fontSelect.dom.value;
45900                 editor.relayCmd('fontname', font);
45901                 editor.deferFocus();
45902             }, editor);
45903             
45904             tb.add(
45905                 editor.fontSelect.dom,
45906                 '-'
45907             );
45908             */
45909             
45910         };
45911         if(!this.disable.formats){
45912             this.formatCombo = new Roo.form.ComboBox({
45913                 store: new Roo.data.SimpleStore({
45914                     id : 'tag',
45915                     fields: ['tag'],
45916                     data : this.formats // from states.js
45917                 }),
45918                 blockFocus : true,
45919                 name : '',
45920                 //autoCreate : {tag: "div",  size: "20"},
45921                 displayField:'tag',
45922                 typeAhead: false,
45923                 mode: 'local',
45924                 editable : false,
45925                 triggerAction: 'all',
45926                 emptyText:'Add tag',
45927                 selectOnFocus:true,
45928                 width:135,
45929                 listeners : {
45930                     'select': function(c, r, i) {
45931                         editorcore.insertTag(r.get('tag'));
45932                         editor.focus();
45933                     }
45934                 }
45935
45936             });
45937             tb.addField(this.formatCombo);
45938             
45939         }
45940         
45941         if(!this.disable.format){
45942             tb.add(
45943                 btn('bold'),
45944                 btn('italic'),
45945                 btn('underline'),
45946                 btn('strikethrough')
45947             );
45948         };
45949         if(!this.disable.fontSize){
45950             tb.add(
45951                 '-',
45952                 
45953                 
45954                 btn('increasefontsize', false, editorcore.adjustFont),
45955                 btn('decreasefontsize', false, editorcore.adjustFont)
45956             );
45957         };
45958         
45959         
45960         if(!this.disable.colors){
45961             tb.add(
45962                 '-', {
45963                     id:editorcore.frameId +'-forecolor',
45964                     cls:'x-btn-icon x-edit-forecolor',
45965                     clickEvent:'mousedown',
45966                     tooltip: this.buttonTips['forecolor'] || undefined,
45967                     tabIndex:-1,
45968                     menu : new Roo.menu.ColorMenu({
45969                         allowReselect: true,
45970                         focus: Roo.emptyFn,
45971                         value:'000000',
45972                         plain:true,
45973                         selectHandler: function(cp, color){
45974                             editorcore.execCmd('forecolor', Roo.isSafari || Roo.isIE ? '#'+color : color);
45975                             editor.deferFocus();
45976                         },
45977                         scope: editorcore,
45978                         clickEvent:'mousedown'
45979                     })
45980                 }, {
45981                     id:editorcore.frameId +'backcolor',
45982                     cls:'x-btn-icon x-edit-backcolor',
45983                     clickEvent:'mousedown',
45984                     tooltip: this.buttonTips['backcolor'] || undefined,
45985                     tabIndex:-1,
45986                     menu : new Roo.menu.ColorMenu({
45987                         focus: Roo.emptyFn,
45988                         value:'FFFFFF',
45989                         plain:true,
45990                         allowReselect: true,
45991                         selectHandler: function(cp, color){
45992                             if(Roo.isGecko){
45993                                 editorcore.execCmd('useCSS', false);
45994                                 editorcore.execCmd('hilitecolor', color);
45995                                 editorcore.execCmd('useCSS', true);
45996                                 editor.deferFocus();
45997                             }else{
45998                                 editorcore.execCmd(Roo.isOpera ? 'hilitecolor' : 'backcolor', 
45999                                     Roo.isSafari || Roo.isIE ? '#'+color : color);
46000                                 editor.deferFocus();
46001                             }
46002                         },
46003                         scope:editorcore,
46004                         clickEvent:'mousedown'
46005                     })
46006                 }
46007             );
46008         };
46009         // now add all the items...
46010         
46011
46012         if(!this.disable.alignments){
46013             tb.add(
46014                 '-',
46015                 btn('justifyleft'),
46016                 btn('justifycenter'),
46017                 btn('justifyright')
46018             );
46019         };
46020
46021         //if(!Roo.isSafari){
46022             if(!this.disable.links){
46023                 tb.add(
46024                     '-',
46025                     btn('createlink', false, this.createLink)    /// MOVE TO HERE?!!?!?!?!
46026                 );
46027             };
46028
46029             if(!this.disable.lists){
46030                 tb.add(
46031                     '-',
46032                     btn('insertorderedlist'),
46033                     btn('insertunorderedlist')
46034                 );
46035             }
46036             if(!this.disable.sourceEdit){
46037                 tb.add(
46038                     '-',
46039                     btn('sourceedit', true, function(btn){
46040                         this.toggleSourceEdit(btn.pressed);
46041                     })
46042                 );
46043             }
46044         //}
46045         
46046         var smenu = { };
46047         // special menu.. - needs to be tidied up..
46048         if (!this.disable.special) {
46049             smenu = {
46050                 text: "&#169;",
46051                 cls: 'x-edit-none',
46052                 
46053                 menu : {
46054                     items : []
46055                 }
46056             };
46057             for (var i =0; i < this.specialChars.length; i++) {
46058                 smenu.menu.items.push({
46059                     
46060                     html: this.specialChars[i],
46061                     handler: function(a,b) {
46062                         editorcore.insertAtCursor(String.fromCharCode(a.html.replace('&#','').replace(';', '')));
46063                         //editor.insertAtCursor(a.html);
46064                         
46065                     },
46066                     tabIndex:-1
46067                 });
46068             }
46069             
46070             
46071             tb.add(smenu);
46072             
46073             
46074         }
46075         
46076         var cmenu = { };
46077         if (!this.disable.cleanStyles) {
46078             cmenu = {
46079                 cls: 'x-btn-icon x-btn-clear',
46080                 
46081                 menu : {
46082                     items : []
46083                 }
46084             };
46085             for (var i =0; i < this.cleanStyles.length; i++) {
46086                 cmenu.menu.items.push({
46087                     actiontype : this.cleanStyles[i],
46088                     html: 'Remove ' + this.cleanStyles[i],
46089                     handler: function(a,b) {
46090 //                        Roo.log(a);
46091 //                        Roo.log(b);
46092                         var c = Roo.get(editorcore.doc.body);
46093                         c.select('[style]').each(function(s) {
46094                             s.dom.style.removeProperty(a.actiontype);
46095                         });
46096                         editorcore.syncValue();
46097                     },
46098                     tabIndex:-1
46099                 });
46100             }
46101              cmenu.menu.items.push({
46102                 actiontype : 'tablewidths',
46103                 html: 'Remove Table Widths',
46104                 handler: function(a,b) {
46105                     editorcore.cleanTableWidths();
46106                     editorcore.syncValue();
46107                 },
46108                 tabIndex:-1
46109             });
46110             cmenu.menu.items.push({
46111                 actiontype : 'word',
46112                 html: 'Remove MS Word Formating',
46113                 handler: function(a,b) {
46114                     editorcore.cleanWord();
46115                     editorcore.syncValue();
46116                 },
46117                 tabIndex:-1
46118             });
46119             
46120             cmenu.menu.items.push({
46121                 actiontype : 'all',
46122                 html: 'Remove All Styles',
46123                 handler: function(a,b) {
46124                     
46125                     var c = Roo.get(editorcore.doc.body);
46126                     c.select('[style]').each(function(s) {
46127                         s.dom.removeAttribute('style');
46128                     });
46129                     editorcore.syncValue();
46130                 },
46131                 tabIndex:-1
46132             });
46133             
46134             cmenu.menu.items.push({
46135                 actiontype : 'all',
46136                 html: 'Remove All CSS Classes',
46137                 handler: function(a,b) {
46138                     
46139                     var c = Roo.get(editorcore.doc.body);
46140                     c.select('[class]').each(function(s) {
46141                         s.dom.removeAttribute('class');
46142                     });
46143                     editorcore.cleanWord();
46144                     editorcore.syncValue();
46145                 },
46146                 tabIndex:-1
46147             });
46148             
46149              cmenu.menu.items.push({
46150                 actiontype : 'tidy',
46151                 html: 'Tidy HTML Source',
46152                 handler: function(a,b) {
46153                     editorcore.doc.body.innerHTML = editorcore.domToHTML();
46154                     editorcore.syncValue();
46155                 },
46156                 tabIndex:-1
46157             });
46158             
46159             
46160             tb.add(cmenu);
46161         }
46162          
46163         if (!this.disable.specialElements) {
46164             var semenu = {
46165                 text: "Other;",
46166                 cls: 'x-edit-none',
46167                 menu : {
46168                     items : []
46169                 }
46170             };
46171             for (var i =0; i < this.specialElements.length; i++) {
46172                 semenu.menu.items.push(
46173                     Roo.apply({ 
46174                         handler: function(a,b) {
46175                             editor.insertAtCursor(this.ihtml);
46176                         }
46177                     }, this.specialElements[i])
46178                 );
46179                     
46180             }
46181             
46182             tb.add(semenu);
46183             
46184             
46185         }
46186          
46187         
46188         if (this.btns) {
46189             for(var i =0; i< this.btns.length;i++) {
46190                 var b = Roo.factory(this.btns[i],Roo.form);
46191                 b.cls =  'x-edit-none';
46192                 
46193                 if(typeof(this.btns[i].cls) != 'undefined' && this.btns[i].cls.indexOf('x-init-enable') !== -1){
46194                     b.cls += ' x-init-enable';
46195                 }
46196                 
46197                 b.scope = editorcore;
46198                 tb.add(b);
46199             }
46200         
46201         }
46202         
46203         
46204         
46205         // disable everything...
46206         
46207         this.tb.items.each(function(item){
46208             
46209            if(
46210                 item.id != editorcore.frameId+ '-sourceedit' && 
46211                 (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)
46212             ){
46213                 
46214                 item.disable();
46215             }
46216         });
46217         this.rendered = true;
46218         
46219         // the all the btns;
46220         editor.on('editorevent', this.updateToolbar, this);
46221         // other toolbars need to implement this..
46222         //editor.on('editmodechange', this.updateToolbar, this);
46223     },
46224     
46225     
46226     relayBtnCmd : function(btn) {
46227         this.editorcore.relayCmd(btn.cmd);
46228     },
46229     // private used internally
46230     createLink : function(){
46231         Roo.log("create link?");
46232         var url = prompt(this.createLinkText, this.defaultLinkValue);
46233         if(url && url != 'http:/'+'/'){
46234             this.editorcore.relayCmd('createlink', url);
46235         }
46236     },
46237
46238     
46239     /**
46240      * Protected method that will not generally be called directly. It triggers
46241      * a toolbar update by reading the markup state of the current selection in the editor.
46242      */
46243     updateToolbar: function(){
46244
46245         if(!this.editorcore.activated){
46246             this.editor.onFirstFocus();
46247             return;
46248         }
46249
46250         var btns = this.tb.items.map, 
46251             doc = this.editorcore.doc,
46252             frameId = this.editorcore.frameId;
46253
46254         if(!this.disable.font && !Roo.isSafari){
46255             /*
46256             var name = (doc.queryCommandValue('FontName')||this.editor.defaultFont).toLowerCase();
46257             if(name != this.fontSelect.dom.value){
46258                 this.fontSelect.dom.value = name;
46259             }
46260             */
46261         }
46262         if(!this.disable.format){
46263             btns[frameId + '-bold'].toggle(doc.queryCommandState('bold'));
46264             btns[frameId + '-italic'].toggle(doc.queryCommandState('italic'));
46265             btns[frameId + '-underline'].toggle(doc.queryCommandState('underline'));
46266             btns[frameId + '-strikethrough'].toggle(doc.queryCommandState('strikethrough'));
46267         }
46268         if(!this.disable.alignments){
46269             btns[frameId + '-justifyleft'].toggle(doc.queryCommandState('justifyleft'));
46270             btns[frameId + '-justifycenter'].toggle(doc.queryCommandState('justifycenter'));
46271             btns[frameId + '-justifyright'].toggle(doc.queryCommandState('justifyright'));
46272         }
46273         if(!Roo.isSafari && !this.disable.lists){
46274             btns[frameId + '-insertorderedlist'].toggle(doc.queryCommandState('insertorderedlist'));
46275             btns[frameId + '-insertunorderedlist'].toggle(doc.queryCommandState('insertunorderedlist'));
46276         }
46277         
46278         var ans = this.editorcore.getAllAncestors();
46279         if (this.formatCombo) {
46280             
46281             
46282             var store = this.formatCombo.store;
46283             this.formatCombo.setValue("");
46284             for (var i =0; i < ans.length;i++) {
46285                 if (ans[i] && store.query('tag',ans[i].tagName.toLowerCase(), false).length) {
46286                     // select it..
46287                     this.formatCombo.setValue(ans[i].tagName.toLowerCase());
46288                     break;
46289                 }
46290             }
46291         }
46292         
46293         
46294         
46295         // hides menus... - so this cant be on a menu...
46296         Roo.menu.MenuMgr.hideAll();
46297
46298         //this.editorsyncValue();
46299     },
46300    
46301     
46302     createFontOptions : function(){
46303         var buf = [], fs = this.fontFamilies, ff, lc;
46304         
46305         
46306         
46307         for(var i = 0, len = fs.length; i< len; i++){
46308             ff = fs[i];
46309             lc = ff.toLowerCase();
46310             buf.push(
46311                 '<option value="',lc,'" style="font-family:',ff,';"',
46312                     (this.defaultFont == lc ? ' selected="true">' : '>'),
46313                     ff,
46314                 '</option>'
46315             );
46316         }
46317         return buf.join('');
46318     },
46319     
46320     toggleSourceEdit : function(sourceEditMode){
46321         
46322         Roo.log("toolbar toogle");
46323         if(sourceEditMode === undefined){
46324             sourceEditMode = !this.sourceEditMode;
46325         }
46326         this.sourceEditMode = sourceEditMode === true;
46327         var btn = this.tb.items.get(this.editorcore.frameId +'-sourceedit');
46328         // just toggle the button?
46329         if(btn.pressed !== this.sourceEditMode){
46330             btn.toggle(this.sourceEditMode);
46331             return;
46332         }
46333         
46334         if(sourceEditMode){
46335             Roo.log("disabling buttons");
46336             this.tb.items.each(function(item){
46337                 if(item.cmd != 'sourceedit' && (typeof(item.cls) != 'undefined' && item.cls.indexOf('x-init-enable') === -1)){
46338                     item.disable();
46339                 }
46340             });
46341           
46342         }else{
46343             Roo.log("enabling buttons");
46344             if(this.editorcore.initialized){
46345                 this.tb.items.each(function(item){
46346                     item.enable();
46347                 });
46348             }
46349             
46350         }
46351         Roo.log("calling toggole on editor");
46352         // tell the editor that it's been pressed..
46353         this.editor.toggleSourceEdit(sourceEditMode);
46354        
46355     },
46356      /**
46357      * Object collection of toolbar tooltips for the buttons in the editor. The key
46358      * is the command id associated with that button and the value is a valid QuickTips object.
46359      * For example:
46360 <pre><code>
46361 {
46362     bold : {
46363         title: 'Bold (Ctrl+B)',
46364         text: 'Make the selected text bold.',
46365         cls: 'x-html-editor-tip'
46366     },
46367     italic : {
46368         title: 'Italic (Ctrl+I)',
46369         text: 'Make the selected text italic.',
46370         cls: 'x-html-editor-tip'
46371     },
46372     ...
46373 </code></pre>
46374     * @type Object
46375      */
46376     buttonTips : {
46377         bold : {
46378             title: 'Bold (Ctrl+B)',
46379             text: 'Make the selected text bold.',
46380             cls: 'x-html-editor-tip'
46381         },
46382         italic : {
46383             title: 'Italic (Ctrl+I)',
46384             text: 'Make the selected text italic.',
46385             cls: 'x-html-editor-tip'
46386         },
46387         underline : {
46388             title: 'Underline (Ctrl+U)',
46389             text: 'Underline the selected text.',
46390             cls: 'x-html-editor-tip'
46391         },
46392         strikethrough : {
46393             title: 'Strikethrough',
46394             text: 'Strikethrough the selected text.',
46395             cls: 'x-html-editor-tip'
46396         },
46397         increasefontsize : {
46398             title: 'Grow Text',
46399             text: 'Increase the font size.',
46400             cls: 'x-html-editor-tip'
46401         },
46402         decreasefontsize : {
46403             title: 'Shrink Text',
46404             text: 'Decrease the font size.',
46405             cls: 'x-html-editor-tip'
46406         },
46407         backcolor : {
46408             title: 'Text Highlight Color',
46409             text: 'Change the background color of the selected text.',
46410             cls: 'x-html-editor-tip'
46411         },
46412         forecolor : {
46413             title: 'Font Color',
46414             text: 'Change the color of the selected text.',
46415             cls: 'x-html-editor-tip'
46416         },
46417         justifyleft : {
46418             title: 'Align Text Left',
46419             text: 'Align text to the left.',
46420             cls: 'x-html-editor-tip'
46421         },
46422         justifycenter : {
46423             title: 'Center Text',
46424             text: 'Center text in the editor.',
46425             cls: 'x-html-editor-tip'
46426         },
46427         justifyright : {
46428             title: 'Align Text Right',
46429             text: 'Align text to the right.',
46430             cls: 'x-html-editor-tip'
46431         },
46432         insertunorderedlist : {
46433             title: 'Bullet List',
46434             text: 'Start a bulleted list.',
46435             cls: 'x-html-editor-tip'
46436         },
46437         insertorderedlist : {
46438             title: 'Numbered List',
46439             text: 'Start a numbered list.',
46440             cls: 'x-html-editor-tip'
46441         },
46442         createlink : {
46443             title: 'Hyperlink',
46444             text: 'Make the selected text a hyperlink.',
46445             cls: 'x-html-editor-tip'
46446         },
46447         sourceedit : {
46448             title: 'Source Edit',
46449             text: 'Switch to source editing mode.',
46450             cls: 'x-html-editor-tip'
46451         }
46452     },
46453     // private
46454     onDestroy : function(){
46455         if(this.rendered){
46456             
46457             this.tb.items.each(function(item){
46458                 if(item.menu){
46459                     item.menu.removeAll();
46460                     if(item.menu.el){
46461                         item.menu.el.destroy();
46462                     }
46463                 }
46464                 item.destroy();
46465             });
46466              
46467         }
46468     },
46469     onFirstFocus: function() {
46470         this.tb.items.each(function(item){
46471            item.enable();
46472         });
46473     }
46474 });
46475
46476
46477
46478
46479 // <script type="text/javascript">
46480 /*
46481  * Based on
46482  * Ext JS Library 1.1.1
46483  * Copyright(c) 2006-2007, Ext JS, LLC.
46484  *  
46485  
46486  */
46487
46488  
46489 /**
46490  * @class Roo.form.HtmlEditor.ToolbarContext
46491  * Context Toolbar
46492  * 
46493  * Usage:
46494  *
46495  new Roo.form.HtmlEditor({
46496     ....
46497     toolbars : [
46498         { xtype: 'ToolbarStandard', styles : {} }
46499         { xtype: 'ToolbarContext', disable : {} }
46500     ]
46501 })
46502
46503      
46504  * 
46505  * @config : {Object} disable List of elements to disable.. (not done yet.)
46506  * @config : {Object} styles  Map of styles available.
46507  * 
46508  */
46509
46510 Roo.form.HtmlEditor.ToolbarContext = function(config)
46511 {
46512     
46513     Roo.apply(this, config);
46514     //Roo.form.HtmlEditorToolbar1.superclass.constructor.call(this, editor.wrap.dom.firstChild, [], config);
46515     // dont call parent... till later.
46516     this.styles = this.styles || {};
46517 }
46518
46519  
46520
46521 Roo.form.HtmlEditor.ToolbarContext.types = {
46522     'IMG' : {
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"],[ "right"],[ "center"],[ "top"]],
46534             width : 80
46535             
46536         },
46537         border: {
46538             title: "Border",
46539             width: 40
46540         },
46541         alt: {
46542             title: "Alt",
46543             width: 120
46544         },
46545         src : {
46546             title: "Src",
46547             width: 220
46548         }
46549         
46550     },
46551     'A' : {
46552         name : {
46553             title: "Name",
46554             width: 50
46555         },
46556         target:  {
46557             title: "Target",
46558             width: 120
46559         },
46560         href:  {
46561             title: "Href",
46562             width: 220
46563         } // border?
46564         
46565     },
46566     'TABLE' : {
46567         rows : {
46568             title: "Rows",
46569             width: 20
46570         },
46571         cols : {
46572             title: "Cols",
46573             width: 20
46574         },
46575         width : {
46576             title: "Width",
46577             width: 40
46578         },
46579         height : {
46580             title: "Height",
46581             width: 40
46582         },
46583         border : {
46584             title: "Border",
46585             width: 20
46586         }
46587     },
46588     'TD' : {
46589         width : {
46590             title: "Width",
46591             width: 40
46592         },
46593         height : {
46594             title: "Height",
46595             width: 40
46596         },   
46597         align: {
46598             title: "Align",
46599             opts : [[""],[ "left"],[ "center"],[ "right"],[ "justify"],[ "char"]],
46600             width: 80
46601         },
46602         valign: {
46603             title: "Valign",
46604             opts : [[""],[ "top"],[ "middle"],[ "bottom"],[ "baseline"]],
46605             width: 80
46606         },
46607         colspan: {
46608             title: "Colspan",
46609             width: 20
46610             
46611         },
46612          'font-family'  : {
46613             title : "Font",
46614             style : 'fontFamily',
46615             displayField: 'display',
46616             optname : 'font-family',
46617             width: 140
46618         }
46619     },
46620     'INPUT' : {
46621         name : {
46622             title: "name",
46623             width: 120
46624         },
46625         value : {
46626             title: "Value",
46627             width: 120
46628         },
46629         width : {
46630             title: "Width",
46631             width: 40
46632         }
46633     },
46634     'LABEL' : {
46635         'for' : {
46636             title: "For",
46637             width: 120
46638         }
46639     },
46640     'TEXTAREA' : {
46641           name : {
46642             title: "name",
46643             width: 120
46644         },
46645         rows : {
46646             title: "Rows",
46647             width: 20
46648         },
46649         cols : {
46650             title: "Cols",
46651             width: 20
46652         }
46653     },
46654     'SELECT' : {
46655         name : {
46656             title: "name",
46657             width: 120
46658         },
46659         selectoptions : {
46660             title: "Options",
46661             width: 200
46662         }
46663     },
46664     
46665     // should we really allow this??
46666     // should this just be 
46667     'BODY' : {
46668         title : {
46669             title: "Title",
46670             width: 200,
46671             disabled : true
46672         }
46673     },
46674     'SPAN' : {
46675         'font-family'  : {
46676             title : "Font",
46677             style : 'fontFamily',
46678             displayField: 'display',
46679             optname : 'font-family',
46680             width: 140
46681         }
46682     },
46683     'DIV' : {
46684         'font-family'  : {
46685             title : "Font",
46686             style : 'fontFamily',
46687             displayField: 'display',
46688             optname : 'font-family',
46689             width: 140
46690         }
46691     },
46692      'P' : {
46693         'font-family'  : {
46694             title : "Font",
46695             style : 'fontFamily',
46696             displayField: 'display',
46697             optname : 'font-family',
46698             width: 140
46699         }
46700     },
46701     
46702     '*' : {
46703         // empty..
46704     }
46705
46706 };
46707
46708 // this should be configurable.. - you can either set it up using stores, or modify options somehwere..
46709 Roo.form.HtmlEditor.ToolbarContext.stores = false;
46710
46711 Roo.form.HtmlEditor.ToolbarContext.options = {
46712         'font-family'  : [ 
46713                 [ 'Helvetica,Arial,sans-serif', 'Helvetica'],
46714                 [ 'Courier New', 'Courier New'],
46715                 [ 'Tahoma', 'Tahoma'],
46716                 [ 'Times New Roman,serif', 'Times'],
46717                 [ 'Verdana','Verdana' ]
46718         ]
46719 };
46720
46721 // fixme - these need to be configurable..
46722  
46723
46724 //Roo.form.HtmlEditor.ToolbarContext.types
46725
46726
46727 Roo.apply(Roo.form.HtmlEditor.ToolbarContext.prototype,  {
46728     
46729     tb: false,
46730     
46731     rendered: false,
46732     
46733     editor : false,
46734     editorcore : false,
46735     /**
46736      * @cfg {Object} disable  List of toolbar elements to disable
46737          
46738      */
46739     disable : false,
46740     /**
46741      * @cfg {Object} styles List of styles 
46742      *    eg. { '*' : [ 'headline' ] , 'TD' : [ 'underline', 'double-underline' ] } 
46743      *
46744      * These must be defined in the page, so they get rendered correctly..
46745      * .headline { }
46746      * TD.underline { }
46747      * 
46748      */
46749     styles : false,
46750     
46751     options: false,
46752     
46753     toolbars : false,
46754     
46755     init : function(editor)
46756     {
46757         this.editor = editor;
46758         this.editorcore = editor.editorcore ? editor.editorcore : editor;
46759         var editorcore = this.editorcore;
46760         
46761         var fid = editorcore.frameId;
46762         var etb = this;
46763         function btn(id, toggle, handler){
46764             var xid = fid + '-'+ id ;
46765             return {
46766                 id : xid,
46767                 cmd : id,
46768                 cls : 'x-btn-icon x-edit-'+id,
46769                 enableToggle:toggle !== false,
46770                 scope: editorcore, // was editor...
46771                 handler:handler||editorcore.relayBtnCmd,
46772                 clickEvent:'mousedown',
46773                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
46774                 tabIndex:-1
46775             };
46776         }
46777         // create a new element.
46778         var wdiv = editor.wrap.createChild({
46779                 tag: 'div'
46780             }, editor.wrap.dom.firstChild.nextSibling, true);
46781         
46782         // can we do this more than once??
46783         
46784          // stop form submits
46785       
46786  
46787         // disable everything...
46788         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46789         this.toolbars = {};
46790            
46791         for (var i in  ty) {
46792           
46793             this.toolbars[i] = this.buildToolbar(ty[i],i);
46794         }
46795         this.tb = this.toolbars.BODY;
46796         this.tb.el.show();
46797         this.buildFooter();
46798         this.footer.show();
46799         editor.on('hide', function( ) { this.footer.hide() }, this);
46800         editor.on('show', function( ) { this.footer.show() }, this);
46801         
46802          
46803         this.rendered = true;
46804         
46805         // the all the btns;
46806         editor.on('editorevent', this.updateToolbar, this);
46807         // other toolbars need to implement this..
46808         //editor.on('editmodechange', this.updateToolbar, this);
46809     },
46810     
46811     
46812     
46813     /**
46814      * Protected method that will not generally be called directly. It triggers
46815      * a toolbar update by reading the markup state of the current selection in the editor.
46816      *
46817      * Note you can force an update by calling on('editorevent', scope, false)
46818      */
46819     updateToolbar: function(editor,ev,sel){
46820
46821         //Roo.log(ev);
46822         // capture mouse up - this is handy for selecting images..
46823         // perhaps should go somewhere else...
46824         if(!this.editorcore.activated){
46825              this.editor.onFirstFocus();
46826             return;
46827         }
46828         
46829         
46830         
46831         // http://developer.yahoo.com/yui/docs/simple-editor.js.html
46832         // selectNode - might want to handle IE?
46833         if (ev &&
46834             (ev.type == 'mouseup' || ev.type == 'click' ) &&
46835             ev.target && ev.target.tagName == 'IMG') {
46836             // they have click on an image...
46837             // let's see if we can change the selection...
46838             sel = ev.target;
46839          
46840               var nodeRange = sel.ownerDocument.createRange();
46841             try {
46842                 nodeRange.selectNode(sel);
46843             } catch (e) {
46844                 nodeRange.selectNodeContents(sel);
46845             }
46846             //nodeRange.collapse(true);
46847             var s = this.editorcore.win.getSelection();
46848             s.removeAllRanges();
46849             s.addRange(nodeRange);
46850         }  
46851         
46852       
46853         var updateFooter = sel ? false : true;
46854         
46855         
46856         var ans = this.editorcore.getAllAncestors();
46857         
46858         // pick
46859         var ty= Roo.form.HtmlEditor.ToolbarContext.types;
46860         
46861         if (!sel) { 
46862             sel = ans.length ? (ans[0] ?  ans[0]  : ans[1]) : this.editorcore.doc.body;
46863             sel = sel ? sel : this.editorcore.doc.body;
46864             sel = sel.tagName.length ? sel : this.editorcore.doc.body;
46865             
46866         }
46867         // pick a menu that exists..
46868         var tn = sel.tagName.toUpperCase();
46869         //sel = typeof(ty[tn]) != 'undefined' ? sel : this.editor.doc.body;
46870         
46871         tn = sel.tagName.toUpperCase();
46872         
46873         var lastSel = this.tb.selectedNode;
46874         
46875         this.tb.selectedNode = sel;
46876         
46877         // if current menu does not match..
46878         
46879         if ((this.tb.name != tn) || (lastSel != this.tb.selectedNode) || ev === false) {
46880                 
46881             this.tb.el.hide();
46882             ///console.log("show: " + tn);
46883             this.tb =  typeof(ty[tn]) != 'undefined' ? this.toolbars[tn] : this.toolbars['*'];
46884             this.tb.el.show();
46885             // update name
46886             this.tb.items.first().el.innerHTML = tn + ':&nbsp;';
46887             
46888             
46889             // update attributes
46890             if (this.tb.fields) {
46891                 this.tb.fields.each(function(e) {
46892                     if (e.stylename) {
46893                         e.setValue(sel.style[e.stylename]);
46894                         return;
46895                     } 
46896                    e.setValue(sel.getAttribute(e.attrname));
46897                 });
46898             }
46899             
46900             var hasStyles = false;
46901             for(var i in this.styles) {
46902                 hasStyles = true;
46903                 break;
46904             }
46905             
46906             // update styles
46907             if (hasStyles) { 
46908                 var st = this.tb.fields.item(0);
46909                 
46910                 st.store.removeAll();
46911                
46912                 
46913                 var cn = sel.className.split(/\s+/);
46914                 
46915                 var avs = [];
46916                 if (this.styles['*']) {
46917                     
46918                     Roo.each(this.styles['*'], function(v) {
46919                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46920                     });
46921                 }
46922                 if (this.styles[tn]) { 
46923                     Roo.each(this.styles[tn], function(v) {
46924                         avs.push( [ v , cn.indexOf(v) > -1 ? 1 : 0 ] );         
46925                     });
46926                 }
46927                 
46928                 st.store.loadData(avs);
46929                 st.collapse();
46930                 st.setValue(cn);
46931             }
46932             // flag our selected Node.
46933             this.tb.selectedNode = sel;
46934            
46935            
46936             Roo.menu.MenuMgr.hideAll();
46937
46938         }
46939         
46940         if (!updateFooter) {
46941             //this.footDisp.dom.innerHTML = ''; 
46942             return;
46943         }
46944         // update the footer
46945         //
46946         var html = '';
46947         
46948         this.footerEls = ans.reverse();
46949         Roo.each(this.footerEls, function(a,i) {
46950             if (!a) { return; }
46951             html += html.length ? ' &gt; '  :  '';
46952             
46953             html += '<span class="x-ed-loc-' + i + '">' + a.tagName + '</span>';
46954             
46955         });
46956        
46957         // 
46958         var sz = this.footDisp.up('td').getSize();
46959         this.footDisp.dom.style.width = (sz.width -10) + 'px';
46960         this.footDisp.dom.style.marginLeft = '5px';
46961         
46962         this.footDisp.dom.style.overflow = 'hidden';
46963         
46964         this.footDisp.dom.innerHTML = html;
46965             
46966         //this.editorsyncValue();
46967     },
46968      
46969     
46970    
46971        
46972     // private
46973     onDestroy : function(){
46974         if(this.rendered){
46975             
46976             this.tb.items.each(function(item){
46977                 if(item.menu){
46978                     item.menu.removeAll();
46979                     if(item.menu.el){
46980                         item.menu.el.destroy();
46981                     }
46982                 }
46983                 item.destroy();
46984             });
46985              
46986         }
46987     },
46988     onFirstFocus: function() {
46989         // need to do this for all the toolbars..
46990         this.tb.items.each(function(item){
46991            item.enable();
46992         });
46993     },
46994     buildToolbar: function(tlist, nm)
46995     {
46996         var editor = this.editor;
46997         var editorcore = this.editorcore;
46998          // create a new element.
46999         var wdiv = editor.wrap.createChild({
47000                 tag: 'div'
47001             }, editor.wrap.dom.firstChild.nextSibling, true);
47002         
47003        
47004         var tb = new Roo.Toolbar(wdiv);
47005         // add the name..
47006         
47007         tb.add(nm+ ":&nbsp;");
47008         
47009         var styles = [];
47010         for(var i in this.styles) {
47011             styles.push(i);
47012         }
47013         
47014         // styles...
47015         if (styles && styles.length) {
47016             
47017             // this needs a multi-select checkbox...
47018             tb.addField( new Roo.form.ComboBox({
47019                 store: new Roo.data.SimpleStore({
47020                     id : 'val',
47021                     fields: ['val', 'selected'],
47022                     data : [] 
47023                 }),
47024                 name : '-roo-edit-className',
47025                 attrname : 'className',
47026                 displayField: 'val',
47027                 typeAhead: false,
47028                 mode: 'local',
47029                 editable : false,
47030                 triggerAction: 'all',
47031                 emptyText:'Select Style',
47032                 selectOnFocus:true,
47033                 width: 130,
47034                 listeners : {
47035                     'select': function(c, r, i) {
47036                         // initial support only for on class per el..
47037                         tb.selectedNode.className =  r ? r.get('val') : '';
47038                         editorcore.syncValue();
47039                     }
47040                 }
47041     
47042             }));
47043         }
47044         
47045         var tbc = Roo.form.HtmlEditor.ToolbarContext;
47046         var tbops = tbc.options;
47047         
47048         for (var i in tlist) {
47049             
47050             var item = tlist[i];
47051             tb.add(item.title + ":&nbsp;");
47052             
47053             
47054             //optname == used so you can configure the options available..
47055             var opts = item.opts ? item.opts : false;
47056             if (item.optname) {
47057                 opts = tbops[item.optname];
47058            
47059             }
47060             
47061             if (opts) {
47062                 // opts == pulldown..
47063                 tb.addField( new Roo.form.ComboBox({
47064                     store:   typeof(tbc.stores[i]) != 'undefined' ?  Roo.factory(tbc.stores[i],Roo.data) : new Roo.data.SimpleStore({
47065                         id : 'val',
47066                         fields: ['val', 'display'],
47067                         data : opts  
47068                     }),
47069                     name : '-roo-edit-' + i,
47070                     attrname : i,
47071                     stylename : item.style ? item.style : false,
47072                     displayField: item.displayField ? item.displayField : 'val',
47073                     valueField :  'val',
47074                     typeAhead: false,
47075                     mode: typeof(tbc.stores[i]) != 'undefined'  ? 'remote' : 'local',
47076                     editable : false,
47077                     triggerAction: 'all',
47078                     emptyText:'Select',
47079                     selectOnFocus:true,
47080                     width: item.width ? item.width  : 130,
47081                     listeners : {
47082                         'select': function(c, r, i) {
47083                             if (c.stylename) {
47084                                 tb.selectedNode.style[c.stylename] =  r.get('val');
47085                                 return;
47086                             }
47087                             tb.selectedNode.setAttribute(c.attrname, r.get('val'));
47088                         }
47089                     }
47090
47091                 }));
47092                 continue;
47093                     
47094                  
47095                 
47096                 tb.addField( new Roo.form.TextField({
47097                     name: i,
47098                     width: 100,
47099                     //allowBlank:false,
47100                     value: ''
47101                 }));
47102                 continue;
47103             }
47104             tb.addField( new Roo.form.TextField({
47105                 name: '-roo-edit-' + i,
47106                 attrname : i,
47107                 
47108                 width: item.width,
47109                 //allowBlank:true,
47110                 value: '',
47111                 listeners: {
47112                     'change' : function(f, nv, ov) {
47113                         tb.selectedNode.setAttribute(f.attrname, nv);
47114                         editorcore.syncValue();
47115                     }
47116                 }
47117             }));
47118              
47119         }
47120         
47121         var _this = this;
47122         
47123         if(nm == 'BODY'){
47124             tb.addSeparator();
47125         
47126             tb.addButton( {
47127                 text: 'Stylesheets',
47128
47129                 listeners : {
47130                     click : function ()
47131                     {
47132                         _this.editor.fireEvent('stylesheetsclick', _this.editor);
47133                     }
47134                 }
47135             });
47136         }
47137         
47138         tb.addFill();
47139         tb.addButton( {
47140             text: 'Remove Tag',
47141     
47142             listeners : {
47143                 click : function ()
47144                 {
47145                     // remove
47146                     // undo does not work.
47147                      
47148                     var sn = tb.selectedNode;
47149                     
47150                     var pn = sn.parentNode;
47151                     
47152                     var stn =  sn.childNodes[0];
47153                     var en = sn.childNodes[sn.childNodes.length - 1 ];
47154                     while (sn.childNodes.length) {
47155                         var node = sn.childNodes[0];
47156                         sn.removeChild(node);
47157                         //Roo.log(node);
47158                         pn.insertBefore(node, sn);
47159                         
47160                     }
47161                     pn.removeChild(sn);
47162                     var range = editorcore.createRange();
47163         
47164                     range.setStart(stn,0);
47165                     range.setEnd(en,0); //????
47166                     //range.selectNode(sel);
47167                     
47168                     
47169                     var selection = editorcore.getSelection();
47170                     selection.removeAllRanges();
47171                     selection.addRange(range);
47172                     
47173                     
47174                     
47175                     //_this.updateToolbar(null, null, pn);
47176                     _this.updateToolbar(null, null, null);
47177                     _this.footDisp.dom.innerHTML = ''; 
47178                 }
47179             }
47180             
47181                     
47182                 
47183             
47184         });
47185         
47186         
47187         tb.el.on('click', function(e){
47188             e.preventDefault(); // what does this do?
47189         });
47190         tb.el.setVisibilityMode( Roo.Element.DISPLAY);
47191         tb.el.hide();
47192         tb.name = nm;
47193         // dont need to disable them... as they will get hidden
47194         return tb;
47195          
47196         
47197     },
47198     buildFooter : function()
47199     {
47200         
47201         var fel = this.editor.wrap.createChild();
47202         this.footer = new Roo.Toolbar(fel);
47203         // toolbar has scrolly on left / right?
47204         var footDisp= new Roo.Toolbar.Fill();
47205         var _t = this;
47206         this.footer.add(
47207             {
47208                 text : '&lt;',
47209                 xtype: 'Button',
47210                 handler : function() {
47211                     _t.footDisp.scrollTo('left',0,true)
47212                 }
47213             }
47214         );
47215         this.footer.add( footDisp );
47216         this.footer.add( 
47217             {
47218                 text : '&gt;',
47219                 xtype: 'Button',
47220                 handler : function() {
47221                     // no animation..
47222                     _t.footDisp.select('span').last().scrollIntoView(_t.footDisp,true);
47223                 }
47224             }
47225         );
47226         var fel = Roo.get(footDisp.el);
47227         fel.addClass('x-editor-context');
47228         this.footDispWrap = fel; 
47229         this.footDispWrap.overflow  = 'hidden';
47230         
47231         this.footDisp = fel.createChild();
47232         this.footDispWrap.on('click', this.onContextClick, this)
47233         
47234         
47235     },
47236     onContextClick : function (ev,dom)
47237     {
47238         ev.preventDefault();
47239         var  cn = dom.className;
47240         //Roo.log(cn);
47241         if (!cn.match(/x-ed-loc-/)) {
47242             return;
47243         }
47244         var n = cn.split('-').pop();
47245         var ans = this.footerEls;
47246         var sel = ans[n];
47247         
47248          // pick
47249         var range = this.editorcore.createRange();
47250         
47251         range.selectNodeContents(sel);
47252         //range.selectNode(sel);
47253         
47254         
47255         var selection = this.editorcore.getSelection();
47256         selection.removeAllRanges();
47257         selection.addRange(range);
47258         
47259         
47260         
47261         this.updateToolbar(null, null, sel);
47262         
47263         
47264     }
47265     
47266     
47267     
47268     
47269     
47270 });
47271
47272
47273
47274
47275
47276 /*
47277  * Based on:
47278  * Ext JS Library 1.1.1
47279  * Copyright(c) 2006-2007, Ext JS, LLC.
47280  *
47281  * Originally Released Under LGPL - original licence link has changed is not relivant.
47282  *
47283  * Fork - LGPL
47284  * <script type="text/javascript">
47285  */
47286  
47287 /**
47288  * @class Roo.form.BasicForm
47289  * @extends Roo.util.Observable
47290  * Supplies the functionality to do "actions" on forms and initialize Roo.form.Field types on existing markup.
47291  * @constructor
47292  * @param {String/HTMLElement/Roo.Element} el The form element or its id
47293  * @param {Object} config Configuration options
47294  */
47295 Roo.form.BasicForm = function(el, config){
47296     this.allItems = [];
47297     this.childForms = [];
47298     Roo.apply(this, config);
47299     /*
47300      * The Roo.form.Field items in this form.
47301      * @type MixedCollection
47302      */
47303      
47304      
47305     this.items = new Roo.util.MixedCollection(false, function(o){
47306         return o.id || (o.id = Roo.id());
47307     });
47308     this.addEvents({
47309         /**
47310          * @event beforeaction
47311          * Fires before any action is performed. Return false to cancel the action.
47312          * @param {Form} this
47313          * @param {Action} action The action to be performed
47314          */
47315         beforeaction: true,
47316         /**
47317          * @event actionfailed
47318          * Fires when an action fails.
47319          * @param {Form} this
47320          * @param {Action} action The action that failed
47321          */
47322         actionfailed : true,
47323         /**
47324          * @event actioncomplete
47325          * Fires when an action is completed.
47326          * @param {Form} this
47327          * @param {Action} action The action that completed
47328          */
47329         actioncomplete : true
47330     });
47331     if(el){
47332         this.initEl(el);
47333     }
47334     Roo.form.BasicForm.superclass.constructor.call(this);
47335     
47336     Roo.form.BasicForm.popover.apply();
47337 };
47338
47339 Roo.extend(Roo.form.BasicForm, Roo.util.Observable, {
47340     /**
47341      * @cfg {String} method
47342      * The request method to use (GET or POST) for form actions if one isn't supplied in the action options.
47343      */
47344     /**
47345      * @cfg {DataReader} reader
47346      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when executing "load" actions.
47347      * This is optional as there is built-in support for processing JSON.
47348      */
47349     /**
47350      * @cfg {DataReader} errorReader
47351      * An Roo.data.DataReader (e.g. {@link Roo.data.XmlReader}) to be used to read data when reading validation errors on "submit" actions.
47352      * This is completely optional as there is built-in support for processing JSON.
47353      */
47354     /**
47355      * @cfg {String} url
47356      * The URL to use for form actions if one isn't supplied in the action options.
47357      */
47358     /**
47359      * @cfg {Boolean} fileUpload
47360      * Set to true if this form is a file upload.
47361      */
47362      
47363     /**
47364      * @cfg {Object} baseParams
47365      * Parameters to pass with all requests. e.g. baseParams: {id: '123', foo: 'bar'}.
47366      */
47367      /**
47368      
47369     /**
47370      * @cfg {Number} timeout Timeout for form actions in seconds (default is 30 seconds).
47371      */
47372     timeout: 30,
47373
47374     // private
47375     activeAction : null,
47376
47377     /**
47378      * @cfg {Boolean} trackResetOnLoad If set to true, form.reset() resets to the last loaded
47379      * or setValues() data instead of when the form was first created.
47380      */
47381     trackResetOnLoad : false,
47382     
47383     
47384     /**
47385      * childForms - used for multi-tab forms
47386      * @type {Array}
47387      */
47388     childForms : false,
47389     
47390     /**
47391      * allItems - full list of fields.
47392      * @type {Array}
47393      */
47394     allItems : false,
47395     
47396     /**
47397      * By default wait messages are displayed with Roo.MessageBox.wait. You can target a specific
47398      * element by passing it or its id or mask the form itself by passing in true.
47399      * @type Mixed
47400      */
47401     waitMsgTarget : false,
47402     
47403     /**
47404      * @type Boolean
47405      */
47406     disableMask : false,
47407     
47408     /**
47409      * @cfg {Boolean} errorMask (true|false) default false
47410      */
47411     errorMask : false,
47412     
47413     /**
47414      * @cfg {Number} maskOffset Default 100
47415      */
47416     maskOffset : 100,
47417
47418     // private
47419     initEl : function(el){
47420         this.el = Roo.get(el);
47421         this.id = this.el.id || Roo.id();
47422         this.el.on('submit', this.onSubmit, this);
47423         this.el.addClass('x-form');
47424     },
47425
47426     // private
47427     onSubmit : function(e){
47428         e.stopEvent();
47429     },
47430
47431     /**
47432      * Returns true if client-side validation on the form is successful.
47433      * @return Boolean
47434      */
47435     isValid : function(){
47436         var valid = true;
47437         var target = false;
47438         this.items.each(function(f){
47439             if(f.validate()){
47440                 return;
47441             }
47442             
47443             valid = false;
47444                 
47445             if(!target && f.el.isVisible(true)){
47446                 target = f;
47447             }
47448         });
47449         
47450         if(this.errorMask && !valid){
47451             Roo.form.BasicForm.popover.mask(this, target);
47452         }
47453         
47454         return valid;
47455     },
47456
47457     /**
47458      * DEPRICATED Returns true if any fields in this form have changed since their original load. 
47459      * @return Boolean
47460      */
47461     isDirty : function(){
47462         var dirty = false;
47463         this.items.each(function(f){
47464            if(f.isDirty()){
47465                dirty = true;
47466                return false;
47467            }
47468         });
47469         return dirty;
47470     },
47471     
47472     /**
47473      * Returns true if any fields in this form have changed since their original load. (New version)
47474      * @return Boolean
47475      */
47476     
47477     hasChanged : function()
47478     {
47479         var dirty = false;
47480         this.items.each(function(f){
47481            if(f.hasChanged()){
47482                dirty = true;
47483                return false;
47484            }
47485         });
47486         return dirty;
47487         
47488     },
47489     /**
47490      * Resets all hasChanged to 'false' -
47491      * The old 'isDirty' used 'original value..' however this breaks reset() and a few other things.
47492      * So hasChanged storage is only to be used for this purpose
47493      * @return Boolean
47494      */
47495     resetHasChanged : function()
47496     {
47497         this.items.each(function(f){
47498            f.resetHasChanged();
47499         });
47500         
47501     },
47502     
47503     
47504     /**
47505      * Performs a predefined action (submit or load) or custom actions you define on this form.
47506      * @param {String} actionName The name of the action type
47507      * @param {Object} options (optional) The options to pass to the action.  All of the config options listed
47508      * below are supported by both the submit and load actions unless otherwise noted (custom actions could also
47509      * accept other config options):
47510      * <pre>
47511 Property          Type             Description
47512 ----------------  ---------------  ----------------------------------------------------------------------------------
47513 url               String           The url for the action (defaults to the form's url)
47514 method            String           The form method to use (defaults to the form's method, or POST if not defined)
47515 params            String/Object    The params to pass (defaults to the form's baseParams, or none if not defined)
47516 clientValidation  Boolean          Applies to submit only.  Pass true to call form.isValid() prior to posting to
47517                                    validate the form on the client (defaults to false)
47518      * </pre>
47519      * @return {BasicForm} this
47520      */
47521     doAction : function(action, options){
47522         if(typeof action == 'string'){
47523             action = new Roo.form.Action.ACTION_TYPES[action](this, options);
47524         }
47525         if(this.fireEvent('beforeaction', this, action) !== false){
47526             this.beforeAction(action);
47527             action.run.defer(100, action);
47528         }
47529         return this;
47530     },
47531
47532     /**
47533      * Shortcut to do a submit action.
47534      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47535      * @return {BasicForm} this
47536      */
47537     submit : function(options){
47538         this.doAction('submit', options);
47539         return this;
47540     },
47541
47542     /**
47543      * Shortcut to do a load action.
47544      * @param {Object} options The options to pass to the action (see {@link #doAction} for details)
47545      * @return {BasicForm} this
47546      */
47547     load : function(options){
47548         this.doAction('load', options);
47549         return this;
47550     },
47551
47552     /**
47553      * Persists the values in this form into the passed Roo.data.Record object in a beginEdit/endEdit block.
47554      * @param {Record} record The record to edit
47555      * @return {BasicForm} this
47556      */
47557     updateRecord : function(record){
47558         record.beginEdit();
47559         var fs = record.fields;
47560         fs.each(function(f){
47561             var field = this.findField(f.name);
47562             if(field){
47563                 record.set(f.name, field.getValue());
47564             }
47565         }, this);
47566         record.endEdit();
47567         return this;
47568     },
47569
47570     /**
47571      * Loads an Roo.data.Record into this form.
47572      * @param {Record} record The record to load
47573      * @return {BasicForm} this
47574      */
47575     loadRecord : function(record){
47576         this.setValues(record.data);
47577         return this;
47578     },
47579
47580     // private
47581     beforeAction : function(action){
47582         var o = action.options;
47583         
47584         if(!this.disableMask) {
47585             if(this.waitMsgTarget === true){
47586                 this.el.mask(o.waitMsg || "Sending", 'x-mask-loading');
47587             }else if(this.waitMsgTarget){
47588                 this.waitMsgTarget = Roo.get(this.waitMsgTarget);
47589                 this.waitMsgTarget.mask(o.waitMsg || "Sending", 'x-mask-loading');
47590             }else {
47591                 Roo.MessageBox.wait(o.waitMsg || "Sending", o.waitTitle || this.waitTitle || 'Please Wait...');
47592             }
47593         }
47594         
47595          
47596     },
47597
47598     // private
47599     afterAction : function(action, success){
47600         this.activeAction = null;
47601         var o = action.options;
47602         
47603         if(!this.disableMask) {
47604             if(this.waitMsgTarget === true){
47605                 this.el.unmask();
47606             }else if(this.waitMsgTarget){
47607                 this.waitMsgTarget.unmask();
47608             }else{
47609                 Roo.MessageBox.updateProgress(1);
47610                 Roo.MessageBox.hide();
47611             }
47612         }
47613         
47614         if(success){
47615             if(o.reset){
47616                 this.reset();
47617             }
47618             Roo.callback(o.success, o.scope, [this, action]);
47619             this.fireEvent('actioncomplete', this, action);
47620             
47621         }else{
47622             
47623             // failure condition..
47624             // we have a scenario where updates need confirming.
47625             // eg. if a locking scenario exists..
47626             // we look for { errors : { needs_confirm : true }} in the response.
47627             if (
47628                 (typeof(action.result) != 'undefined')  &&
47629                 (typeof(action.result.errors) != 'undefined')  &&
47630                 (typeof(action.result.errors.needs_confirm) != 'undefined')
47631            ){
47632                 var _t = this;
47633                 Roo.MessageBox.confirm(
47634                     "Change requires confirmation",
47635                     action.result.errorMsg,
47636                     function(r) {
47637                         if (r != 'yes') {
47638                             return;
47639                         }
47640                         _t.doAction('submit', { params :  { _submit_confirmed : 1 } }  );
47641                     }
47642                     
47643                 );
47644                 
47645                 
47646                 
47647                 return;
47648             }
47649             
47650             Roo.callback(o.failure, o.scope, [this, action]);
47651             // show an error message if no failed handler is set..
47652             if (!this.hasListener('actionfailed')) {
47653                 Roo.MessageBox.alert("Error",
47654                     (typeof(action.result) != 'undefined' && typeof(action.result.errorMsg) != 'undefined') ?
47655                         action.result.errorMsg :
47656                         "Saving Failed, please check your entries or try again"
47657                 );
47658             }
47659             
47660             this.fireEvent('actionfailed', this, action);
47661         }
47662         
47663     },
47664
47665     /**
47666      * Find a Roo.form.Field in this form by id, dataIndex, name or hiddenName
47667      * @param {String} id The value to search for
47668      * @return Field
47669      */
47670     findField : function(id){
47671         var field = this.items.get(id);
47672         if(!field){
47673             this.items.each(function(f){
47674                 if(f.isFormField && (f.dataIndex == id || f.id == id || f.getName() == id)){
47675                     field = f;
47676                     return false;
47677                 }
47678             });
47679         }
47680         return field || null;
47681     },
47682
47683     /**
47684      * Add a secondary form to this one, 
47685      * Used to provide tabbed forms. One form is primary, with hidden values 
47686      * which mirror the elements from the other forms.
47687      * 
47688      * @param {Roo.form.Form} form to add.
47689      * 
47690      */
47691     addForm : function(form)
47692     {
47693        
47694         if (this.childForms.indexOf(form) > -1) {
47695             // already added..
47696             return;
47697         }
47698         this.childForms.push(form);
47699         var n = '';
47700         Roo.each(form.allItems, function (fe) {
47701             
47702             n = typeof(fe.getName) == 'undefined' ? fe.name : fe.getName();
47703             if (this.findField(n)) { // already added..
47704                 return;
47705             }
47706             var add = new Roo.form.Hidden({
47707                 name : n
47708             });
47709             add.render(this.el);
47710             
47711             this.add( add );
47712         }, this);
47713         
47714     },
47715     /**
47716      * Mark fields in this form invalid in bulk.
47717      * @param {Array/Object} errors Either an array in the form [{id:'fieldId', msg:'The message'},...] or an object hash of {id: msg, id2: msg2}
47718      * @return {BasicForm} this
47719      */
47720     markInvalid : function(errors){
47721         if(errors instanceof Array){
47722             for(var i = 0, len = errors.length; i < len; i++){
47723                 var fieldError = errors[i];
47724                 var f = this.findField(fieldError.id);
47725                 if(f){
47726                     f.markInvalid(fieldError.msg);
47727                 }
47728             }
47729         }else{
47730             var field, id;
47731             for(id in errors){
47732                 if(typeof errors[id] != 'function' && (field = this.findField(id))){
47733                     field.markInvalid(errors[id]);
47734                 }
47735             }
47736         }
47737         Roo.each(this.childForms || [], function (f) {
47738             f.markInvalid(errors);
47739         });
47740         
47741         return this;
47742     },
47743
47744     /**
47745      * Set values for fields in this form in bulk.
47746      * @param {Array/Object} values Either an array in the form [{id:'fieldId', value:'foo'},...] or an object hash of {id: value, id2: value2}
47747      * @return {BasicForm} this
47748      */
47749     setValues : function(values){
47750         if(values instanceof Array){ // array of objects
47751             for(var i = 0, len = values.length; i < len; i++){
47752                 var v = values[i];
47753                 var f = this.findField(v.id);
47754                 if(f){
47755                     f.setValue(v.value);
47756                     if(this.trackResetOnLoad){
47757                         f.originalValue = f.getValue();
47758                     }
47759                 }
47760             }
47761         }else{ // object hash
47762             var field, id;
47763             for(id in values){
47764                 if(typeof values[id] != 'function' && (field = this.findField(id))){
47765                     
47766                     if (field.setFromData && 
47767                         field.valueField && 
47768                         field.displayField &&
47769                         // combos' with local stores can 
47770                         // be queried via setValue()
47771                         // to set their value..
47772                         (field.store && !field.store.isLocal)
47773                         ) {
47774                         // it's a combo
47775                         var sd = { };
47776                         sd[field.valueField] = typeof(values[field.hiddenName]) == 'undefined' ? '' : values[field.hiddenName];
47777                         sd[field.displayField] = typeof(values[field.name]) == 'undefined' ? '' : values[field.name];
47778                         field.setFromData(sd);
47779                         
47780                     } else {
47781                         field.setValue(values[id]);
47782                     }
47783                     
47784                     
47785                     if(this.trackResetOnLoad){
47786                         field.originalValue = field.getValue();
47787                     }
47788                 }
47789             }
47790         }
47791         this.resetHasChanged();
47792         
47793         
47794         Roo.each(this.childForms || [], function (f) {
47795             f.setValues(values);
47796             f.resetHasChanged();
47797         });
47798                 
47799         return this;
47800     },
47801  
47802     /**
47803      * Returns the fields in this form as an object with key/value pairs. If multiple fields exist with the same name
47804      * they are returned as an array.
47805      * @param {Boolean} asString
47806      * @return {Object}
47807      */
47808     getValues : function(asString){
47809         if (this.childForms) {
47810             // copy values from the child forms
47811             Roo.each(this.childForms, function (f) {
47812                 this.setValues(f.getValues());
47813             }, this);
47814         }
47815         
47816         // use formdata
47817         if (typeof(FormData) != 'undefined' && asString !== true) {
47818             var fd = (new FormData(this.el.dom)).entries();
47819             var ret = {};
47820             var ent = fd.next();
47821             while (!ent.done) {
47822                 ret[ent.value[0]] = ent.value[1]; // not sure how this will handle duplicates..
47823                 ent = fd.next();
47824             };
47825             return ret;
47826         }
47827         
47828         
47829         var fs = Roo.lib.Ajax.serializeForm(this.el.dom);
47830         if(asString === true){
47831             return fs;
47832         }
47833         return Roo.urlDecode(fs);
47834     },
47835     
47836     /**
47837      * Returns the fields in this form as an object with key/value pairs. 
47838      * This differs from getValues as it calls getValue on each child item, rather than using dom data.
47839      * @return {Object}
47840      */
47841     getFieldValues : function(with_hidden)
47842     {
47843         if (this.childForms) {
47844             // copy values from the child forms
47845             // should this call getFieldValues - probably not as we do not currently copy
47846             // hidden fields when we generate..
47847             Roo.each(this.childForms, function (f) {
47848                 this.setValues(f.getValues());
47849             }, this);
47850         }
47851         
47852         var ret = {};
47853         this.items.each(function(f){
47854             if (!f.getName()) {
47855                 return;
47856             }
47857             var v = f.getValue();
47858             if (f.inputType =='radio') {
47859                 if (typeof(ret[f.getName()]) == 'undefined') {
47860                     ret[f.getName()] = ''; // empty..
47861                 }
47862                 
47863                 if (!f.el.dom.checked) {
47864                     return;
47865                     
47866                 }
47867                 v = f.el.dom.value;
47868                 
47869             }
47870             
47871             // not sure if this supported any more..
47872             if ((typeof(v) == 'object') && f.getRawValue) {
47873                 v = f.getRawValue() ; // dates..
47874             }
47875             // combo boxes where name != hiddenName...
47876             if (f.name != f.getName()) {
47877                 ret[f.name] = f.getRawValue();
47878             }
47879             ret[f.getName()] = v;
47880         });
47881         
47882         return ret;
47883     },
47884
47885     /**
47886      * Clears all invalid messages in this form.
47887      * @return {BasicForm} this
47888      */
47889     clearInvalid : function(){
47890         this.items.each(function(f){
47891            f.clearInvalid();
47892         });
47893         
47894         Roo.each(this.childForms || [], function (f) {
47895             f.clearInvalid();
47896         });
47897         
47898         
47899         return this;
47900     },
47901
47902     /**
47903      * Resets this form.
47904      * @return {BasicForm} this
47905      */
47906     reset : function(){
47907         this.items.each(function(f){
47908             f.reset();
47909         });
47910         
47911         Roo.each(this.childForms || [], function (f) {
47912             f.reset();
47913         });
47914         this.resetHasChanged();
47915         
47916         return this;
47917     },
47918
47919     /**
47920      * Add Roo.form components to this form.
47921      * @param {Field} field1
47922      * @param {Field} field2 (optional)
47923      * @param {Field} etc (optional)
47924      * @return {BasicForm} this
47925      */
47926     add : function(){
47927         this.items.addAll(Array.prototype.slice.call(arguments, 0));
47928         return this;
47929     },
47930
47931
47932     /**
47933      * Removes a field from the items collection (does NOT remove its markup).
47934      * @param {Field} field
47935      * @return {BasicForm} this
47936      */
47937     remove : function(field){
47938         this.items.remove(field);
47939         return this;
47940     },
47941
47942     /**
47943      * Looks at the fields in this form, checks them for an id attribute,
47944      * and calls applyTo on the existing dom element with that id.
47945      * @return {BasicForm} this
47946      */
47947     render : function(){
47948         this.items.each(function(f){
47949             if(f.isFormField && !f.rendered && document.getElementById(f.id)){ // if the element exists
47950                 f.applyTo(f.id);
47951             }
47952         });
47953         return this;
47954     },
47955
47956     /**
47957      * Calls {@link Ext#apply} for all fields in this form with the passed object.
47958      * @param {Object} values
47959      * @return {BasicForm} this
47960      */
47961     applyToFields : function(o){
47962         this.items.each(function(f){
47963            Roo.apply(f, o);
47964         });
47965         return this;
47966     },
47967
47968     /**
47969      * Calls {@link Ext#applyIf} for all field in this form with the passed object.
47970      * @param {Object} values
47971      * @return {BasicForm} this
47972      */
47973     applyIfToFields : function(o){
47974         this.items.each(function(f){
47975            Roo.applyIf(f, o);
47976         });
47977         return this;
47978     }
47979 });
47980
47981 // back compat
47982 Roo.BasicForm = Roo.form.BasicForm;
47983
47984 Roo.apply(Roo.form.BasicForm, {
47985     
47986     popover : {
47987         
47988         padding : 5,
47989         
47990         isApplied : false,
47991         
47992         isMasked : false,
47993         
47994         form : false,
47995         
47996         target : false,
47997         
47998         intervalID : false,
47999         
48000         maskEl : false,
48001         
48002         apply : function()
48003         {
48004             if(this.isApplied){
48005                 return;
48006             }
48007             
48008             this.maskEl = {
48009                 top : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-top-mask" }, true),
48010                 left : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-left-mask" }, true),
48011                 bottom : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-bottom-mask" }, true),
48012                 right : Roo.DomHelper.append(Roo.get(document.body), { tag: "div", cls:"x-dlg-mask roo-form-right-mask" }, true)
48013             };
48014             
48015             this.maskEl.top.enableDisplayMode("block");
48016             this.maskEl.left.enableDisplayMode("block");
48017             this.maskEl.bottom.enableDisplayMode("block");
48018             this.maskEl.right.enableDisplayMode("block");
48019             
48020             Roo.get(document.body).on('click', function(){
48021                 this.unmask();
48022             }, this);
48023             
48024             Roo.get(document.body).on('touchstart', function(){
48025                 this.unmask();
48026             }, this);
48027             
48028             this.isApplied = true
48029         },
48030         
48031         mask : function(form, target)
48032         {
48033             this.form = form;
48034             
48035             this.target = target;
48036             
48037             if(!this.form.errorMask || !target.el){
48038                 return;
48039             }
48040             
48041             var scrollable = this.target.el.findScrollableParent() || this.target.el.findParent('div.x-layout-active-content', 100, true) || Roo.get(document.body);
48042             
48043             var ot = this.target.el.calcOffsetsTo(scrollable);
48044             
48045             var scrollTo = ot[1] - this.form.maskOffset;
48046             
48047             scrollTo = Math.min(scrollTo, scrollable.dom.scrollHeight);
48048             
48049             scrollable.scrollTo('top', scrollTo);
48050             
48051             var el = this.target.wrap || this.target.el;
48052             
48053             var box = el.getBox();
48054             
48055             this.maskEl.top.setStyle('position', 'absolute');
48056             this.maskEl.top.setStyle('z-index', 10000);
48057             this.maskEl.top.setSize(Roo.lib.Dom.getDocumentWidth(), box.y - this.padding);
48058             this.maskEl.top.setLeft(0);
48059             this.maskEl.top.setTop(0);
48060             this.maskEl.top.show();
48061             
48062             this.maskEl.left.setStyle('position', 'absolute');
48063             this.maskEl.left.setStyle('z-index', 10000);
48064             this.maskEl.left.setSize(box.x - this.padding, box.height + this.padding * 2);
48065             this.maskEl.left.setLeft(0);
48066             this.maskEl.left.setTop(box.y - this.padding);
48067             this.maskEl.left.show();
48068
48069             this.maskEl.bottom.setStyle('position', 'absolute');
48070             this.maskEl.bottom.setStyle('z-index', 10000);
48071             this.maskEl.bottom.setSize(Roo.lib.Dom.getDocumentWidth(), Roo.lib.Dom.getDocumentHeight() - box.bottom - this.padding);
48072             this.maskEl.bottom.setLeft(0);
48073             this.maskEl.bottom.setTop(box.bottom + this.padding);
48074             this.maskEl.bottom.show();
48075
48076             this.maskEl.right.setStyle('position', 'absolute');
48077             this.maskEl.right.setStyle('z-index', 10000);
48078             this.maskEl.right.setSize(Roo.lib.Dom.getDocumentWidth() - box.right - this.padding, box.height + this.padding * 2);
48079             this.maskEl.right.setLeft(box.right + this.padding);
48080             this.maskEl.right.setTop(box.y - this.padding);
48081             this.maskEl.right.show();
48082
48083             this.intervalID = window.setInterval(function() {
48084                 Roo.form.BasicForm.popover.unmask();
48085             }, 10000);
48086
48087             window.onwheel = function(){ return false;};
48088             
48089             (function(){ this.isMasked = true; }).defer(500, this);
48090             
48091         },
48092         
48093         unmask : function()
48094         {
48095             if(!this.isApplied || !this.isMasked || !this.form || !this.target || !this.form.errorMask){
48096                 return;
48097             }
48098             
48099             this.maskEl.top.setStyle('position', 'absolute');
48100             this.maskEl.top.setSize(0, 0).setXY([0, 0]);
48101             this.maskEl.top.hide();
48102
48103             this.maskEl.left.setStyle('position', 'absolute');
48104             this.maskEl.left.setSize(0, 0).setXY([0, 0]);
48105             this.maskEl.left.hide();
48106
48107             this.maskEl.bottom.setStyle('position', 'absolute');
48108             this.maskEl.bottom.setSize(0, 0).setXY([0, 0]);
48109             this.maskEl.bottom.hide();
48110
48111             this.maskEl.right.setStyle('position', 'absolute');
48112             this.maskEl.right.setSize(0, 0).setXY([0, 0]);
48113             this.maskEl.right.hide();
48114             
48115             window.onwheel = function(){ return true;};
48116             
48117             if(this.intervalID){
48118                 window.clearInterval(this.intervalID);
48119                 this.intervalID = false;
48120             }
48121             
48122             this.isMasked = false;
48123             
48124         }
48125         
48126     }
48127     
48128 });/*
48129  * Based on:
48130  * Ext JS Library 1.1.1
48131  * Copyright(c) 2006-2007, Ext JS, LLC.
48132  *
48133  * Originally Released Under LGPL - original licence link has changed is not relivant.
48134  *
48135  * Fork - LGPL
48136  * <script type="text/javascript">
48137  */
48138
48139 /**
48140  * @class Roo.form.Form
48141  * @extends Roo.form.BasicForm
48142  * Adds the ability to dynamically render forms with JavaScript to {@link Roo.form.BasicForm}.
48143  * @constructor
48144  * @param {Object} config Configuration options
48145  */
48146 Roo.form.Form = function(config){
48147     var xitems =  [];
48148     if (config.items) {
48149         xitems = config.items;
48150         delete config.items;
48151     }
48152    
48153     
48154     Roo.form.Form.superclass.constructor.call(this, null, config);
48155     this.url = this.url || this.action;
48156     if(!this.root){
48157         this.root = new Roo.form.Layout(Roo.applyIf({
48158             id: Roo.id()
48159         }, config));
48160     }
48161     this.active = this.root;
48162     /**
48163      * Array of all the buttons that have been added to this form via {@link addButton}
48164      * @type Array
48165      */
48166     this.buttons = [];
48167     this.allItems = [];
48168     this.addEvents({
48169         /**
48170          * @event clientvalidation
48171          * If the monitorValid config option is true, this event fires repetitively to notify of valid state
48172          * @param {Form} this
48173          * @param {Boolean} valid true if the form has passed client-side validation
48174          */
48175         clientvalidation: true,
48176         /**
48177          * @event rendered
48178          * Fires when the form is rendered
48179          * @param {Roo.form.Form} form
48180          */
48181         rendered : true
48182     });
48183     
48184     if (this.progressUrl) {
48185             // push a hidden field onto the list of fields..
48186             this.addxtype( {
48187                     xns: Roo.form, 
48188                     xtype : 'Hidden', 
48189                     name : 'UPLOAD_IDENTIFIER' 
48190             });
48191         }
48192         
48193     
48194     Roo.each(xitems, this.addxtype, this);
48195     
48196 };
48197
48198 Roo.extend(Roo.form.Form, Roo.form.BasicForm, {
48199     /**
48200      * @cfg {Number} labelWidth The width of labels. This property cascades to child containers.
48201      */
48202     /**
48203      * @cfg {String} itemCls A css class to apply to the x-form-item of fields. This property cascades to child containers.
48204      */
48205     /**
48206      * @cfg {String} buttonAlign Valid values are "left," "center" and "right" (defaults to "center")
48207      */
48208     buttonAlign:'center',
48209
48210     /**
48211      * @cfg {Number} minButtonWidth Minimum width of all buttons in pixels (defaults to 75)
48212      */
48213     minButtonWidth:75,
48214
48215     /**
48216      * @cfg {String} labelAlign Valid values are "left," "top" and "right" (defaults to "left").
48217      * This property cascades to child containers if not set.
48218      */
48219     labelAlign:'left',
48220
48221     /**
48222      * @cfg {Boolean} monitorValid If true the form monitors its valid state <b>client-side</b> and
48223      * fires a looping event with that state. This is required to bind buttons to the valid
48224      * state using the config value formBind:true on the button.
48225      */
48226     monitorValid : false,
48227
48228     /**
48229      * @cfg {Number} monitorPoll The milliseconds to poll valid state, ignored if monitorValid is not true (defaults to 200)
48230      */
48231     monitorPoll : 200,
48232     
48233     /**
48234      * @cfg {String} progressUrl - Url to return progress data 
48235      */
48236     
48237     progressUrl : false,
48238     /**
48239      * @cfg {boolean|FormData} formData - true to use new 'FormData' post, or set to a new FormData({dom form}) Object, if
48240      * sending a formdata with extra parameters - eg uploaded elements.
48241      */
48242     
48243     formData : false,
48244     
48245     /**
48246      * Opens a new {@link Roo.form.Column} container in the layout stack. If fields are passed after the config, the
48247      * fields are added and the column is closed. If no fields are passed the column remains open
48248      * until end() is called.
48249      * @param {Object} config The config to pass to the column
48250      * @param {Field} field1 (optional)
48251      * @param {Field} field2 (optional)
48252      * @param {Field} etc (optional)
48253      * @return Column The column container object
48254      */
48255     column : function(c){
48256         var col = new Roo.form.Column(c);
48257         this.start(col);
48258         if(arguments.length > 1){ // duplicate code required because of Opera
48259             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48260             this.end();
48261         }
48262         return col;
48263     },
48264
48265     /**
48266      * Opens a new {@link Roo.form.FieldSet} container in the layout stack. If fields are passed after the config, the
48267      * fields are added and the fieldset is closed. If no fields are passed the fieldset remains open
48268      * until end() is called.
48269      * @param {Object} config The config to pass to the fieldset
48270      * @param {Field} field1 (optional)
48271      * @param {Field} field2 (optional)
48272      * @param {Field} etc (optional)
48273      * @return FieldSet The fieldset container object
48274      */
48275     fieldset : function(c){
48276         var fs = new Roo.form.FieldSet(c);
48277         this.start(fs);
48278         if(arguments.length > 1){ // duplicate code required because of Opera
48279             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48280             this.end();
48281         }
48282         return fs;
48283     },
48284
48285     /**
48286      * Opens a new {@link Roo.form.Layout} container in the layout stack. If fields are passed after the config, the
48287      * fields are added and the container is closed. If no fields are passed the container remains open
48288      * until end() is called.
48289      * @param {Object} config The config to pass to the Layout
48290      * @param {Field} field1 (optional)
48291      * @param {Field} field2 (optional)
48292      * @param {Field} etc (optional)
48293      * @return Layout The container object
48294      */
48295     container : function(c){
48296         var l = new Roo.form.Layout(c);
48297         this.start(l);
48298         if(arguments.length > 1){ // duplicate code required because of Opera
48299             this.add.apply(this, Array.prototype.slice.call(arguments, 1));
48300             this.end();
48301         }
48302         return l;
48303     },
48304
48305     /**
48306      * Opens the passed container in the layout stack. The container can be any {@link Roo.form.Layout} or subclass.
48307      * @param {Object} container A Roo.form.Layout or subclass of Layout
48308      * @return {Form} this
48309      */
48310     start : function(c){
48311         // cascade label info
48312         Roo.applyIf(c, {'labelAlign': this.active.labelAlign, 'labelWidth': this.active.labelWidth, 'itemCls': this.active.itemCls});
48313         this.active.stack.push(c);
48314         c.ownerCt = this.active;
48315         this.active = c;
48316         return this;
48317     },
48318
48319     /**
48320      * Closes the current open container
48321      * @return {Form} this
48322      */
48323     end : function(){
48324         if(this.active == this.root){
48325             return this;
48326         }
48327         this.active = this.active.ownerCt;
48328         return this;
48329     },
48330
48331     /**
48332      * Add Roo.form components to the current open container (e.g. column, fieldset, etc.).  Fields added via this method
48333      * can also be passed with an additional property of fieldLabel, which if supplied, will provide the text to display
48334      * as the label of the field.
48335      * @param {Field} field1
48336      * @param {Field} field2 (optional)
48337      * @param {Field} etc. (optional)
48338      * @return {Form} this
48339      */
48340     add : function(){
48341         this.active.stack.push.apply(this.active.stack, arguments);
48342         this.allItems.push.apply(this.allItems,arguments);
48343         var r = [];
48344         for(var i = 0, a = arguments, len = a.length; i < len; i++) {
48345             if(a[i].isFormField){
48346                 r.push(a[i]);
48347             }
48348         }
48349         if(r.length > 0){
48350             Roo.form.Form.superclass.add.apply(this, r);
48351         }
48352         return this;
48353     },
48354     
48355
48356     
48357     
48358     
48359      /**
48360      * Find any element that has been added to a form, using it's ID or name
48361      * This can include framesets, columns etc. along with regular fields..
48362      * @param {String} id - id or name to find.
48363      
48364      * @return {Element} e - or false if nothing found.
48365      */
48366     findbyId : function(id)
48367     {
48368         var ret = false;
48369         if (!id) {
48370             return ret;
48371         }
48372         Roo.each(this.allItems, function(f){
48373             if (f.id == id || f.name == id ){
48374                 ret = f;
48375                 return false;
48376             }
48377         });
48378         return ret;
48379     },
48380
48381     
48382     
48383     /**
48384      * Render this form into the passed container. This should only be called once!
48385      * @param {String/HTMLElement/Element} container The element this component should be rendered into
48386      * @return {Form} this
48387      */
48388     render : function(ct)
48389     {
48390         
48391         
48392         
48393         ct = Roo.get(ct);
48394         var o = this.autoCreate || {
48395             tag: 'form',
48396             method : this.method || 'POST',
48397             id : this.id || Roo.id()
48398         };
48399         this.initEl(ct.createChild(o));
48400
48401         this.root.render(this.el);
48402         
48403        
48404              
48405         this.items.each(function(f){
48406             f.render('x-form-el-'+f.id);
48407         });
48408
48409         if(this.buttons.length > 0){
48410             // tables are required to maintain order and for correct IE layout
48411             var tb = this.el.createChild({cls:'x-form-btns-ct', cn: {
48412                 cls:"x-form-btns x-form-btns-"+this.buttonAlign,
48413                 html:'<table cellspacing="0"><tbody><tr></tr></tbody></table><div class="x-clear"></div>'
48414             }}, null, true);
48415             var tr = tb.getElementsByTagName('tr')[0];
48416             for(var i = 0, len = this.buttons.length; i < len; i++) {
48417                 var b = this.buttons[i];
48418                 var td = document.createElement('td');
48419                 td.className = 'x-form-btn-td';
48420                 b.render(tr.appendChild(td));
48421             }
48422         }
48423         if(this.monitorValid){ // initialize after render
48424             this.startMonitoring();
48425         }
48426         this.fireEvent('rendered', this);
48427         return this;
48428     },
48429
48430     /**
48431      * Adds a button to the footer of the form - this <b>must</b> be called before the form is rendered.
48432      * @param {String/Object} config A string becomes the button text, an object can either be a Button config
48433      * object or a valid Roo.DomHelper element config
48434      * @param {Function} handler The function called when the button is clicked
48435      * @param {Object} scope (optional) The scope of the handler function
48436      * @return {Roo.Button}
48437      */
48438     addButton : function(config, handler, scope){
48439         var bc = {
48440             handler: handler,
48441             scope: scope,
48442             minWidth: this.minButtonWidth,
48443             hideParent:true
48444         };
48445         if(typeof config == "string"){
48446             bc.text = config;
48447         }else{
48448             Roo.apply(bc, config);
48449         }
48450         var btn = new Roo.Button(null, bc);
48451         this.buttons.push(btn);
48452         return btn;
48453     },
48454
48455      /**
48456      * Adds a series of form elements (using the xtype property as the factory method.
48457      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column, (and 'end' to close a block)
48458      * @param {Object} config 
48459      */
48460     
48461     addxtype : function()
48462     {
48463         var ar = Array.prototype.slice.call(arguments, 0);
48464         var ret = false;
48465         for(var i = 0; i < ar.length; i++) {
48466             if (!ar[i]) {
48467                 continue; // skip -- if this happends something invalid got sent, we 
48468                 // should ignore it, as basically that interface element will not show up
48469                 // and that should be pretty obvious!!
48470             }
48471             
48472             if (Roo.form[ar[i].xtype]) {
48473                 ar[i].form = this;
48474                 var fe = Roo.factory(ar[i], Roo.form);
48475                 if (!ret) {
48476                     ret = fe;
48477                 }
48478                 fe.form = this;
48479                 if (fe.store) {
48480                     fe.store.form = this;
48481                 }
48482                 if (fe.isLayout) {  
48483                          
48484                     this.start(fe);
48485                     this.allItems.push(fe);
48486                     if (fe.items && fe.addxtype) {
48487                         fe.addxtype.apply(fe, fe.items);
48488                         delete fe.items;
48489                     }
48490                      this.end();
48491                     continue;
48492                 }
48493                 
48494                 
48495                  
48496                 this.add(fe);
48497               //  console.log('adding ' + ar[i].xtype);
48498             }
48499             if (ar[i].xtype == 'Button') {  
48500                 //console.log('adding button');
48501                 //console.log(ar[i]);
48502                 this.addButton(ar[i]);
48503                 this.allItems.push(fe);
48504                 continue;
48505             }
48506             
48507             if (ar[i].xtype == 'end') { // so we can add fieldsets... / layout etc.
48508                 alert('end is not supported on xtype any more, use items');
48509             //    this.end();
48510             //    //console.log('adding end');
48511             }
48512             
48513         }
48514         return ret;
48515     },
48516     
48517     /**
48518      * Starts monitoring of the valid state of this form. Usually this is done by passing the config
48519      * option "monitorValid"
48520      */
48521     startMonitoring : function(){
48522         if(!this.bound){
48523             this.bound = true;
48524             Roo.TaskMgr.start({
48525                 run : this.bindHandler,
48526                 interval : this.monitorPoll || 200,
48527                 scope: this
48528             });
48529         }
48530     },
48531
48532     /**
48533      * Stops monitoring of the valid state of this form
48534      */
48535     stopMonitoring : function(){
48536         this.bound = false;
48537     },
48538
48539     // private
48540     bindHandler : function(){
48541         if(!this.bound){
48542             return false; // stops binding
48543         }
48544         var valid = true;
48545         this.items.each(function(f){
48546             if(!f.isValid(true)){
48547                 valid = false;
48548                 return false;
48549             }
48550         });
48551         for(var i = 0, len = this.buttons.length; i < len; i++){
48552             var btn = this.buttons[i];
48553             if(btn.formBind === true && btn.disabled === valid){
48554                 btn.setDisabled(!valid);
48555             }
48556         }
48557         this.fireEvent('clientvalidation', this, valid);
48558     }
48559     
48560     
48561     
48562     
48563     
48564     
48565     
48566     
48567 });
48568
48569
48570 // back compat
48571 Roo.Form = Roo.form.Form;
48572 /*
48573  * Based on:
48574  * Ext JS Library 1.1.1
48575  * Copyright(c) 2006-2007, Ext JS, LLC.
48576  *
48577  * Originally Released Under LGPL - original licence link has changed is not relivant.
48578  *
48579  * Fork - LGPL
48580  * <script type="text/javascript">
48581  */
48582
48583 // as we use this in bootstrap.
48584 Roo.namespace('Roo.form');
48585  /**
48586  * @class Roo.form.Action
48587  * Internal Class used to handle form actions
48588  * @constructor
48589  * @param {Roo.form.BasicForm} el The form element or its id
48590  * @param {Object} config Configuration options
48591  */
48592
48593  
48594  
48595 // define the action interface
48596 Roo.form.Action = function(form, options){
48597     this.form = form;
48598     this.options = options || {};
48599 };
48600 /**
48601  * Client Validation Failed
48602  * @const 
48603  */
48604 Roo.form.Action.CLIENT_INVALID = 'client';
48605 /**
48606  * Server Validation Failed
48607  * @const 
48608  */
48609 Roo.form.Action.SERVER_INVALID = 'server';
48610  /**
48611  * Connect to Server Failed
48612  * @const 
48613  */
48614 Roo.form.Action.CONNECT_FAILURE = 'connect';
48615 /**
48616  * Reading Data from Server Failed
48617  * @const 
48618  */
48619 Roo.form.Action.LOAD_FAILURE = 'load';
48620
48621 Roo.form.Action.prototype = {
48622     type : 'default',
48623     failureType : undefined,
48624     response : undefined,
48625     result : undefined,
48626
48627     // interface method
48628     run : function(options){
48629
48630     },
48631
48632     // interface method
48633     success : function(response){
48634
48635     },
48636
48637     // interface method
48638     handleResponse : function(response){
48639
48640     },
48641
48642     // default connection failure
48643     failure : function(response){
48644         
48645         this.response = response;
48646         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48647         this.form.afterAction(this, false);
48648     },
48649
48650     processResponse : function(response){
48651         this.response = response;
48652         if(!response.responseText){
48653             return true;
48654         }
48655         this.result = this.handleResponse(response);
48656         return this.result;
48657     },
48658
48659     // utility functions used internally
48660     getUrl : function(appendParams){
48661         var url = this.options.url || this.form.url || this.form.el.dom.action;
48662         if(appendParams){
48663             var p = this.getParams();
48664             if(p){
48665                 url += (url.indexOf('?') != -1 ? '&' : '?') + p;
48666             }
48667         }
48668         return url;
48669     },
48670
48671     getMethod : function(){
48672         return (this.options.method || this.form.method || this.form.el.dom.method || 'POST').toUpperCase();
48673     },
48674
48675     getParams : function(){
48676         var bp = this.form.baseParams;
48677         var p = this.options.params;
48678         if(p){
48679             if(typeof p == "object"){
48680                 p = Roo.urlEncode(Roo.applyIf(p, bp));
48681             }else if(typeof p == 'string' && bp){
48682                 p += '&' + Roo.urlEncode(bp);
48683             }
48684         }else if(bp){
48685             p = Roo.urlEncode(bp);
48686         }
48687         return p;
48688     },
48689
48690     createCallback : function(){
48691         return {
48692             success: this.success,
48693             failure: this.failure,
48694             scope: this,
48695             timeout: (this.form.timeout*1000),
48696             upload: this.form.fileUpload ? this.success : undefined
48697         };
48698     }
48699 };
48700
48701 Roo.form.Action.Submit = function(form, options){
48702     Roo.form.Action.Submit.superclass.constructor.call(this, form, options);
48703 };
48704
48705 Roo.extend(Roo.form.Action.Submit, Roo.form.Action, {
48706     type : 'submit',
48707
48708     haveProgress : false,
48709     uploadComplete : false,
48710     
48711     // uploadProgress indicator.
48712     uploadProgress : function()
48713     {
48714         if (!this.form.progressUrl) {
48715             return;
48716         }
48717         
48718         if (!this.haveProgress) {
48719             Roo.MessageBox.progress("Uploading", "Uploading");
48720         }
48721         if (this.uploadComplete) {
48722            Roo.MessageBox.hide();
48723            return;
48724         }
48725         
48726         this.haveProgress = true;
48727    
48728         var uid = this.form.findField('UPLOAD_IDENTIFIER').getValue();
48729         
48730         var c = new Roo.data.Connection();
48731         c.request({
48732             url : this.form.progressUrl,
48733             params: {
48734                 id : uid
48735             },
48736             method: 'GET',
48737             success : function(req){
48738                //console.log(data);
48739                 var rdata = false;
48740                 var edata;
48741                 try  {
48742                    rdata = Roo.decode(req.responseText)
48743                 } catch (e) {
48744                     Roo.log("Invalid data from server..");
48745                     Roo.log(edata);
48746                     return;
48747                 }
48748                 if (!rdata || !rdata.success) {
48749                     Roo.log(rdata);
48750                     Roo.MessageBox.alert(Roo.encode(rdata));
48751                     return;
48752                 }
48753                 var data = rdata.data;
48754                 
48755                 if (this.uploadComplete) {
48756                    Roo.MessageBox.hide();
48757                    return;
48758                 }
48759                    
48760                 if (data){
48761                     Roo.MessageBox.updateProgress(data.bytes_uploaded/data.bytes_total,
48762                        Math.floor((data.bytes_total - data.bytes_uploaded)/1000) + 'k remaining'
48763                     );
48764                 }
48765                 this.uploadProgress.defer(2000,this);
48766             },
48767        
48768             failure: function(data) {
48769                 Roo.log('progress url failed ');
48770                 Roo.log(data);
48771             },
48772             scope : this
48773         });
48774            
48775     },
48776     
48777     
48778     run : function()
48779     {
48780         // run get Values on the form, so it syncs any secondary forms.
48781         this.form.getValues();
48782         
48783         var o = this.options;
48784         var method = this.getMethod();
48785         var isPost = method == 'POST';
48786         if(o.clientValidation === false || this.form.isValid()){
48787             
48788             if (this.form.progressUrl) {
48789                 this.form.findField('UPLOAD_IDENTIFIER').setValue(
48790                     (new Date() * 1) + '' + Math.random());
48791                     
48792             } 
48793             
48794             
48795             Roo.Ajax.request(Roo.apply(this.createCallback(), {
48796                 form:this.form.el.dom,
48797                 url:this.getUrl(!isPost),
48798                 method: method,
48799                 params:isPost ? this.getParams() : null,
48800                 isUpload: this.form.fileUpload,
48801                 formData : this.form.formData
48802             }));
48803             
48804             this.uploadProgress();
48805
48806         }else if (o.clientValidation !== false){ // client validation failed
48807             this.failureType = Roo.form.Action.CLIENT_INVALID;
48808             this.form.afterAction(this, false);
48809         }
48810     },
48811
48812     success : function(response)
48813     {
48814         this.uploadComplete= true;
48815         if (this.haveProgress) {
48816             Roo.MessageBox.hide();
48817         }
48818         
48819         
48820         var result = this.processResponse(response);
48821         if(result === true || result.success){
48822             this.form.afterAction(this, true);
48823             return;
48824         }
48825         if(result.errors){
48826             this.form.markInvalid(result.errors);
48827             this.failureType = Roo.form.Action.SERVER_INVALID;
48828         }
48829         this.form.afterAction(this, false);
48830     },
48831     failure : function(response)
48832     {
48833         this.uploadComplete= true;
48834         if (this.haveProgress) {
48835             Roo.MessageBox.hide();
48836         }
48837         
48838         this.response = response;
48839         this.failureType = Roo.form.Action.CONNECT_FAILURE;
48840         this.form.afterAction(this, false);
48841     },
48842     
48843     handleResponse : function(response){
48844         if(this.form.errorReader){
48845             var rs = this.form.errorReader.read(response);
48846             var errors = [];
48847             if(rs.records){
48848                 for(var i = 0, len = rs.records.length; i < len; i++) {
48849                     var r = rs.records[i];
48850                     errors[i] = r.data;
48851                 }
48852             }
48853             if(errors.length < 1){
48854                 errors = null;
48855             }
48856             return {
48857                 success : rs.success,
48858                 errors : errors
48859             };
48860         }
48861         var ret = false;
48862         try {
48863             ret = Roo.decode(response.responseText);
48864         } catch (e) {
48865             ret = {
48866                 success: false,
48867                 errorMsg: "Failed to read server message: " + (response ? response.responseText : ' - no message'),
48868                 errors : []
48869             };
48870         }
48871         return ret;
48872         
48873     }
48874 });
48875
48876
48877 Roo.form.Action.Load = function(form, options){
48878     Roo.form.Action.Load.superclass.constructor.call(this, form, options);
48879     this.reader = this.form.reader;
48880 };
48881
48882 Roo.extend(Roo.form.Action.Load, Roo.form.Action, {
48883     type : 'load',
48884
48885     run : function(){
48886         
48887         Roo.Ajax.request(Roo.apply(
48888                 this.createCallback(), {
48889                     method:this.getMethod(),
48890                     url:this.getUrl(false),
48891                     params:this.getParams()
48892         }));
48893     },
48894
48895     success : function(response){
48896         
48897         var result = this.processResponse(response);
48898         if(result === true || !result.success || !result.data){
48899             this.failureType = Roo.form.Action.LOAD_FAILURE;
48900             this.form.afterAction(this, false);
48901             return;
48902         }
48903         this.form.clearInvalid();
48904         this.form.setValues(result.data);
48905         this.form.afterAction(this, true);
48906     },
48907
48908     handleResponse : function(response){
48909         if(this.form.reader){
48910             var rs = this.form.reader.read(response);
48911             var data = rs.records && rs.records[0] ? rs.records[0].data : null;
48912             return {
48913                 success : rs.success,
48914                 data : data
48915             };
48916         }
48917         return Roo.decode(response.responseText);
48918     }
48919 });
48920
48921 Roo.form.Action.ACTION_TYPES = {
48922     'load' : Roo.form.Action.Load,
48923     'submit' : Roo.form.Action.Submit
48924 };/*
48925  * Based on:
48926  * Ext JS Library 1.1.1
48927  * Copyright(c) 2006-2007, Ext JS, LLC.
48928  *
48929  * Originally Released Under LGPL - original licence link has changed is not relivant.
48930  *
48931  * Fork - LGPL
48932  * <script type="text/javascript">
48933  */
48934  
48935 /**
48936  * @class Roo.form.Layout
48937  * @extends Roo.Component
48938  * Creates a container for layout and rendering of fields in an {@link Roo.form.Form}.
48939  * @constructor
48940  * @param {Object} config Configuration options
48941  */
48942 Roo.form.Layout = function(config){
48943     var xitems = [];
48944     if (config.items) {
48945         xitems = config.items;
48946         delete config.items;
48947     }
48948     Roo.form.Layout.superclass.constructor.call(this, config);
48949     this.stack = [];
48950     Roo.each(xitems, this.addxtype, this);
48951      
48952 };
48953
48954 Roo.extend(Roo.form.Layout, Roo.Component, {
48955     /**
48956      * @cfg {String/Object} autoCreate
48957      * A DomHelper element spec used to autocreate the layout (defaults to {tag: 'div', cls: 'x-form-ct'})
48958      */
48959     /**
48960      * @cfg {String/Object/Function} style
48961      * A style specification string, e.g. "width:100px", or object in the form {width:"100px"}, or
48962      * a function which returns such a specification.
48963      */
48964     /**
48965      * @cfg {String} labelAlign
48966      * Valid values are "left," "top" and "right" (defaults to "left")
48967      */
48968     /**
48969      * @cfg {Number} labelWidth
48970      * Fixed width in pixels of all field labels (defaults to undefined)
48971      */
48972     /**
48973      * @cfg {Boolean} clear
48974      * True to add a clearing element at the end of this layout, equivalent to CSS clear: both (defaults to true)
48975      */
48976     clear : true,
48977     /**
48978      * @cfg {String} labelSeparator
48979      * The separator to use after field labels (defaults to ':')
48980      */
48981     labelSeparator : ':',
48982     /**
48983      * @cfg {Boolean} hideLabels
48984      * True to suppress the display of field labels in this layout (defaults to false)
48985      */
48986     hideLabels : false,
48987
48988     // private
48989     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct'},
48990     
48991     isLayout : true,
48992     
48993     // private
48994     onRender : function(ct, position){
48995         if(this.el){ // from markup
48996             this.el = Roo.get(this.el);
48997         }else {  // generate
48998             var cfg = this.getAutoCreate();
48999             this.el = ct.createChild(cfg, position);
49000         }
49001         if(this.style){
49002             this.el.applyStyles(this.style);
49003         }
49004         if(this.labelAlign){
49005             this.el.addClass('x-form-label-'+this.labelAlign);
49006         }
49007         if(this.hideLabels){
49008             this.labelStyle = "display:none";
49009             this.elementStyle = "padding-left:0;";
49010         }else{
49011             if(typeof this.labelWidth == 'number'){
49012                 this.labelStyle = "width:"+this.labelWidth+"px;";
49013                 this.elementStyle = "padding-left:"+((this.labelWidth+(typeof this.labelPad == 'number' ? this.labelPad : 5))+'px')+";";
49014             }
49015             if(this.labelAlign == 'top'){
49016                 this.labelStyle = "width:auto;";
49017                 this.elementStyle = "padding-left:0;";
49018             }
49019         }
49020         var stack = this.stack;
49021         var slen = stack.length;
49022         if(slen > 0){
49023             if(!this.fieldTpl){
49024                 var t = new Roo.Template(
49025                     '<div class="x-form-item {5}">',
49026                         '<label for="{0}" style="{2}">{1}{4}</label>',
49027                         '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49028                         '</div>',
49029                     '</div><div class="x-form-clear-left"></div>'
49030                 );
49031                 t.disableFormats = true;
49032                 t.compile();
49033                 Roo.form.Layout.prototype.fieldTpl = t;
49034             }
49035             for(var i = 0; i < slen; i++) {
49036                 if(stack[i].isFormField){
49037                     this.renderField(stack[i]);
49038                 }else{
49039                     this.renderComponent(stack[i]);
49040                 }
49041             }
49042         }
49043         if(this.clear){
49044             this.el.createChild({cls:'x-form-clear'});
49045         }
49046     },
49047
49048     // private
49049     renderField : function(f){
49050         f.fieldEl = Roo.get(this.fieldTpl.append(this.el, [
49051                f.id, //0
49052                f.fieldLabel, //1
49053                f.labelStyle||this.labelStyle||'', //2
49054                this.elementStyle||'', //3
49055                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator, //4
49056                f.itemCls||this.itemCls||''  //5
49057        ], true).getPrevSibling());
49058     },
49059
49060     // private
49061     renderComponent : function(c){
49062         c.render(c.isLayout ? this.el : this.el.createChild());    
49063     },
49064     /**
49065      * Adds a object form elements (using the xtype property as the factory method.)
49066      * Valid xtypes are:  TextField, TextArea .... Button, Layout, FieldSet, Column
49067      * @param {Object} config 
49068      */
49069     addxtype : function(o)
49070     {
49071         // create the lement.
49072         o.form = this.form;
49073         var fe = Roo.factory(o, Roo.form);
49074         this.form.allItems.push(fe);
49075         this.stack.push(fe);
49076         
49077         if (fe.isFormField) {
49078             this.form.items.add(fe);
49079         }
49080          
49081         return fe;
49082     }
49083 });
49084
49085 /**
49086  * @class Roo.form.Column
49087  * @extends Roo.form.Layout
49088  * Creates a column container for layout and rendering of fields in an {@link Roo.form.Form}.
49089  * @constructor
49090  * @param {Object} config Configuration options
49091  */
49092 Roo.form.Column = function(config){
49093     Roo.form.Column.superclass.constructor.call(this, config);
49094 };
49095
49096 Roo.extend(Roo.form.Column, Roo.form.Layout, {
49097     /**
49098      * @cfg {Number/String} width
49099      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49100      */
49101     /**
49102      * @cfg {String/Object} autoCreate
49103      * A DomHelper element spec used to autocreate the column (defaults to {tag: 'div', cls: 'x-form-ct x-form-column'})
49104      */
49105
49106     // private
49107     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-column'},
49108
49109     // private
49110     onRender : function(ct, position){
49111         Roo.form.Column.superclass.onRender.call(this, ct, position);
49112         if(this.width){
49113             this.el.setWidth(this.width);
49114         }
49115     }
49116 });
49117
49118
49119 /**
49120  * @class Roo.form.Row
49121  * @extends Roo.form.Layout
49122  * Creates a row container for layout and rendering of fields in an {@link Roo.form.Form}.
49123  * @constructor
49124  * @param {Object} config Configuration options
49125  */
49126
49127  
49128 Roo.form.Row = function(config){
49129     Roo.form.Row.superclass.constructor.call(this, config);
49130 };
49131  
49132 Roo.extend(Roo.form.Row, Roo.form.Layout, {
49133       /**
49134      * @cfg {Number/String} width
49135      * The fixed width of the column in pixels or CSS value (defaults to "auto")
49136      */
49137     /**
49138      * @cfg {Number/String} height
49139      * The fixed height of the column in pixels or CSS value (defaults to "auto")
49140      */
49141     defaultAutoCreate : {tag: 'div', cls: 'x-form-ct x-form-row'},
49142     
49143     padWidth : 20,
49144     // private
49145     onRender : function(ct, position){
49146         //console.log('row render');
49147         if(!this.rowTpl){
49148             var t = new Roo.Template(
49149                 '<div class="x-form-item {5}" style="float:left;width:{6}px">',
49150                     '<label for="{0}" style="{2}">{1}{4}</label>',
49151                     '<div class="x-form-element" id="x-form-el-{0}" style="{3}">',
49152                     '</div>',
49153                 '</div>'
49154             );
49155             t.disableFormats = true;
49156             t.compile();
49157             Roo.form.Layout.prototype.rowTpl = t;
49158         }
49159         this.fieldTpl = this.rowTpl;
49160         
49161         //console.log('lw' + this.labelWidth +', la:' + this.labelAlign);
49162         var labelWidth = 100;
49163         
49164         if ((this.labelAlign != 'top')) {
49165             if (typeof this.labelWidth == 'number') {
49166                 labelWidth = this.labelWidth
49167             }
49168             this.padWidth =  20 + labelWidth;
49169             
49170         }
49171         
49172         Roo.form.Column.superclass.onRender.call(this, ct, position);
49173         if(this.width){
49174             this.el.setWidth(this.width);
49175         }
49176         if(this.height){
49177             this.el.setHeight(this.height);
49178         }
49179     },
49180     
49181     // private
49182     renderField : function(f){
49183         f.fieldEl = this.fieldTpl.append(this.el, [
49184                f.id, f.fieldLabel,
49185                f.labelStyle||this.labelStyle||'',
49186                this.elementStyle||'',
49187                typeof f.labelSeparator == 'undefined' ? this.labelSeparator : f.labelSeparator,
49188                f.itemCls||this.itemCls||'',
49189                f.width ? f.width + this.padWidth : 160 + this.padWidth
49190        ],true);
49191     }
49192 });
49193  
49194
49195 /**
49196  * @class Roo.form.FieldSet
49197  * @extends Roo.form.Layout
49198  * Creates a fieldset container for layout and rendering of fields in an {@link Roo.form.Form}.
49199  * @constructor
49200  * @param {Object} config Configuration options
49201  */
49202 Roo.form.FieldSet = function(config){
49203     Roo.form.FieldSet.superclass.constructor.call(this, config);
49204 };
49205
49206 Roo.extend(Roo.form.FieldSet, Roo.form.Layout, {
49207     /**
49208      * @cfg {String} legend
49209      * The text to display as the legend for the FieldSet (defaults to '')
49210      */
49211     /**
49212      * @cfg {String/Object} autoCreate
49213      * A DomHelper element spec used to autocreate the fieldset (defaults to {tag: 'fieldset', cn: {tag:'legend'}})
49214      */
49215
49216     // private
49217     defaultAutoCreate : {tag: 'fieldset', cn: {tag:'legend'}},
49218
49219     // private
49220     onRender : function(ct, position){
49221         Roo.form.FieldSet.superclass.onRender.call(this, ct, position);
49222         if(this.legend){
49223             this.setLegend(this.legend);
49224         }
49225     },
49226
49227     // private
49228     setLegend : function(text){
49229         if(this.rendered){
49230             this.el.child('legend').update(text);
49231         }
49232     }
49233 });/*
49234  * Based on:
49235  * Ext JS Library 1.1.1
49236  * Copyright(c) 2006-2007, Ext JS, LLC.
49237  *
49238  * Originally Released Under LGPL - original licence link has changed is not relivant.
49239  *
49240  * Fork - LGPL
49241  * <script type="text/javascript">
49242  */
49243 /**
49244  * @class Roo.form.VTypes
49245  * Overridable validation definitions. The validations provided are basic and intended to be easily customizable and extended.
49246  * @singleton
49247  */
49248 Roo.form.VTypes = function(){
49249     // closure these in so they are only created once.
49250     var alpha = /^[a-zA-Z_]+$/;
49251     var alphanum = /^[a-zA-Z0-9_]+$/;
49252     var email = /^([\w]+)(.[\w]+)*@([\w-]+\.){1,5}([A-Za-z]){2,24}$/;
49253     var url = /(((https?)|(ftp)):\/\/([\-\w]+\.)+\w{2,3}(\/[%\-\w]+(\.\w{2,})?)*(([\w\-\.\?\\\/+@&#;`~=%!]*)(\.\w{2,})?)*\/?)/i;
49254
49255     // All these messages and functions are configurable
49256     return {
49257         /**
49258          * The function used to validate email addresses
49259          * @param {String} value The email address
49260          */
49261         'email' : function(v){
49262             return email.test(v);
49263         },
49264         /**
49265          * The error text to display when the email validation function returns false
49266          * @type String
49267          */
49268         'emailText' : 'This field should be an e-mail address in the format "user@domain.com"',
49269         /**
49270          * The keystroke filter mask to be applied on email input
49271          * @type RegExp
49272          */
49273         'emailMask' : /[a-z0-9_\.\-@]/i,
49274
49275         /**
49276          * The function used to validate URLs
49277          * @param {String} value The URL
49278          */
49279         'url' : function(v){
49280             return url.test(v);
49281         },
49282         /**
49283          * The error text to display when the url validation function returns false
49284          * @type String
49285          */
49286         'urlText' : 'This field should be a URL in the format "http:/'+'/www.domain.com"',
49287         
49288         /**
49289          * The function used to validate alpha values
49290          * @param {String} value The value
49291          */
49292         'alpha' : function(v){
49293             return alpha.test(v);
49294         },
49295         /**
49296          * The error text to display when the alpha validation function returns false
49297          * @type String
49298          */
49299         'alphaText' : 'This field should only contain letters and _',
49300         /**
49301          * The keystroke filter mask to be applied on alpha input
49302          * @type RegExp
49303          */
49304         'alphaMask' : /[a-z_]/i,
49305
49306         /**
49307          * The function used to validate alphanumeric values
49308          * @param {String} value The value
49309          */
49310         'alphanum' : function(v){
49311             return alphanum.test(v);
49312         },
49313         /**
49314          * The error text to display when the alphanumeric validation function returns false
49315          * @type String
49316          */
49317         'alphanumText' : 'This field should only contain letters, numbers and _',
49318         /**
49319          * The keystroke filter mask to be applied on alphanumeric input
49320          * @type RegExp
49321          */
49322         'alphanumMask' : /[a-z0-9_]/i
49323     };
49324 }();//<script type="text/javascript">
49325
49326 /**
49327  * @class Roo.form.FCKeditor
49328  * @extends Roo.form.TextArea
49329  * Wrapper around the FCKEditor http://www.fckeditor.net
49330  * @constructor
49331  * Creates a new FCKeditor
49332  * @param {Object} config Configuration options
49333  */
49334 Roo.form.FCKeditor = function(config){
49335     Roo.form.FCKeditor.superclass.constructor.call(this, config);
49336     this.addEvents({
49337          /**
49338          * @event editorinit
49339          * Fired when the editor is initialized - you can add extra handlers here..
49340          * @param {FCKeditor} this
49341          * @param {Object} the FCK object.
49342          */
49343         editorinit : true
49344     });
49345     
49346     
49347 };
49348 Roo.form.FCKeditor.editors = { };
49349 Roo.extend(Roo.form.FCKeditor, Roo.form.TextArea,
49350 {
49351     //defaultAutoCreate : {
49352     //    tag : "textarea",style   : "width:100px;height:60px;" ,autocomplete    : "off"
49353     //},
49354     // private
49355     /**
49356      * @cfg {Object} fck options - see fck manual for details.
49357      */
49358     fckconfig : false,
49359     
49360     /**
49361      * @cfg {Object} fck toolbar set (Basic or Default)
49362      */
49363     toolbarSet : 'Basic',
49364     /**
49365      * @cfg {Object} fck BasePath
49366      */ 
49367     basePath : '/fckeditor/',
49368     
49369     
49370     frame : false,
49371     
49372     value : '',
49373     
49374    
49375     onRender : function(ct, position)
49376     {
49377         if(!this.el){
49378             this.defaultAutoCreate = {
49379                 tag: "textarea",
49380                 style:"width:300px;height:60px;",
49381                 autocomplete: "new-password"
49382             };
49383         }
49384         Roo.form.FCKeditor.superclass.onRender.call(this, ct, position);
49385         /*
49386         if(this.grow){
49387             this.textSizeEl = Roo.DomHelper.append(document.body, {tag: "pre", cls: "x-form-grow-sizer"});
49388             if(this.preventScrollbars){
49389                 this.el.setStyle("overflow", "hidden");
49390             }
49391             this.el.setHeight(this.growMin);
49392         }
49393         */
49394         //console.log('onrender' + this.getId() );
49395         Roo.form.FCKeditor.editors[this.getId()] = this;
49396          
49397
49398         this.replaceTextarea() ;
49399         
49400     },
49401     
49402     getEditor : function() {
49403         return this.fckEditor;
49404     },
49405     /**
49406      * Sets a data value into the field and validates it.  To set the value directly without validation see {@link #setRawValue}.
49407      * @param {Mixed} value The value to set
49408      */
49409     
49410     
49411     setValue : function(value)
49412     {
49413         //console.log('setValue: ' + value);
49414         
49415         if(typeof(value) == 'undefined') { // not sure why this is happending...
49416             return;
49417         }
49418         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49419         
49420         //if(!this.el || !this.getEditor()) {
49421         //    this.value = value;
49422             //this.setValue.defer(100,this,[value]);    
49423         //    return;
49424         //} 
49425         
49426         if(!this.getEditor()) {
49427             return;
49428         }
49429         
49430         this.getEditor().SetData(value);
49431         
49432         //
49433
49434     },
49435
49436     /**
49437      * Returns the normalized data value (undefined or emptyText will be returned as '').  To return the raw value see {@link #getRawValue}.
49438      * @return {Mixed} value The field value
49439      */
49440     getValue : function()
49441     {
49442         
49443         if (this.frame && this.frame.dom.style.display == 'none') {
49444             return Roo.form.FCKeditor.superclass.getValue.call(this);
49445         }
49446         
49447         if(!this.el || !this.getEditor()) {
49448            
49449            // this.getValue.defer(100,this); 
49450             return this.value;
49451         }
49452        
49453         
49454         var value=this.getEditor().GetData();
49455         Roo.form.FCKeditor.superclass.setValue.apply(this,[value]);
49456         return Roo.form.FCKeditor.superclass.getValue.call(this);
49457         
49458
49459     },
49460
49461     /**
49462      * Returns the raw data value which may or may not be a valid, defined value.  To return a normalized value see {@link #getValue}.
49463      * @return {Mixed} value The field value
49464      */
49465     getRawValue : function()
49466     {
49467         if (this.frame && this.frame.dom.style.display == 'none') {
49468             return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49469         }
49470         
49471         if(!this.el || !this.getEditor()) {
49472             //this.getRawValue.defer(100,this); 
49473             return this.value;
49474             return;
49475         }
49476         
49477         
49478         
49479         var value=this.getEditor().GetData();
49480         Roo.form.FCKeditor.superclass.setRawValue.apply(this,[value]);
49481         return Roo.form.FCKeditor.superclass.getRawValue.call(this);
49482          
49483     },
49484     
49485     setSize : function(w,h) {
49486         
49487         
49488         
49489         //if (this.frame && this.frame.dom.style.display == 'none') {
49490         //    Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49491         //    return;
49492         //}
49493         //if(!this.el || !this.getEditor()) {
49494         //    this.setSize.defer(100,this, [w,h]); 
49495         //    return;
49496         //}
49497         
49498         
49499         
49500         Roo.form.FCKeditor.superclass.setSize.apply(this, [w, h]);
49501         
49502         this.frame.dom.setAttribute('width', w);
49503         this.frame.dom.setAttribute('height', h);
49504         this.frame.setSize(w,h);
49505         
49506     },
49507     
49508     toggleSourceEdit : function(value) {
49509         
49510       
49511          
49512         this.el.dom.style.display = value ? '' : 'none';
49513         this.frame.dom.style.display = value ?  'none' : '';
49514         
49515     },
49516     
49517     
49518     focus: function(tag)
49519     {
49520         if (this.frame.dom.style.display == 'none') {
49521             return Roo.form.FCKeditor.superclass.focus.call(this);
49522         }
49523         if(!this.el || !this.getEditor()) {
49524             this.focus.defer(100,this, [tag]); 
49525             return;
49526         }
49527         
49528         
49529         
49530         
49531         var tgs = this.getEditor().EditorDocument.getElementsByTagName(tag);
49532         this.getEditor().Focus();
49533         if (tgs.length) {
49534             if (!this.getEditor().Selection.GetSelection()) {
49535                 this.focus.defer(100,this, [tag]); 
49536                 return;
49537             }
49538             
49539             
49540             var r = this.getEditor().EditorDocument.createRange();
49541             r.setStart(tgs[0],0);
49542             r.setEnd(tgs[0],0);
49543             this.getEditor().Selection.GetSelection().removeAllRanges();
49544             this.getEditor().Selection.GetSelection().addRange(r);
49545             this.getEditor().Focus();
49546         }
49547         
49548     },
49549     
49550     
49551     
49552     replaceTextarea : function()
49553     {
49554         if ( document.getElementById( this.getId() + '___Frame' ) ) {
49555             return ;
49556         }
49557         //if ( !this.checkBrowser || this._isCompatibleBrowser() )
49558         //{
49559             // We must check the elements firstly using the Id and then the name.
49560         var oTextarea = document.getElementById( this.getId() );
49561         
49562         var colElementsByName = document.getElementsByName( this.getId() ) ;
49563          
49564         oTextarea.style.display = 'none' ;
49565
49566         if ( oTextarea.tabIndex ) {            
49567             this.TabIndex = oTextarea.tabIndex ;
49568         }
49569         
49570         this._insertHtmlBefore( this._getConfigHtml(), oTextarea ) ;
49571         this._insertHtmlBefore( this._getIFrameHtml(), oTextarea ) ;
49572         this.frame = Roo.get(this.getId() + '___Frame')
49573     },
49574     
49575     _getConfigHtml : function()
49576     {
49577         var sConfig = '' ;
49578
49579         for ( var o in this.fckconfig ) {
49580             sConfig += sConfig.length > 0  ? '&amp;' : '';
49581             sConfig += encodeURIComponent( o ) + '=' + encodeURIComponent( this.fckconfig[o] ) ;
49582         }
49583
49584         return '<input type="hidden" id="' + this.getId() + '___Config" value="' + sConfig + '" style="display:none" />' ;
49585     },
49586     
49587     
49588     _getIFrameHtml : function()
49589     {
49590         var sFile = 'fckeditor.html' ;
49591         /* no idea what this is about..
49592         try
49593         {
49594             if ( (/fcksource=true/i).test( window.top.location.search ) )
49595                 sFile = 'fckeditor.original.html' ;
49596         }
49597         catch (e) { 
49598         */
49599
49600         var sLink = this.basePath + 'editor/' + sFile + '?InstanceName=' + encodeURIComponent( this.getId() ) ;
49601         sLink += this.toolbarSet ? ( '&amp;Toolbar=' + this.toolbarSet)  : '';
49602         
49603         
49604         var html = '<iframe id="' + this.getId() +
49605             '___Frame" src="' + sLink +
49606             '" width="' + this.width +
49607             '" height="' + this.height + '"' +
49608             (this.tabIndex ?  ' tabindex="' + this.tabIndex + '"' :'' ) +
49609             ' frameborder="0" scrolling="no"></iframe>' ;
49610
49611         return html ;
49612     },
49613     
49614     _insertHtmlBefore : function( html, element )
49615     {
49616         if ( element.insertAdjacentHTML )       {
49617             // IE
49618             element.insertAdjacentHTML( 'beforeBegin', html ) ;
49619         } else { // Gecko
49620             var oRange = document.createRange() ;
49621             oRange.setStartBefore( element ) ;
49622             var oFragment = oRange.createContextualFragment( html );
49623             element.parentNode.insertBefore( oFragment, element ) ;
49624         }
49625     }
49626     
49627     
49628   
49629     
49630     
49631     
49632     
49633
49634 });
49635
49636 //Roo.reg('fckeditor', Roo.form.FCKeditor);
49637
49638 function FCKeditor_OnComplete(editorInstance){
49639     var f = Roo.form.FCKeditor.editors[editorInstance.Name];
49640     f.fckEditor = editorInstance;
49641     //console.log("loaded");
49642     f.fireEvent('editorinit', f, editorInstance);
49643
49644   
49645
49646  
49647
49648
49649
49650
49651
49652
49653
49654
49655
49656
49657
49658
49659
49660
49661
49662 //<script type="text/javascript">
49663 /**
49664  * @class Roo.form.GridField
49665  * @extends Roo.form.Field
49666  * Embed a grid (or editable grid into a form)
49667  * STATUS ALPHA
49668  * 
49669  * This embeds a grid in a form, the value of the field should be the json encoded array of rows
49670  * it needs 
49671  * xgrid.store = Roo.data.Store
49672  * xgrid.store.proxy = Roo.data.MemoryProxy (data = [] )
49673  * xgrid.store.reader = Roo.data.JsonReader 
49674  * 
49675  * 
49676  * @constructor
49677  * Creates a new GridField
49678  * @param {Object} config Configuration options
49679  */
49680 Roo.form.GridField = function(config){
49681     Roo.form.GridField.superclass.constructor.call(this, config);
49682      
49683 };
49684
49685 Roo.extend(Roo.form.GridField, Roo.form.Field,  {
49686     /**
49687      * @cfg {Number} width  - used to restrict width of grid..
49688      */
49689     width : 100,
49690     /**
49691      * @cfg {Number} height - used to restrict height of grid..
49692      */
49693     height : 50,
49694      /**
49695      * @cfg {Object} xgrid (xtype'd description of grid) { xtype : 'Grid', dataSource: .... }
49696          * 
49697          *}
49698      */
49699     xgrid : false, 
49700     /**
49701      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49702      * {tag: "input", type: "checkbox", autocomplete: "off"})
49703      */
49704    // defaultAutoCreate : { tag: 'div' },
49705     defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'new-password'},
49706     /**
49707      * @cfg {String} addTitle Text to include for adding a title.
49708      */
49709     addTitle : false,
49710     //
49711     onResize : function(){
49712         Roo.form.Field.superclass.onResize.apply(this, arguments);
49713     },
49714
49715     initEvents : function(){
49716         // Roo.form.Checkbox.superclass.initEvents.call(this);
49717         // has no events...
49718        
49719     },
49720
49721
49722     getResizeEl : function(){
49723         return this.wrap;
49724     },
49725
49726     getPositionEl : function(){
49727         return this.wrap;
49728     },
49729
49730     // private
49731     onRender : function(ct, position){
49732         
49733         this.style = this.style || 'overflow: hidden; border:1px solid #c3daf9;';
49734         var style = this.style;
49735         delete this.style;
49736         
49737         Roo.form.GridField.superclass.onRender.call(this, ct, position);
49738         this.wrap = this.el.wrap({cls: ''}); // not sure why ive done thsi...
49739         this.viewEl = this.wrap.createChild({ tag: 'div' });
49740         if (style) {
49741             this.viewEl.applyStyles(style);
49742         }
49743         if (this.width) {
49744             this.viewEl.setWidth(this.width);
49745         }
49746         if (this.height) {
49747             this.viewEl.setHeight(this.height);
49748         }
49749         //if(this.inputValue !== undefined){
49750         //this.setValue(this.value);
49751         
49752         
49753         this.grid = new Roo.grid[this.xgrid.xtype](this.viewEl, this.xgrid);
49754         
49755         
49756         this.grid.render();
49757         this.grid.getDataSource().on('remove', this.refreshValue, this);
49758         this.grid.getDataSource().on('update', this.refreshValue, this);
49759         this.grid.on('afteredit', this.refreshValue, this);
49760  
49761     },
49762      
49763     
49764     /**
49765      * Sets the value of the item. 
49766      * @param {String} either an object  or a string..
49767      */
49768     setValue : function(v){
49769         //this.value = v;
49770         v = v || []; // empty set..
49771         // this does not seem smart - it really only affects memoryproxy grids..
49772         if (this.grid && this.grid.getDataSource() && typeof(v) != 'undefined') {
49773             var ds = this.grid.getDataSource();
49774             // assumes a json reader..
49775             var data = {}
49776             data[ds.reader.meta.root ] =  typeof(v) == 'string' ? Roo.decode(v) : v;
49777             ds.loadData( data);
49778         }
49779         // clear selection so it does not get stale.
49780         if (this.grid.sm) { 
49781             this.grid.sm.clearSelections();
49782         }
49783         
49784         Roo.form.GridField.superclass.setValue.call(this, v);
49785         this.refreshValue();
49786         // should load data in the grid really....
49787     },
49788     
49789     // private
49790     refreshValue: function() {
49791          var val = [];
49792         this.grid.getDataSource().each(function(r) {
49793             val.push(r.data);
49794         });
49795         this.el.dom.value = Roo.encode(val);
49796     }
49797     
49798      
49799     
49800     
49801 });/*
49802  * Based on:
49803  * Ext JS Library 1.1.1
49804  * Copyright(c) 2006-2007, Ext JS, LLC.
49805  *
49806  * Originally Released Under LGPL - original licence link has changed is not relivant.
49807  *
49808  * Fork - LGPL
49809  * <script type="text/javascript">
49810  */
49811 /**
49812  * @class Roo.form.DisplayField
49813  * @extends Roo.form.Field
49814  * A generic Field to display non-editable data.
49815  * @cfg {Boolean} closable (true|false) default false
49816  * @constructor
49817  * Creates a new Display Field item.
49818  * @param {Object} config Configuration options
49819  */
49820 Roo.form.DisplayField = function(config){
49821     Roo.form.DisplayField.superclass.constructor.call(this, config);
49822     
49823     this.addEvents({
49824         /**
49825          * @event close
49826          * Fires after the click the close btn
49827              * @param {Roo.form.DisplayField} this
49828              */
49829         close : true
49830     });
49831 };
49832
49833 Roo.extend(Roo.form.DisplayField, Roo.form.TextField,  {
49834     inputType:      'hidden',
49835     allowBlank:     true,
49836     readOnly:         true,
49837     
49838  
49839     /**
49840      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49841      */
49842     focusClass : undefined,
49843     /**
49844      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49845      */
49846     fieldClass: 'x-form-field',
49847     
49848      /**
49849      * @cfg {Function} valueRenderer The renderer for the field (so you can reformat output). should return raw HTML
49850      */
49851     valueRenderer: undefined,
49852     
49853     width: 100,
49854     /**
49855      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49856      * {tag: "input", type: "checkbox", autocomplete: "off"})
49857      */
49858      
49859  //   defaultAutoCreate : { tag: 'input', type: 'hidden', autocomplete: 'off'},
49860  
49861     closable : false,
49862     
49863     onResize : function(){
49864         Roo.form.DisplayField.superclass.onResize.apply(this, arguments);
49865         
49866     },
49867
49868     initEvents : function(){
49869         // Roo.form.Checkbox.superclass.initEvents.call(this);
49870         // has no events...
49871         
49872         if(this.closable){
49873             this.closeEl.on('click', this.onClose, this);
49874         }
49875        
49876     },
49877
49878
49879     getResizeEl : function(){
49880         return this.wrap;
49881     },
49882
49883     getPositionEl : function(){
49884         return this.wrap;
49885     },
49886
49887     // private
49888     onRender : function(ct, position){
49889         
49890         Roo.form.DisplayField.superclass.onRender.call(this, ct, position);
49891         //if(this.inputValue !== undefined){
49892         this.wrap = this.el.wrap();
49893         
49894         this.viewEl = this.wrap.createChild({ tag: 'div', cls: 'x-form-displayfield'});
49895         
49896         if(this.closable){
49897             this.closeEl = this.wrap.createChild({ tag: 'div', cls: 'x-dlg-close'});
49898         }
49899         
49900         if (this.bodyStyle) {
49901             this.viewEl.applyStyles(this.bodyStyle);
49902         }
49903         //this.viewEl.setStyle('padding', '2px');
49904         
49905         this.setValue(this.value);
49906         
49907     },
49908 /*
49909     // private
49910     initValue : Roo.emptyFn,
49911
49912   */
49913
49914         // private
49915     onClick : function(){
49916         
49917     },
49918
49919     /**
49920      * Sets the checked state of the checkbox.
49921      * @param {Boolean/String} checked True, 'true', '1', or 'on' to check the checkbox, any other value will uncheck it.
49922      */
49923     setValue : function(v){
49924         this.value = v;
49925         var html = this.valueRenderer ?  this.valueRenderer(v) : String.format('{0}', v);
49926         // this might be called before we have a dom element..
49927         if (!this.viewEl) {
49928             return;
49929         }
49930         this.viewEl.dom.innerHTML = html;
49931         Roo.form.DisplayField.superclass.setValue.call(this, v);
49932
49933     },
49934     
49935     onClose : function(e)
49936     {
49937         e.preventDefault();
49938         
49939         this.fireEvent('close', this);
49940     }
49941 });/*
49942  * 
49943  * Licence- LGPL
49944  * 
49945  */
49946
49947 /**
49948  * @class Roo.form.DayPicker
49949  * @extends Roo.form.Field
49950  * A Day picker show [M] [T] [W] ....
49951  * @constructor
49952  * Creates a new Day Picker
49953  * @param {Object} config Configuration options
49954  */
49955 Roo.form.DayPicker= function(config){
49956     Roo.form.DayPicker.superclass.constructor.call(this, config);
49957      
49958 };
49959
49960 Roo.extend(Roo.form.DayPicker, Roo.form.Field,  {
49961     /**
49962      * @cfg {String} focusClass The CSS class to use when the checkbox receives focus (defaults to undefined)
49963      */
49964     focusClass : undefined,
49965     /**
49966      * @cfg {String} fieldClass The default CSS class for the checkbox (defaults to "x-form-field")
49967      */
49968     fieldClass: "x-form-field",
49969    
49970     /**
49971      * @cfg {String/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to
49972      * {tag: "input", type: "checkbox", autocomplete: "off"})
49973      */
49974     defaultAutoCreate : { tag: "input", type: 'hidden', autocomplete: "new-password"},
49975     
49976    
49977     actionMode : 'viewEl', 
49978     //
49979     // private
49980  
49981     inputType : 'hidden',
49982     
49983      
49984     inputElement: false, // real input element?
49985     basedOn: false, // ????
49986     
49987     isFormField: true, // not sure where this is needed!!!!
49988
49989     onResize : function(){
49990         Roo.form.Checkbox.superclass.onResize.apply(this, arguments);
49991         if(!this.boxLabel){
49992             this.el.alignTo(this.wrap, 'c-c');
49993         }
49994     },
49995
49996     initEvents : function(){
49997         Roo.form.Checkbox.superclass.initEvents.call(this);
49998         this.el.on("click", this.onClick,  this);
49999         this.el.on("change", this.onClick,  this);
50000     },
50001
50002
50003     getResizeEl : function(){
50004         return this.wrap;
50005     },
50006
50007     getPositionEl : function(){
50008         return this.wrap;
50009     },
50010
50011     
50012     // private
50013     onRender : function(ct, position){
50014         Roo.form.Checkbox.superclass.onRender.call(this, ct, position);
50015        
50016         this.wrap = this.el.wrap({cls: 'x-form-daypick-item '});
50017         
50018         var r1 = '<table><tr>';
50019         var r2 = '<tr class="x-form-daypick-icons">';
50020         for (var i=0; i < 7; i++) {
50021             r1+= '<td><div>' + Date.dayNames[i].substring(0,3) + '</div></td>';
50022             r2+= '<td><img class="x-menu-item-icon" src="' + Roo.BLANK_IMAGE_URL  +'"></td>';
50023         }
50024         
50025         var viewEl = this.wrap.createChild( r1 + '</tr>' + r2 + '</tr></table>');
50026         viewEl.select('img').on('click', this.onClick, this);
50027         this.viewEl = viewEl;   
50028         
50029         
50030         // this will not work on Chrome!!!
50031         this.el.on('DOMAttrModified', this.setFromHidden,  this); //ff
50032         this.el.on('propertychange', this.setFromHidden,  this);  //ie
50033         
50034         
50035           
50036
50037     },
50038
50039     // private
50040     initValue : Roo.emptyFn,
50041
50042     /**
50043      * Returns the checked state of the checkbox.
50044      * @return {Boolean} True if checked, else false
50045      */
50046     getValue : function(){
50047         return this.el.dom.value;
50048         
50049     },
50050
50051         // private
50052     onClick : function(e){ 
50053         //this.setChecked(!this.checked);
50054         Roo.get(e.target).toggleClass('x-menu-item-checked');
50055         this.refreshValue();
50056         //if(this.el.dom.checked != this.checked){
50057         //    this.setValue(this.el.dom.checked);
50058        // }
50059     },
50060     
50061     // private
50062     refreshValue : function()
50063     {
50064         var val = '';
50065         this.viewEl.select('img',true).each(function(e,i,n)  {
50066             val += e.is(".x-menu-item-checked") ? String(n) : '';
50067         });
50068         this.setValue(val, true);
50069     },
50070
50071     /**
50072      * Sets the checked state of the checkbox.
50073      * On is always based on a string comparison between inputValue and the param.
50074      * @param {Boolean/String} value - the value to set 
50075      * @param {Boolean/String} suppressEvent - whether to suppress the checkchange event.
50076      */
50077     setValue : function(v,suppressEvent){
50078         if (!this.el.dom) {
50079             return;
50080         }
50081         var old = this.el.dom.value ;
50082         this.el.dom.value = v;
50083         if (suppressEvent) {
50084             return ;
50085         }
50086          
50087         // update display..
50088         this.viewEl.select('img',true).each(function(e,i,n)  {
50089             
50090             var on = e.is(".x-menu-item-checked");
50091             var newv = v.indexOf(String(n)) > -1;
50092             if (on != newv) {
50093                 e.toggleClass('x-menu-item-checked');
50094             }
50095             
50096         });
50097         
50098         
50099         this.fireEvent('change', this, v, old);
50100         
50101         
50102     },
50103    
50104     // handle setting of hidden value by some other method!!?!?
50105     setFromHidden: function()
50106     {
50107         if(!this.el){
50108             return;
50109         }
50110         //console.log("SET FROM HIDDEN");
50111         //alert('setFrom hidden');
50112         this.setValue(this.el.dom.value);
50113     },
50114     
50115     onDestroy : function()
50116     {
50117         if(this.viewEl){
50118             Roo.get(this.viewEl).remove();
50119         }
50120          
50121         Roo.form.DayPicker.superclass.onDestroy.call(this);
50122     }
50123
50124 });/*
50125  * RooJS Library 1.1.1
50126  * Copyright(c) 2008-2011  Alan Knowles
50127  *
50128  * License - LGPL
50129  */
50130  
50131
50132 /**
50133  * @class Roo.form.ComboCheck
50134  * @extends Roo.form.ComboBox
50135  * A combobox for multiple select items.
50136  *
50137  * FIXME - could do with a reset button..
50138  * 
50139  * @constructor
50140  * Create a new ComboCheck
50141  * @param {Object} config Configuration options
50142  */
50143 Roo.form.ComboCheck = function(config){
50144     Roo.form.ComboCheck.superclass.constructor.call(this, config);
50145     // should verify some data...
50146     // like
50147     // hiddenName = required..
50148     // displayField = required
50149     // valudField == required
50150     var req= [ 'hiddenName', 'displayField', 'valueField' ];
50151     var _t = this;
50152     Roo.each(req, function(e) {
50153         if ((typeof(_t[e]) == 'undefined' ) || !_t[e].length) {
50154             throw "Roo.form.ComboCheck : missing value for: " + e;
50155         }
50156     });
50157     
50158     
50159 };
50160
50161 Roo.extend(Roo.form.ComboCheck, Roo.form.ComboBox, {
50162      
50163      
50164     editable : false,
50165      
50166     selectedClass: 'x-menu-item-checked', 
50167     
50168     // private
50169     onRender : function(ct, position){
50170         var _t = this;
50171         
50172         
50173         
50174         if(!this.tpl){
50175             var cls = 'x-combo-list';
50176
50177             
50178             this.tpl =  new Roo.Template({
50179                 html :  '<div class="'+cls+'-item x-menu-check-item">' +
50180                    '<img class="x-menu-item-icon" style="margin: 0px;" src="' + Roo.BLANK_IMAGE_URL + '">' + 
50181                    '<span>{' + this.displayField + '}</span>' +
50182                     '</div>' 
50183                 
50184             });
50185         }
50186  
50187         
50188         Roo.form.ComboCheck.superclass.onRender.call(this, ct, position);
50189         this.view.singleSelect = false;
50190         this.view.multiSelect = true;
50191         this.view.toggleSelect = true;
50192         this.pageTb.add(new Roo.Toolbar.Fill(), {
50193             
50194             text: 'Done',
50195             handler: function()
50196             {
50197                 _t.collapse();
50198             }
50199         });
50200     },
50201     
50202     onViewOver : function(e, t){
50203         // do nothing...
50204         return;
50205         
50206     },
50207     
50208     onViewClick : function(doFocus,index){
50209         return;
50210         
50211     },
50212     select: function () {
50213         //Roo.log("SELECT CALLED");
50214     },
50215      
50216     selectByValue : function(xv, scrollIntoView){
50217         var ar = this.getValueArray();
50218         var sels = [];
50219         
50220         Roo.each(ar, function(v) {
50221             if(v === undefined || v === null){
50222                 return;
50223             }
50224             var r = this.findRecord(this.valueField, v);
50225             if(r){
50226                 sels.push(this.store.indexOf(r))
50227                 
50228             }
50229         },this);
50230         this.view.select(sels);
50231         return false;
50232     },
50233     
50234     
50235     
50236     onSelect : function(record, index){
50237        // Roo.log("onselect Called");
50238        // this is only called by the clear button now..
50239         this.view.clearSelections();
50240         this.setValue('[]');
50241         if (this.value != this.valueBefore) {
50242             this.fireEvent('change', this, this.value, this.valueBefore);
50243             this.valueBefore = this.value;
50244         }
50245     },
50246     getValueArray : function()
50247     {
50248         var ar = [] ;
50249         
50250         try {
50251             //Roo.log(this.value);
50252             if (typeof(this.value) == 'undefined') {
50253                 return [];
50254             }
50255             var ar = Roo.decode(this.value);
50256             return  ar instanceof Array ? ar : []; //?? valid?
50257             
50258         } catch(e) {
50259             Roo.log(e + "\nRoo.form.ComboCheck:getValueArray  invalid data:" + this.getValue());
50260             return [];
50261         }
50262          
50263     },
50264     expand : function ()
50265     {
50266         
50267         Roo.form.ComboCheck.superclass.expand.call(this);
50268         this.valueBefore = typeof(this.value) == 'undefined' ? '' : this.value;
50269         //this.valueBefore = typeof(this.valueBefore) == 'undefined' ? '' : this.valueBefore;
50270         
50271
50272     },
50273     
50274     collapse : function(){
50275         Roo.form.ComboCheck.superclass.collapse.call(this);
50276         var sl = this.view.getSelectedIndexes();
50277         var st = this.store;
50278         var nv = [];
50279         var tv = [];
50280         var r;
50281         Roo.each(sl, function(i) {
50282             r = st.getAt(i);
50283             nv.push(r.get(this.valueField));
50284         },this);
50285         this.setValue(Roo.encode(nv));
50286         if (this.value != this.valueBefore) {
50287
50288             this.fireEvent('change', this, this.value, this.valueBefore);
50289             this.valueBefore = this.value;
50290         }
50291         
50292     },
50293     
50294     setValue : function(v){
50295         // Roo.log(v);
50296         this.value = v;
50297         
50298         var vals = this.getValueArray();
50299         var tv = [];
50300         Roo.each(vals, function(k) {
50301             var r = this.findRecord(this.valueField, k);
50302             if(r){
50303                 tv.push(r.data[this.displayField]);
50304             }else if(this.valueNotFoundText !== undefined){
50305                 tv.push( this.valueNotFoundText );
50306             }
50307         },this);
50308        // Roo.log(tv);
50309         
50310         Roo.form.ComboBox.superclass.setValue.call(this, tv.join(', '));
50311         this.hiddenField.value = v;
50312         this.value = v;
50313     }
50314     
50315 });/*
50316  * Based on:
50317  * Ext JS Library 1.1.1
50318  * Copyright(c) 2006-2007, Ext JS, LLC.
50319  *
50320  * Originally Released Under LGPL - original licence link has changed is not relivant.
50321  *
50322  * Fork - LGPL
50323  * <script type="text/javascript">
50324  */
50325  
50326 /**
50327  * @class Roo.form.Signature
50328  * @extends Roo.form.Field
50329  * Signature field.  
50330  * @constructor
50331  * 
50332  * @param {Object} config Configuration options
50333  */
50334
50335 Roo.form.Signature = function(config){
50336     Roo.form.Signature.superclass.constructor.call(this, config);
50337     
50338     this.addEvents({// not in used??
50339          /**
50340          * @event confirm
50341          * Fires when the 'confirm' icon is pressed (add a listener to enable add button)
50342              * @param {Roo.form.Signature} combo This combo box
50343              */
50344         'confirm' : true,
50345         /**
50346          * @event reset
50347          * Fires when the 'edit' icon is pressed (add a listener to enable add button)
50348              * @param {Roo.form.ComboBox} combo This combo box
50349              * @param {Roo.data.Record|false} record The data record returned from the underlying store (or false on nothing selected)
50350              */
50351         'reset' : true
50352     });
50353 };
50354
50355 Roo.extend(Roo.form.Signature, Roo.form.Field,  {
50356     /**
50357      * @cfg {Object} labels Label to use when rendering a form.
50358      * defaults to 
50359      * labels : { 
50360      *      clear : "Clear",
50361      *      confirm : "Confirm"
50362      *  }
50363      */
50364     labels : { 
50365         clear : "Clear",
50366         confirm : "Confirm"
50367     },
50368     /**
50369      * @cfg {Number} width The signature panel width (defaults to 300)
50370      */
50371     width: 300,
50372     /**
50373      * @cfg {Number} height The signature panel height (defaults to 100)
50374      */
50375     height : 100,
50376     /**
50377      * @cfg {Boolean} allowBlank False to validate that the value length > 0 (defaults to false)
50378      */
50379     allowBlank : false,
50380     
50381     //private
50382     // {Object} signPanel The signature SVG panel element (defaults to {})
50383     signPanel : {},
50384     // {Boolean} isMouseDown False to validate that the mouse down event (defaults to false)
50385     isMouseDown : false,
50386     // {Boolean} isConfirmed validate the signature is confirmed or not for submitting form (defaults to false)
50387     isConfirmed : false,
50388     // {String} signatureTmp SVG mapping string (defaults to empty string)
50389     signatureTmp : '',
50390     
50391     
50392     defaultAutoCreate : { // modified by initCompnoent..
50393         tag: "input",
50394         type:"hidden"
50395     },
50396
50397     // private
50398     onRender : function(ct, position){
50399         
50400         Roo.form.Signature.superclass.onRender.call(this, ct, position);
50401         
50402         this.wrap = this.el.wrap({
50403             cls:'x-form-signature-wrap', style : 'width: ' + this.width + 'px', cn:{cls:'x-form-signature'}
50404         });
50405         
50406         this.createToolbar(this);
50407         this.signPanel = this.wrap.createChild({
50408                 tag: 'div',
50409                 style: 'width: ' + this.width + 'px; height: ' + this.height + 'px; border: 0;'
50410             }, this.el
50411         );
50412             
50413         this.svgID = Roo.id();
50414         this.svgEl = this.signPanel.createChild({
50415               xmlns : 'http://www.w3.org/2000/svg',
50416               tag : 'svg',
50417               id : this.svgID + "-svg",
50418               width: this.width,
50419               height: this.height,
50420               viewBox: '0 0 '+this.width+' '+this.height,
50421               cn : [
50422                 {
50423                     tag: "rect",
50424                     id: this.svgID + "-svg-r",
50425                     width: this.width,
50426                     height: this.height,
50427                     fill: "#ffa"
50428                 },
50429                 {
50430                     tag: "line",
50431                     id: this.svgID + "-svg-l",
50432                     x1: "0", // start
50433                     y1: (this.height*0.8), // start set the line in 80% of height
50434                     x2: this.width, // end
50435                     y2: (this.height*0.8), // end set the line in 80% of height
50436                     'stroke': "#666",
50437                     'stroke-width': "1",
50438                     'stroke-dasharray': "3",
50439                     'shape-rendering': "crispEdges",
50440                     'pointer-events': "none"
50441                 },
50442                 {
50443                     tag: "path",
50444                     id: this.svgID + "-svg-p",
50445                     'stroke': "navy",
50446                     'stroke-width': "3",
50447                     'fill': "none",
50448                     'pointer-events': 'none'
50449                 }
50450               ]
50451         });
50452         this.createSVG();
50453         this.svgBox = this.svgEl.dom.getScreenCTM();
50454     },
50455     createSVG : function(){ 
50456         var svg = this.signPanel;
50457         var r = svg.select('#'+ this.svgID + '-svg-r', true).first().dom;
50458         var t = this;
50459
50460         r.addEventListener('mousedown', function(e) { return t.down(e); }, false);
50461         r.addEventListener('mousemove', function(e) { return t.move(e); }, false);
50462         r.addEventListener('mouseup', function(e) { return t.up(e); }, false);
50463         r.addEventListener('mouseout', function(e) { return t.up(e); }, false);
50464         r.addEventListener('touchstart', function(e) { return t.down(e); }, false);
50465         r.addEventListener('touchmove', function(e) { return t.move(e); }, false);
50466         r.addEventListener('touchend', function(e) { return t.up(e); }, false);
50467         
50468     },
50469     isTouchEvent : function(e){
50470         return e.type.match(/^touch/);
50471     },
50472     getCoords : function (e) {
50473         var pt    = this.svgEl.dom.createSVGPoint();
50474         pt.x = e.clientX; 
50475         pt.y = e.clientY;
50476         if (this.isTouchEvent(e)) {
50477             pt.x =  e.targetTouches[0].clientX;
50478             pt.y = e.targetTouches[0].clientY;
50479         }
50480         var a = this.svgEl.dom.getScreenCTM();
50481         var b = a.inverse();
50482         var mx = pt.matrixTransform(b);
50483         return mx.x + ',' + mx.y;
50484     },
50485     //mouse event headler 
50486     down : function (e) {
50487         this.signatureTmp += 'M' + this.getCoords(e) + ' ';
50488         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr('d', this.signatureTmp);
50489         
50490         this.isMouseDown = true;
50491         
50492         e.preventDefault();
50493     },
50494     move : function (e) {
50495         if (this.isMouseDown) {
50496             this.signatureTmp += 'L' + this.getCoords(e) + ' ';
50497             this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', this.signatureTmp);
50498         }
50499         
50500         e.preventDefault();
50501     },
50502     up : function (e) {
50503         this.isMouseDown = false;
50504         var sp = this.signatureTmp.split(' ');
50505         
50506         if(sp.length > 1){
50507             if(!sp[sp.length-2].match(/^L/)){
50508                 sp.pop();
50509                 sp.pop();
50510                 sp.push("");
50511                 this.signatureTmp = sp.join(" ");
50512             }
50513         }
50514         if(this.getValue() != this.signatureTmp){
50515             this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50516             this.isConfirmed = false;
50517         }
50518         e.preventDefault();
50519     },
50520     
50521     /**
50522      * Protected method that will not generally be called directly. It
50523      * is called when the editor creates its toolbar. Override this method if you need to
50524      * add custom toolbar buttons.
50525      * @param {HtmlEditor} editor
50526      */
50527     createToolbar : function(editor){
50528          function btn(id, toggle, handler){
50529             var xid = fid + '-'+ id ;
50530             return {
50531                 id : xid,
50532                 cmd : id,
50533                 cls : 'x-btn-icon x-edit-'+id,
50534                 enableToggle:toggle !== false,
50535                 scope: editor, // was editor...
50536                 handler:handler||editor.relayBtnCmd,
50537                 clickEvent:'mousedown',
50538                 tooltip: etb.buttonTips[id] || undefined, ///tips ???
50539                 tabIndex:-1
50540             };
50541         }
50542         
50543         
50544         var tb = new Roo.Toolbar(editor.wrap.dom.firstChild);
50545         this.tb = tb;
50546         this.tb.add(
50547            {
50548                 cls : ' x-signature-btn x-signature-'+id,
50549                 scope: editor, // was editor...
50550                 handler: this.reset,
50551                 clickEvent:'mousedown',
50552                 text: this.labels.clear
50553             },
50554             {
50555                  xtype : 'Fill',
50556                  xns: Roo.Toolbar
50557             }, 
50558             {
50559                 cls : '  x-signature-btn x-signature-'+id,
50560                 scope: editor, // was editor...
50561                 handler: this.confirmHandler,
50562                 clickEvent:'mousedown',
50563                 text: this.labels.confirm
50564             }
50565         );
50566     
50567     },
50568     //public
50569     /**
50570      * when user is clicked confirm then show this image.....
50571      * 
50572      * @return {String} Image Data URI
50573      */
50574     getImageDataURI : function(){
50575         var svg = this.svgEl.dom.parentNode.innerHTML;
50576         var src = 'data:image/svg+xml;base64,'+window.btoa(svg);
50577         return src; 
50578     },
50579     /**
50580      * 
50581      * @return {Boolean} this.isConfirmed
50582      */
50583     getConfirmed : function(){
50584         return this.isConfirmed;
50585     },
50586     /**
50587      * 
50588      * @return {Number} this.width
50589      */
50590     getWidth : function(){
50591         return this.width;
50592     },
50593     /**
50594      * 
50595      * @return {Number} this.height
50596      */
50597     getHeight : function(){
50598         return this.height;
50599     },
50600     // private
50601     getSignature : function(){
50602         return this.signatureTmp;
50603     },
50604     // private
50605     reset : function(){
50606         this.signatureTmp = '';
50607         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50608         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', '');
50609         this.isConfirmed = false;
50610         Roo.form.Signature.superclass.reset.call(this);
50611     },
50612     setSignature : function(s){
50613         this.signatureTmp = s;
50614         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#ffa');
50615         this.signPanel.select('#'+ this.svgID + '-svg-p', true).first().attr( 'd', s);
50616         this.setValue(s);
50617         this.isConfirmed = false;
50618         Roo.form.Signature.superclass.reset.call(this);
50619     }, 
50620     test : function(){
50621 //        Roo.log(this.signPanel.dom.contentWindow.up())
50622     },
50623     //private
50624     setConfirmed : function(){
50625         
50626         
50627         
50628 //        Roo.log(Roo.get(this.signPanel.dom.contentWindow.r).attr('fill', '#cfc'));
50629     },
50630     // private
50631     confirmHandler : function(){
50632         if(!this.getSignature()){
50633             return;
50634         }
50635         
50636         this.signPanel.select('#'+ this.svgID + '-svg-r', true).first().attr('fill', '#cfc');
50637         this.setValue(this.getSignature());
50638         this.isConfirmed = true;
50639         
50640         this.fireEvent('confirm', this);
50641     },
50642     // private
50643     // Subclasses should provide the validation implementation by overriding this
50644     validateValue : function(value){
50645         if(this.allowBlank){
50646             return true;
50647         }
50648         
50649         if(this.isConfirmed){
50650             return true;
50651         }
50652         return false;
50653     }
50654 });/*
50655  * Based on:
50656  * Ext JS Library 1.1.1
50657  * Copyright(c) 2006-2007, Ext JS, LLC.
50658  *
50659  * Originally Released Under LGPL - original licence link has changed is not relivant.
50660  *
50661  * Fork - LGPL
50662  * <script type="text/javascript">
50663  */
50664  
50665
50666 /**
50667  * @class Roo.form.ComboBox
50668  * @extends Roo.form.TriggerField
50669  * A combobox control with support for autocomplete, remote-loading, paging and many other features.
50670  * @constructor
50671  * Create a new ComboBox.
50672  * @param {Object} config Configuration options
50673  */
50674 Roo.form.Select = function(config){
50675     Roo.form.Select.superclass.constructor.call(this, config);
50676      
50677 };
50678
50679 Roo.extend(Roo.form.Select , Roo.form.ComboBox, {
50680     /**
50681      * @cfg {String/HTMLElement/Element} transform The id, DOM node or element of an existing select to convert to a ComboBox
50682      */
50683     /**
50684      * @cfg {Boolean} lazyRender True to prevent the ComboBox from rendering until requested (should always be used when
50685      * rendering into an Roo.Editor, defaults to false)
50686      */
50687     /**
50688      * @cfg {Boolean/Object} autoCreate A DomHelper element spec, or true for a default element spec (defaults to:
50689      * {tag: "input", type: "text", size: "24", autocomplete: "off"})
50690      */
50691     /**
50692      * @cfg {Roo.data.Store} store The data store to which this combo is bound (defaults to undefined)
50693      */
50694     /**
50695      * @cfg {String} title If supplied, a header element is created containing this text and added into the top of
50696      * the dropdown list (defaults to undefined, with no header element)
50697      */
50698
50699      /**
50700      * @cfg {String/Roo.Template} tpl The template to use to render the output
50701      */
50702      
50703     // private
50704     defaultAutoCreate : {tag: "select"  },
50705     /**
50706      * @cfg {Number} listWidth The width in pixels of the dropdown list (defaults to the width of the ComboBox field)
50707      */
50708     listWidth: undefined,
50709     /**
50710      * @cfg {String} displayField The underlying data field name to bind to this CombBox (defaults to undefined if
50711      * mode = 'remote' or 'text' if mode = 'local')
50712      */
50713     displayField: undefined,
50714     /**
50715      * @cfg {String} valueField The underlying data value name to bind to this CombBox (defaults to undefined if
50716      * mode = 'remote' or 'value' if mode = 'local'). 
50717      * Note: use of a valueField requires the user make a selection
50718      * in order for a value to be mapped.
50719      */
50720     valueField: undefined,
50721     
50722     
50723     /**
50724      * @cfg {String} hiddenName If specified, a hidden form field with this name is dynamically generated to store the
50725      * field's data value (defaults to the underlying DOM element's name)
50726      */
50727     hiddenName: undefined,
50728     /**
50729      * @cfg {String} listClass CSS class to apply to the dropdown list element (defaults to '')
50730      */
50731     listClass: '',
50732     /**
50733      * @cfg {String} selectedClass CSS class to apply to the selected item in the dropdown list (defaults to 'x-combo-selected')
50734      */
50735     selectedClass: 'x-combo-selected',
50736     /**
50737      * @cfg {String} triggerClass An additional CSS class used to style the trigger button.  The trigger will always get the
50738      * class 'x-form-trigger' and triggerClass will be <b>appended</b> if specified (defaults to 'x-form-arrow-trigger'
50739      * which displays a downward arrow icon).
50740      */
50741     triggerClass : 'x-form-arrow-trigger',
50742     /**
50743      * @cfg {Boolean/String} shadow True or "sides" for the default effect, "frame" for 4-way shadow, and "drop" for bottom-right
50744      */
50745     shadow:'sides',
50746     /**
50747      * @cfg {String} listAlign A valid anchor position value. See {@link Roo.Element#alignTo} for details on supported
50748      * anchor positions (defaults to 'tl-bl')
50749      */
50750     listAlign: 'tl-bl?',
50751     /**
50752      * @cfg {Number} maxHeight The maximum height in pixels of the dropdown list before scrollbars are shown (defaults to 300)
50753      */
50754     maxHeight: 300,
50755     /**
50756      * @cfg {String} triggerAction The action to execute when the trigger field is activated.  Use 'all' to run the
50757      * query specified by the allQuery config option (defaults to 'query')
50758      */
50759     triggerAction: 'query',
50760     /**
50761      * @cfg {Number} minChars The minimum number of characters the user must type before autocomplete and typeahead activate
50762      * (defaults to 4, does not apply if editable = false)
50763      */
50764     minChars : 4,
50765     /**
50766      * @cfg {Boolean} typeAhead True to populate and autoselect the remainder of the text being typed after a configurable
50767      * delay (typeAheadDelay) if it matches a known value (defaults to false)
50768      */
50769     typeAhead: false,
50770     /**
50771      * @cfg {Number} queryDelay The length of time in milliseconds to delay between the start of typing and sending the
50772      * query to filter the dropdown list (defaults to 500 if mode = 'remote' or 10 if mode = 'local')
50773      */
50774     queryDelay: 500,
50775     /**
50776      * @cfg {Number} pageSize If greater than 0, a paging toolbar is displayed in the footer of the dropdown list and the
50777      * filter queries will execute with page start and limit parameters.  Only applies when mode = 'remote' (defaults to 0)
50778      */
50779     pageSize: 0,
50780     /**
50781      * @cfg {Boolean} selectOnFocus True to select any existing text in the field immediately on focus.  Only applies
50782      * when editable = true (defaults to false)
50783      */
50784     selectOnFocus:false,
50785     /**
50786      * @cfg {String} queryParam Name of the query as it will be passed on the querystring (defaults to 'query')
50787      */
50788     queryParam: 'query',
50789     /**
50790      * @cfg {String} loadingText The text to display in the dropdown list while data is loading.  Only applies
50791      * when mode = 'remote' (defaults to 'Loading...')
50792      */
50793     loadingText: 'Loading...',
50794     /**
50795      * @cfg {Boolean} resizable True to add a resize handle to the bottom of the dropdown list (defaults to false)
50796      */
50797     resizable: false,
50798     /**
50799      * @cfg {Number} handleHeight The height in pixels of the dropdown list resize handle if resizable = true (defaults to 8)
50800      */
50801     handleHeight : 8,
50802     /**
50803      * @cfg {Boolean} editable False to prevent the user from typing text directly into the field, just like a
50804      * traditional select (defaults to true)
50805      */
50806     editable: true,
50807     /**
50808      * @cfg {String} allQuery The text query to send to the server to return all records for the list with no filtering (defaults to '')
50809      */
50810     allQuery: '',
50811     /**
50812      * @cfg {String} mode Set to 'local' if the ComboBox loads local data (defaults to 'remote' which loads from the server)
50813      */
50814     mode: 'remote',
50815     /**
50816      * @cfg {Number} minListWidth The minimum width of the dropdown list in pixels (defaults to 70, will be ignored if
50817      * listWidth has a higher value)
50818      */
50819     minListWidth : 70,
50820     /**
50821      * @cfg {Boolean} forceSelection True to restrict the selected value to one of the values in the list, false to
50822      * allow the user to set arbitrary text into the field (defaults to false)
50823      */
50824     forceSelection:false,
50825     /**
50826      * @cfg {Number} typeAheadDelay The length of time in milliseconds to wait until the typeahead text is displayed
50827      * if typeAhead = true (defaults to 250)
50828      */
50829     typeAheadDelay : 250,
50830     /**
50831      * @cfg {String} valueNotFoundText When using a name/value combo, if the value passed to setValue is not found in
50832      * the store, valueNotFoundText will be displayed as the field text if defined (defaults to undefined)
50833      */
50834     valueNotFoundText : undefined,
50835     
50836     /**
50837      * @cfg {String} defaultValue The value displayed after loading the store.
50838      */
50839     defaultValue: '',
50840     
50841     /**
50842      * @cfg {Boolean} blockFocus Prevents all focus calls, so it can work with things like HTML edtor bar
50843      */
50844     blockFocus : false,
50845     
50846     /**
50847      * @cfg {Boolean} disableClear Disable showing of clear button.
50848      */
50849     disableClear : false,
50850     /**
50851      * @cfg {Boolean} alwaysQuery  Disable caching of results, and always send query
50852      */
50853     alwaysQuery : false,
50854     
50855     //private
50856     addicon : false,
50857     editicon: false,
50858     
50859     // element that contains real text value.. (when hidden is used..)
50860      
50861     // private
50862     onRender : function(ct, position){
50863         Roo.form.Field.prototype.onRender.call(this, ct, position);
50864         
50865         if(this.store){
50866             this.store.on('beforeload', this.onBeforeLoad, this);
50867             this.store.on('load', this.onLoad, this);
50868             this.store.on('loadexception', this.onLoadException, this);
50869             this.store.load({});
50870         }
50871         
50872         
50873         
50874     },
50875
50876     // private
50877     initEvents : function(){
50878         //Roo.form.ComboBox.superclass.initEvents.call(this);
50879  
50880     },
50881
50882     onDestroy : function(){
50883        
50884         if(this.store){
50885             this.store.un('beforeload', this.onBeforeLoad, this);
50886             this.store.un('load', this.onLoad, this);
50887             this.store.un('loadexception', this.onLoadException, this);
50888         }
50889         //Roo.form.ComboBox.superclass.onDestroy.call(this);
50890     },
50891
50892     // private
50893     fireKey : function(e){
50894         if(e.isNavKeyPress() && !this.list.isVisible()){
50895             this.fireEvent("specialkey", this, e);
50896         }
50897     },
50898
50899     // private
50900     onResize: function(w, h){
50901         
50902         return; 
50903     
50904         
50905     },
50906
50907     /**
50908      * Allow or prevent the user from directly editing the field text.  If false is passed,
50909      * the user will only be able to select from the items defined in the dropdown list.  This method
50910      * is the runtime equivalent of setting the 'editable' config option at config time.
50911      * @param {Boolean} value True to allow the user to directly edit the field text
50912      */
50913     setEditable : function(value){
50914          
50915     },
50916
50917     // private
50918     onBeforeLoad : function(){
50919         
50920         Roo.log("Select before load");
50921         return;
50922     
50923         this.innerList.update(this.loadingText ?
50924                '<div class="loading-indicator">'+this.loadingText+'</div>' : '');
50925         //this.restrictHeight();
50926         this.selectedIndex = -1;
50927     },
50928
50929     // private
50930     onLoad : function(){
50931
50932     
50933         var dom = this.el.dom;
50934         dom.innerHTML = '';
50935          var od = dom.ownerDocument;
50936          
50937         if (this.emptyText) {
50938             var op = od.createElement('option');
50939             op.setAttribute('value', '');
50940             op.innerHTML = String.format('{0}', this.emptyText);
50941             dom.appendChild(op);
50942         }
50943         if(this.store.getCount() > 0){
50944            
50945             var vf = this.valueField;
50946             var df = this.displayField;
50947             this.store.data.each(function(r) {
50948                 // which colmsn to use... testing - cdoe / title..
50949                 var op = od.createElement('option');
50950                 op.setAttribute('value', r.data[vf]);
50951                 op.innerHTML = String.format('{0}', r.data[df]);
50952                 dom.appendChild(op);
50953             });
50954             if (typeof(this.defaultValue != 'undefined')) {
50955                 this.setValue(this.defaultValue);
50956             }
50957             
50958              
50959         }else{
50960             //this.onEmptyResults();
50961         }
50962         //this.el.focus();
50963     },
50964     // private
50965     onLoadException : function()
50966     {
50967         dom.innerHTML = '';
50968             
50969         Roo.log("Select on load exception");
50970         return;
50971     
50972         this.collapse();
50973         Roo.log(this.store.reader.jsonData);
50974         if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
50975             Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
50976         }
50977         
50978         
50979     },
50980     // private
50981     onTypeAhead : function(){
50982          
50983     },
50984
50985     // private
50986     onSelect : function(record, index){
50987         Roo.log('on select?');
50988         return;
50989         if(this.fireEvent('beforeselect', this, record, index) !== false){
50990             this.setFromData(index > -1 ? record.data : false);
50991             this.collapse();
50992             this.fireEvent('select', this, record, index);
50993         }
50994     },
50995
50996     /**
50997      * Returns the currently selected field value or empty string if no value is set.
50998      * @return {String} value The selected value
50999      */
51000     getValue : function(){
51001         var dom = this.el.dom;
51002         this.value = dom.options[dom.selectedIndex].value;
51003         return this.value;
51004         
51005     },
51006
51007     /**
51008      * Clears any text/value currently set in the field
51009      */
51010     clearValue : function(){
51011         this.value = '';
51012         this.el.dom.selectedIndex = this.emptyText ? 0 : -1;
51013         
51014     },
51015
51016     /**
51017      * Sets the specified value into the field.  If the value finds a match, the corresponding record text
51018      * will be displayed in the field.  If the value does not match the data value of an existing item,
51019      * and the valueNotFoundText config option is defined, it will be displayed as the default field text.
51020      * Otherwise the field will be blank (although the value will still be set).
51021      * @param {String} value The value to match
51022      */
51023     setValue : function(v){
51024         var d = this.el.dom;
51025         for (var i =0; i < d.options.length;i++) {
51026             if (v == d.options[i].value) {
51027                 d.selectedIndex = i;
51028                 this.value = v;
51029                 return;
51030             }
51031         }
51032         this.clearValue();
51033     },
51034     /**
51035      * @property {Object} the last set data for the element
51036      */
51037     
51038     lastData : false,
51039     /**
51040      * Sets the value of the field based on a object which is related to the record format for the store.
51041      * @param {Object} value the value to set as. or false on reset?
51042      */
51043     setFromData : function(o){
51044         Roo.log('setfrom data?');
51045          
51046         
51047         
51048     },
51049     // private
51050     reset : function(){
51051         this.clearValue();
51052     },
51053     // private
51054     findRecord : function(prop, value){
51055         
51056         return false;
51057     
51058         var record;
51059         if(this.store.getCount() > 0){
51060             this.store.each(function(r){
51061                 if(r.data[prop] == value){
51062                     record = r;
51063                     return false;
51064                 }
51065                 return true;
51066             });
51067         }
51068         return record;
51069     },
51070     
51071     getName: function()
51072     {
51073         // returns hidden if it's set..
51074         if (!this.rendered) {return ''};
51075         return !this.hiddenName && this.el.dom.name  ? this.el.dom.name : (this.hiddenName || '');
51076         
51077     },
51078      
51079
51080     
51081
51082     // private
51083     onEmptyResults : function(){
51084         Roo.log('empty results');
51085         //this.collapse();
51086     },
51087
51088     /**
51089      * Returns true if the dropdown list is expanded, else false.
51090      */
51091     isExpanded : function(){
51092         return false;
51093     },
51094
51095     /**
51096      * Select an item in the dropdown list by its data value. This function does NOT cause the select event to fire.
51097      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51098      * @param {String} value The data value of the item to select
51099      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51100      * selected item if it is not currently in view (defaults to true)
51101      * @return {Boolean} True if the value matched an item in the list, else false
51102      */
51103     selectByValue : function(v, scrollIntoView){
51104         Roo.log('select By Value');
51105         return false;
51106     
51107         if(v !== undefined && v !== null){
51108             var r = this.findRecord(this.valueField || this.displayField, v);
51109             if(r){
51110                 this.select(this.store.indexOf(r), scrollIntoView);
51111                 return true;
51112             }
51113         }
51114         return false;
51115     },
51116
51117     /**
51118      * Select an item in the dropdown list by its numeric index in the list. This function does NOT cause the select event to fire.
51119      * The store must be loaded and the list expanded for this function to work, otherwise use setValue.
51120      * @param {Number} index The zero-based index of the list item to select
51121      * @param {Boolean} scrollIntoView False to prevent the dropdown list from autoscrolling to display the
51122      * selected item if it is not currently in view (defaults to true)
51123      */
51124     select : function(index, scrollIntoView){
51125         Roo.log('select ');
51126         return  ;
51127         
51128         this.selectedIndex = index;
51129         this.view.select(index);
51130         if(scrollIntoView !== false){
51131             var el = this.view.getNode(index);
51132             if(el){
51133                 this.innerList.scrollChildIntoView(el, false);
51134             }
51135         }
51136     },
51137
51138       
51139
51140     // private
51141     validateBlur : function(){
51142         
51143         return;
51144         
51145     },
51146
51147     // private
51148     initQuery : function(){
51149         this.doQuery(this.getRawValue());
51150     },
51151
51152     // private
51153     doForce : function(){
51154         if(this.el.dom.value.length > 0){
51155             this.el.dom.value =
51156                 this.lastSelectionText === undefined ? '' : this.lastSelectionText;
51157              
51158         }
51159     },
51160
51161     /**
51162      * Execute a query to filter the dropdown list.  Fires the beforequery event prior to performing the
51163      * query allowing the query action to be canceled if needed.
51164      * @param {String} query The SQL query to execute
51165      * @param {Boolean} forceAll True to force the query to execute even if there are currently fewer characters
51166      * in the field than the minimum specified by the minChars config option.  It also clears any filter previously
51167      * saved in the current store (defaults to false)
51168      */
51169     doQuery : function(q, forceAll){
51170         
51171         Roo.log('doQuery?');
51172         if(q === undefined || q === null){
51173             q = '';
51174         }
51175         var qe = {
51176             query: q,
51177             forceAll: forceAll,
51178             combo: this,
51179             cancel:false
51180         };
51181         if(this.fireEvent('beforequery', qe)===false || qe.cancel){
51182             return false;
51183         }
51184         q = qe.query;
51185         forceAll = qe.forceAll;
51186         if(forceAll === true || (q.length >= this.minChars)){
51187             if(this.lastQuery != q || this.alwaysQuery){
51188                 this.lastQuery = q;
51189                 if(this.mode == 'local'){
51190                     this.selectedIndex = -1;
51191                     if(forceAll){
51192                         this.store.clearFilter();
51193                     }else{
51194                         this.store.filter(this.displayField, q);
51195                     }
51196                     this.onLoad();
51197                 }else{
51198                     this.store.baseParams[this.queryParam] = q;
51199                     this.store.load({
51200                         params: this.getParams(q)
51201                     });
51202                     this.expand();
51203                 }
51204             }else{
51205                 this.selectedIndex = -1;
51206                 this.onLoad();   
51207             }
51208         }
51209     },
51210
51211     // private
51212     getParams : function(q){
51213         var p = {};
51214         //p[this.queryParam] = q;
51215         if(this.pageSize){
51216             p.start = 0;
51217             p.limit = this.pageSize;
51218         }
51219         return p;
51220     },
51221
51222     /**
51223      * Hides the dropdown list if it is currently expanded. Fires the 'collapse' event on completion.
51224      */
51225     collapse : function(){
51226         
51227     },
51228
51229     // private
51230     collapseIf : function(e){
51231         
51232     },
51233
51234     /**
51235      * Expands the dropdown list if it is currently hidden. Fires the 'expand' event on completion.
51236      */
51237     expand : function(){
51238         
51239     } ,
51240
51241     // private
51242      
51243
51244     /** 
51245     * @cfg {Boolean} grow 
51246     * @hide 
51247     */
51248     /** 
51249     * @cfg {Number} growMin 
51250     * @hide 
51251     */
51252     /** 
51253     * @cfg {Number} growMax 
51254     * @hide 
51255     */
51256     /**
51257      * @hide
51258      * @method autoSize
51259      */
51260     
51261     setWidth : function()
51262     {
51263         
51264     },
51265     getResizeEl : function(){
51266         return this.el;
51267     }
51268 });//<script type="text/javasscript">
51269  
51270
51271 /**
51272  * @class Roo.DDView
51273  * A DnD enabled version of Roo.View.
51274  * @param {Element/String} container The Element in which to create the View.
51275  * @param {String} tpl The template string used to create the markup for each element of the View
51276  * @param {Object} config The configuration properties. These include all the config options of
51277  * {@link Roo.View} plus some specific to this class.<br>
51278  * <p>
51279  * Drag/drop is implemented by adding {@link Roo.data.Record}s to the target DDView. If copying is
51280  * not being performed, the original {@link Roo.data.Record} is removed from the source DDView.<br>
51281  * <p>
51282  * The following extra CSS rules are needed to provide insertion point highlighting:<pre><code>
51283 .x-view-drag-insert-above {
51284         border-top:1px dotted #3366cc;
51285 }
51286 .x-view-drag-insert-below {
51287         border-bottom:1px dotted #3366cc;
51288 }
51289 </code></pre>
51290  * 
51291  */
51292  
51293 Roo.DDView = function(container, tpl, config) {
51294     Roo.DDView.superclass.constructor.apply(this, arguments);
51295     this.getEl().setStyle("outline", "0px none");
51296     this.getEl().unselectable();
51297     if (this.dragGroup) {
51298                 this.setDraggable(this.dragGroup.split(","));
51299     }
51300     if (this.dropGroup) {
51301                 this.setDroppable(this.dropGroup.split(","));
51302     }
51303     if (this.deletable) {
51304         this.setDeletable();
51305     }
51306     this.isDirtyFlag = false;
51307         this.addEvents({
51308                 "drop" : true
51309         });
51310 };
51311
51312 Roo.extend(Roo.DDView, Roo.View, {
51313 /**     @cfg {String/Array} dragGroup The ddgroup name(s) for the View's DragZone. */
51314 /**     @cfg {String/Array} dropGroup The ddgroup name(s) for the View's DropZone. */
51315 /**     @cfg {Boolean} copy Causes drag operations to copy nodes rather than move. */
51316 /**     @cfg {Boolean} allowCopy Causes ctrl/drag operations to copy nodes rather than move. */
51317
51318         isFormField: true,
51319
51320         reset: Roo.emptyFn,
51321         
51322         clearInvalid: Roo.form.Field.prototype.clearInvalid,
51323
51324         validate: function() {
51325                 return true;
51326         },
51327         
51328         destroy: function() {
51329                 this.purgeListeners();
51330                 this.getEl.removeAllListeners();
51331                 this.getEl().remove();
51332                 if (this.dragZone) {
51333                         if (this.dragZone.destroy) {
51334                                 this.dragZone.destroy();
51335                         }
51336                 }
51337                 if (this.dropZone) {
51338                         if (this.dropZone.destroy) {
51339                                 this.dropZone.destroy();
51340                         }
51341                 }
51342         },
51343
51344 /**     Allows this class to be an Roo.form.Field so it can be found using {@link Roo.form.BasicForm#findField}. */
51345         getName: function() {
51346                 return this.name;
51347         },
51348
51349 /**     Loads the View from a JSON string representing the Records to put into the Store. */
51350         setValue: function(v) {
51351                 if (!this.store) {
51352                         throw "DDView.setValue(). DDView must be constructed with a valid Store";
51353                 }
51354                 var data = {};
51355                 data[this.store.reader.meta.root] = v ? [].concat(v) : [];
51356                 this.store.proxy = new Roo.data.MemoryProxy(data);
51357                 this.store.load();
51358         },
51359
51360 /**     @return {String} a parenthesised list of the ids of the Records in the View. */
51361         getValue: function() {
51362                 var result = '(';
51363                 this.store.each(function(rec) {
51364                         result += rec.id + ',';
51365                 });
51366                 return result.substr(0, result.length - 1) + ')';
51367         },
51368         
51369         getIds: function() {
51370                 var i = 0, result = new Array(this.store.getCount());
51371                 this.store.each(function(rec) {
51372                         result[i++] = rec.id;
51373                 });
51374                 return result;
51375         },
51376         
51377         isDirty: function() {
51378                 return this.isDirtyFlag;
51379         },
51380
51381 /**
51382  *      Part of the Roo.dd.DropZone interface. If no target node is found, the
51383  *      whole Element becomes the target, and this causes the drop gesture to append.
51384  */
51385     getTargetFromEvent : function(e) {
51386                 var target = e.getTarget();
51387                 while ((target !== null) && (target.parentNode != this.el.dom)) {
51388                 target = target.parentNode;
51389                 }
51390                 if (!target) {
51391                         target = this.el.dom.lastChild || this.el.dom;
51392                 }
51393                 return target;
51394     },
51395
51396 /**
51397  *      Create the drag data which consists of an object which has the property "ddel" as
51398  *      the drag proxy element. 
51399  */
51400     getDragData : function(e) {
51401         var target = this.findItemFromChild(e.getTarget());
51402                 if(target) {
51403                         this.handleSelection(e);
51404                         var selNodes = this.getSelectedNodes();
51405             var dragData = {
51406                 source: this,
51407                 copy: this.copy || (this.allowCopy && e.ctrlKey),
51408                 nodes: selNodes,
51409                 records: []
51410                         };
51411                         var selectedIndices = this.getSelectedIndexes();
51412                         for (var i = 0; i < selectedIndices.length; i++) {
51413                                 dragData.records.push(this.store.getAt(selectedIndices[i]));
51414                         }
51415                         if (selNodes.length == 1) {
51416                                 dragData.ddel = target.cloneNode(true); // the div element
51417                         } else {
51418                                 var div = document.createElement('div'); // create the multi element drag "ghost"
51419                                 div.className = 'multi-proxy';
51420                                 for (var i = 0, len = selNodes.length; i < len; i++) {
51421                                         div.appendChild(selNodes[i].cloneNode(true));
51422                                 }
51423                                 dragData.ddel = div;
51424                         }
51425             //console.log(dragData)
51426             //console.log(dragData.ddel.innerHTML)
51427                         return dragData;
51428                 }
51429         //console.log('nodragData')
51430                 return false;
51431     },
51432     
51433 /**     Specify to which ddGroup items in this DDView may be dragged. */
51434     setDraggable: function(ddGroup) {
51435         if (ddGroup instanceof Array) {
51436                 Roo.each(ddGroup, this.setDraggable, this);
51437                 return;
51438         }
51439         if (this.dragZone) {
51440                 this.dragZone.addToGroup(ddGroup);
51441         } else {
51442                         this.dragZone = new Roo.dd.DragZone(this.getEl(), {
51443                                 containerScroll: true,
51444                                 ddGroup: ddGroup 
51445
51446                         });
51447 //                      Draggability implies selection. DragZone's mousedown selects the element.
51448                         if (!this.multiSelect) { this.singleSelect = true; }
51449
51450 //                      Wire the DragZone's handlers up to methods in *this*
51451                         this.dragZone.getDragData = this.getDragData.createDelegate(this);
51452                 }
51453     },
51454
51455 /**     Specify from which ddGroup this DDView accepts drops. */
51456     setDroppable: function(ddGroup) {
51457         if (ddGroup instanceof Array) {
51458                 Roo.each(ddGroup, this.setDroppable, this);
51459                 return;
51460         }
51461         if (this.dropZone) {
51462                 this.dropZone.addToGroup(ddGroup);
51463         } else {
51464                         this.dropZone = new Roo.dd.DropZone(this.getEl(), {
51465                                 containerScroll: true,
51466                                 ddGroup: ddGroup
51467                         });
51468
51469 //                      Wire the DropZone's handlers up to methods in *this*
51470                         this.dropZone.getTargetFromEvent = this.getTargetFromEvent.createDelegate(this);
51471                         this.dropZone.onNodeEnter = this.onNodeEnter.createDelegate(this);
51472                         this.dropZone.onNodeOver = this.onNodeOver.createDelegate(this);
51473                         this.dropZone.onNodeOut = this.onNodeOut.createDelegate(this);
51474                         this.dropZone.onNodeDrop = this.onNodeDrop.createDelegate(this);
51475                 }
51476     },
51477
51478 /**     Decide whether to drop above or below a View node. */
51479     getDropPoint : function(e, n, dd){
51480         if (n == this.el.dom) { return "above"; }
51481                 var t = Roo.lib.Dom.getY(n), b = t + n.offsetHeight;
51482                 var c = t + (b - t) / 2;
51483                 var y = Roo.lib.Event.getPageY(e);
51484                 if(y <= c) {
51485                         return "above";
51486                 }else{
51487                         return "below";
51488                 }
51489     },
51490
51491     onNodeEnter : function(n, dd, e, data){
51492                 return false;
51493     },
51494     
51495     onNodeOver : function(n, dd, e, data){
51496                 var pt = this.getDropPoint(e, n, dd);
51497                 // set the insert point style on the target node
51498                 var dragElClass = this.dropNotAllowed;
51499                 if (pt) {
51500                         var targetElClass;
51501                         if (pt == "above"){
51502                                 dragElClass = n.previousSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-above";
51503                                 targetElClass = "x-view-drag-insert-above";
51504                         } else {
51505                                 dragElClass = n.nextSibling ? "x-tree-drop-ok-between" : "x-tree-drop-ok-below";
51506                                 targetElClass = "x-view-drag-insert-below";
51507                         }
51508                         if (this.lastInsertClass != targetElClass){
51509                                 Roo.fly(n).replaceClass(this.lastInsertClass, targetElClass);
51510                                 this.lastInsertClass = targetElClass;
51511                         }
51512                 }
51513                 return dragElClass;
51514         },
51515
51516     onNodeOut : function(n, dd, e, data){
51517                 this.removeDropIndicators(n);
51518     },
51519
51520     onNodeDrop : function(n, dd, e, data){
51521         if (this.fireEvent("drop", this, n, dd, e, data) === false) {
51522                 return false;
51523         }
51524         var pt = this.getDropPoint(e, n, dd);
51525                 var insertAt = (n == this.el.dom) ? this.nodes.length : n.nodeIndex;
51526                 if (pt == "below") { insertAt++; }
51527                 for (var i = 0; i < data.records.length; i++) {
51528                         var r = data.records[i];
51529                         var dup = this.store.getById(r.id);
51530                         if (dup && (dd != this.dragZone)) {
51531                                 Roo.fly(this.getNode(this.store.indexOf(dup))).frame("red", 1);
51532                         } else {
51533                                 if (data.copy) {
51534                                         this.store.insert(insertAt++, r.copy());
51535                                 } else {
51536                                         data.source.isDirtyFlag = true;
51537                                         r.store.remove(r);
51538                                         this.store.insert(insertAt++, r);
51539                                 }
51540                                 this.isDirtyFlag = true;
51541                         }
51542                 }
51543                 this.dragZone.cachedTarget = null;
51544                 return true;
51545     },
51546
51547     removeDropIndicators : function(n){
51548                 if(n){
51549                         Roo.fly(n).removeClass([
51550                                 "x-view-drag-insert-above",
51551                                 "x-view-drag-insert-below"]);
51552                         this.lastInsertClass = "_noclass";
51553                 }
51554     },
51555
51556 /**
51557  *      Utility method. Add a delete option to the DDView's context menu.
51558  *      @param {String} imageUrl The URL of the "delete" icon image.
51559  */
51560         setDeletable: function(imageUrl) {
51561                 if (!this.singleSelect && !this.multiSelect) {
51562                         this.singleSelect = true;
51563                 }
51564                 var c = this.getContextMenu();
51565                 this.contextMenu.on("itemclick", function(item) {
51566                         switch (item.id) {
51567                                 case "delete":
51568                                         this.remove(this.getSelectedIndexes());
51569                                         break;
51570                         }
51571                 }, this);
51572                 this.contextMenu.add({
51573                         icon: imageUrl,
51574                         id: "delete",
51575                         text: 'Delete'
51576                 });
51577         },
51578         
51579 /**     Return the context menu for this DDView. */
51580         getContextMenu: function() {
51581                 if (!this.contextMenu) {
51582 //                      Create the View's context menu
51583                         this.contextMenu = new Roo.menu.Menu({
51584                                 id: this.id + "-contextmenu"
51585                         });
51586                         this.el.on("contextmenu", this.showContextMenu, this);
51587                 }
51588                 return this.contextMenu;
51589         },
51590         
51591         disableContextMenu: function() {
51592                 if (this.contextMenu) {
51593                         this.el.un("contextmenu", this.showContextMenu, this);
51594                 }
51595         },
51596
51597         showContextMenu: function(e, item) {
51598         item = this.findItemFromChild(e.getTarget());
51599                 if (item) {
51600                         e.stopEvent();
51601                         this.select(this.getNode(item), this.multiSelect && e.ctrlKey, true);
51602                         this.contextMenu.showAt(e.getXY());
51603             }
51604     },
51605
51606 /**
51607  *      Remove {@link Roo.data.Record}s at the specified indices.
51608  *      @param {Array/Number} selectedIndices The index (or Array of indices) of Records to remove.
51609  */
51610     remove: function(selectedIndices) {
51611                 selectedIndices = [].concat(selectedIndices);
51612                 for (var i = 0; i < selectedIndices.length; i++) {
51613                         var rec = this.store.getAt(selectedIndices[i]);
51614                         this.store.remove(rec);
51615                 }
51616     },
51617
51618 /**
51619  *      Double click fires the event, but also, if this is draggable, and there is only one other
51620  *      related DropZone, it transfers the selected node.
51621  */
51622     onDblClick : function(e){
51623         var item = this.findItemFromChild(e.getTarget());
51624         if(item){
51625             if (this.fireEvent("dblclick", this, this.indexOf(item), item, e) === false) {
51626                 return false;
51627             }
51628             if (this.dragGroup) {
51629                     var targets = Roo.dd.DragDropMgr.getRelated(this.dragZone, true);
51630                     while (targets.indexOf(this.dropZone) > -1) {
51631                             targets.remove(this.dropZone);
51632                                 }
51633                     if (targets.length == 1) {
51634                                         this.dragZone.cachedTarget = null;
51635                         var el = Roo.get(targets[0].getEl());
51636                         var box = el.getBox(true);
51637                         targets[0].onNodeDrop(el.dom, {
51638                                 target: el.dom,
51639                                 xy: [box.x, box.y + box.height - 1]
51640                         }, null, this.getDragData(e));
51641                     }
51642                 }
51643         }
51644     },
51645     
51646     handleSelection: function(e) {
51647                 this.dragZone.cachedTarget = null;
51648         var item = this.findItemFromChild(e.getTarget());
51649         if (!item) {
51650                 this.clearSelections(true);
51651                 return;
51652         }
51653                 if (item && (this.multiSelect || this.singleSelect)){
51654                         if(this.multiSelect && e.shiftKey && (!e.ctrlKey) && this.lastSelection){
51655                                 this.select(this.getNodes(this.indexOf(this.lastSelection), item.nodeIndex), false);
51656                         }else if (this.isSelected(this.getNode(item)) && e.ctrlKey){
51657                                 this.unselect(item);
51658                         } else {
51659                                 this.select(item, this.multiSelect && e.ctrlKey);
51660                                 this.lastSelection = item;
51661                         }
51662                 }
51663     },
51664
51665     onItemClick : function(item, index, e){
51666                 if(this.fireEvent("beforeclick", this, index, item, e) === false){
51667                         return false;
51668                 }
51669                 return true;
51670     },
51671
51672     unselect : function(nodeInfo, suppressEvent){
51673                 var node = this.getNode(nodeInfo);
51674                 if(node && this.isSelected(node)){
51675                         if(this.fireEvent("beforeselect", this, node, this.selections) !== false){
51676                                 Roo.fly(node).removeClass(this.selectedClass);
51677                                 this.selections.remove(node);
51678                                 if(!suppressEvent){
51679                                         this.fireEvent("selectionchange", this, this.selections);
51680                                 }
51681                         }
51682                 }
51683     }
51684 });
51685 /*
51686  * Based on:
51687  * Ext JS Library 1.1.1
51688  * Copyright(c) 2006-2007, Ext JS, LLC.
51689  *
51690  * Originally Released Under LGPL - original licence link has changed is not relivant.
51691  *
51692  * Fork - LGPL
51693  * <script type="text/javascript">
51694  */
51695  
51696 /**
51697  * @class Roo.LayoutManager
51698  * @extends Roo.util.Observable
51699  * Base class for layout managers.
51700  */
51701 Roo.LayoutManager = function(container, config){
51702     Roo.LayoutManager.superclass.constructor.call(this);
51703     this.el = Roo.get(container);
51704     // ie scrollbar fix
51705     if(this.el.dom == document.body && Roo.isIE && !config.allowScroll){
51706         document.body.scroll = "no";
51707     }else if(this.el.dom != document.body && this.el.getStyle('position') == 'static'){
51708         this.el.position('relative');
51709     }
51710     this.id = this.el.id;
51711     this.el.addClass("x-layout-container");
51712     /** false to disable window resize monitoring @type Boolean */
51713     this.monitorWindowResize = true;
51714     this.regions = {};
51715     this.addEvents({
51716         /**
51717          * @event layout
51718          * Fires when a layout is performed. 
51719          * @param {Roo.LayoutManager} this
51720          */
51721         "layout" : true,
51722         /**
51723          * @event regionresized
51724          * Fires when the user resizes a region. 
51725          * @param {Roo.LayoutRegion} region The resized region
51726          * @param {Number} newSize The new size (width for east/west, height for north/south)
51727          */
51728         "regionresized" : true,
51729         /**
51730          * @event regioncollapsed
51731          * Fires when a region is collapsed. 
51732          * @param {Roo.LayoutRegion} region The collapsed region
51733          */
51734         "regioncollapsed" : true,
51735         /**
51736          * @event regionexpanded
51737          * Fires when a region is expanded.  
51738          * @param {Roo.LayoutRegion} region The expanded region
51739          */
51740         "regionexpanded" : true
51741     });
51742     this.updating = false;
51743     Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
51744 };
51745
51746 Roo.extend(Roo.LayoutManager, Roo.util.Observable, {
51747     /**
51748      * Returns true if this layout is currently being updated
51749      * @return {Boolean}
51750      */
51751     isUpdating : function(){
51752         return this.updating; 
51753     },
51754     
51755     /**
51756      * Suspend the LayoutManager from doing auto-layouts while
51757      * making multiple add or remove calls
51758      */
51759     beginUpdate : function(){
51760         this.updating = true;    
51761     },
51762     
51763     /**
51764      * Restore auto-layouts and optionally disable the manager from performing a layout
51765      * @param {Boolean} noLayout true to disable a layout update 
51766      */
51767     endUpdate : function(noLayout){
51768         this.updating = false;
51769         if(!noLayout){
51770             this.layout();
51771         }    
51772     },
51773     
51774     layout: function(){
51775         
51776     },
51777     
51778     onRegionResized : function(region, newSize){
51779         this.fireEvent("regionresized", region, newSize);
51780         this.layout();
51781     },
51782     
51783     onRegionCollapsed : function(region){
51784         this.fireEvent("regioncollapsed", region);
51785     },
51786     
51787     onRegionExpanded : function(region){
51788         this.fireEvent("regionexpanded", region);
51789     },
51790         
51791     /**
51792      * Returns the size of the current view. This method normalizes document.body and element embedded layouts and
51793      * performs box-model adjustments.
51794      * @return {Object} The size as an object {width: (the width), height: (the height)}
51795      */
51796     getViewSize : function(){
51797         var size;
51798         if(this.el.dom != document.body){
51799             size = this.el.getSize();
51800         }else{
51801             size = {width: Roo.lib.Dom.getViewWidth(), height: Roo.lib.Dom.getViewHeight()};
51802         }
51803         size.width -= this.el.getBorderWidth("lr")-this.el.getPadding("lr");
51804         size.height -= this.el.getBorderWidth("tb")-this.el.getPadding("tb");
51805         return size;
51806     },
51807     
51808     /**
51809      * Returns the Element this layout is bound to.
51810      * @return {Roo.Element}
51811      */
51812     getEl : function(){
51813         return this.el;
51814     },
51815     
51816     /**
51817      * Returns the specified region.
51818      * @param {String} target The region key ('center', 'north', 'south', 'east' or 'west')
51819      * @return {Roo.LayoutRegion}
51820      */
51821     getRegion : function(target){
51822         return this.regions[target.toLowerCase()];
51823     },
51824     
51825     onWindowResize : function(){
51826         if(this.monitorWindowResize){
51827             this.layout();
51828         }
51829     }
51830 });/*
51831  * Based on:
51832  * Ext JS Library 1.1.1
51833  * Copyright(c) 2006-2007, Ext JS, LLC.
51834  *
51835  * Originally Released Under LGPL - original licence link has changed is not relivant.
51836  *
51837  * Fork - LGPL
51838  * <script type="text/javascript">
51839  */
51840 /**
51841  * @class Roo.BorderLayout
51842  * @extends Roo.LayoutManager
51843  * This class represents a common layout manager used in desktop applications. For screenshots and more details,
51844  * please see: <br><br>
51845  * <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>
51846  * <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>
51847  * Example:
51848  <pre><code>
51849  var layout = new Roo.BorderLayout(document.body, {
51850     north: {
51851         initialSize: 25,
51852         titlebar: false
51853     },
51854     west: {
51855         split:true,
51856         initialSize: 200,
51857         minSize: 175,
51858         maxSize: 400,
51859         titlebar: true,
51860         collapsible: true
51861     },
51862     east: {
51863         split:true,
51864         initialSize: 202,
51865         minSize: 175,
51866         maxSize: 400,
51867         titlebar: true,
51868         collapsible: true
51869     },
51870     south: {
51871         split:true,
51872         initialSize: 100,
51873         minSize: 100,
51874         maxSize: 200,
51875         titlebar: true,
51876         collapsible: true
51877     },
51878     center: {
51879         titlebar: true,
51880         autoScroll:true,
51881         resizeTabs: true,
51882         minTabWidth: 50,
51883         preferredTabWidth: 150
51884     }
51885 });
51886
51887 // shorthand
51888 var CP = Roo.ContentPanel;
51889
51890 layout.beginUpdate();
51891 layout.add("north", new CP("north", "North"));
51892 layout.add("south", new CP("south", {title: "South", closable: true}));
51893 layout.add("west", new CP("west", {title: "West"}));
51894 layout.add("east", new CP("autoTabs", {title: "Auto Tabs", closable: true}));
51895 layout.add("center", new CP("center1", {title: "Close Me", closable: true}));
51896 layout.add("center", new CP("center2", {title: "Center Panel", closable: false}));
51897 layout.getRegion("center").showPanel("center1");
51898 layout.endUpdate();
51899 </code></pre>
51900
51901 <b>The container the layout is rendered into can be either the body element or any other element.
51902 If it is not the body element, the container needs to either be an absolute positioned element,
51903 or you will need to add "position:relative" to the css of the container.  You will also need to specify
51904 the container size if it is not the body element.</b>
51905
51906 * @constructor
51907 * Create a new BorderLayout
51908 * @param {String/HTMLElement/Element} container The container this layout is bound to
51909 * @param {Object} config Configuration options
51910  */
51911 Roo.BorderLayout = function(container, config){
51912     config = config || {};
51913     Roo.BorderLayout.superclass.constructor.call(this, container, config);
51914     this.factory = config.factory || Roo.BorderLayout.RegionFactory;
51915     for(var i = 0, len = this.factory.validRegions.length; i < len; i++) {
51916         var target = this.factory.validRegions[i];
51917         if(config[target]){
51918             this.addRegion(target, config[target]);
51919         }
51920     }
51921 };
51922
51923 Roo.extend(Roo.BorderLayout, Roo.LayoutManager, {
51924     /**
51925      * Creates and adds a new region if it doesn't already exist.
51926      * @param {String} target The target region key (north, south, east, west or center).
51927      * @param {Object} config The regions config object
51928      * @return {BorderLayoutRegion} The new region
51929      */
51930     addRegion : function(target, config){
51931         if(!this.regions[target]){
51932             var r = this.factory.create(target, this, config);
51933             this.bindRegion(target, r);
51934         }
51935         return this.regions[target];
51936     },
51937
51938     // private (kinda)
51939     bindRegion : function(name, r){
51940         this.regions[name] = r;
51941         r.on("visibilitychange", this.layout, this);
51942         r.on("paneladded", this.layout, this);
51943         r.on("panelremoved", this.layout, this);
51944         r.on("invalidated", this.layout, this);
51945         r.on("resized", this.onRegionResized, this);
51946         r.on("collapsed", this.onRegionCollapsed, this);
51947         r.on("expanded", this.onRegionExpanded, this);
51948     },
51949
51950     /**
51951      * Performs a layout update.
51952      */
51953     layout : function(){
51954         if(this.updating) {
51955             return;
51956         }
51957         var size = this.getViewSize();
51958         var w = size.width;
51959         var h = size.height;
51960         var centerW = w;
51961         var centerH = h;
51962         var centerY = 0;
51963         var centerX = 0;
51964         //var x = 0, y = 0;
51965
51966         var rs = this.regions;
51967         var north = rs["north"];
51968         var south = rs["south"]; 
51969         var west = rs["west"];
51970         var east = rs["east"];
51971         var center = rs["center"];
51972         //if(this.hideOnLayout){ // not supported anymore
51973             //c.el.setStyle("display", "none");
51974         //}
51975         if(north && north.isVisible()){
51976             var b = north.getBox();
51977             var m = north.getMargins();
51978             b.width = w - (m.left+m.right);
51979             b.x = m.left;
51980             b.y = m.top;
51981             centerY = b.height + b.y + m.bottom;
51982             centerH -= centerY;
51983             north.updateBox(this.safeBox(b));
51984         }
51985         if(south && south.isVisible()){
51986             var b = south.getBox();
51987             var m = south.getMargins();
51988             b.width = w - (m.left+m.right);
51989             b.x = m.left;
51990             var totalHeight = (b.height + m.top + m.bottom);
51991             b.y = h - totalHeight + m.top;
51992             centerH -= totalHeight;
51993             south.updateBox(this.safeBox(b));
51994         }
51995         if(west && west.isVisible()){
51996             var b = west.getBox();
51997             var m = west.getMargins();
51998             b.height = centerH - (m.top+m.bottom);
51999             b.x = m.left;
52000             b.y = centerY + m.top;
52001             var totalWidth = (b.width + m.left + m.right);
52002             centerX += totalWidth;
52003             centerW -= totalWidth;
52004             west.updateBox(this.safeBox(b));
52005         }
52006         if(east && east.isVisible()){
52007             var b = east.getBox();
52008             var m = east.getMargins();
52009             b.height = centerH - (m.top+m.bottom);
52010             var totalWidth = (b.width + m.left + m.right);
52011             b.x = w - totalWidth + m.left;
52012             b.y = centerY + m.top;
52013             centerW -= totalWidth;
52014             east.updateBox(this.safeBox(b));
52015         }
52016         if(center){
52017             var m = center.getMargins();
52018             var centerBox = {
52019                 x: centerX + m.left,
52020                 y: centerY + m.top,
52021                 width: centerW - (m.left+m.right),
52022                 height: centerH - (m.top+m.bottom)
52023             };
52024             //if(this.hideOnLayout){
52025                 //center.el.setStyle("display", "block");
52026             //}
52027             center.updateBox(this.safeBox(centerBox));
52028         }
52029         this.el.repaint();
52030         this.fireEvent("layout", this);
52031     },
52032
52033     // private
52034     safeBox : function(box){
52035         box.width = Math.max(0, box.width);
52036         box.height = Math.max(0, box.height);
52037         return box;
52038     },
52039
52040     /**
52041      * Adds a ContentPanel (or subclass) to this layout.
52042      * @param {String} target The target region key (north, south, east, west or center).
52043      * @param {Roo.ContentPanel} panel The panel to add
52044      * @return {Roo.ContentPanel} The added panel
52045      */
52046     add : function(target, panel){
52047          
52048         target = target.toLowerCase();
52049         return this.regions[target].add(panel);
52050     },
52051
52052     /**
52053      * Remove a ContentPanel (or subclass) to this layout.
52054      * @param {String} target The target region key (north, south, east, west or center).
52055      * @param {Number/String/Roo.ContentPanel} panel The index, id or panel to remove
52056      * @return {Roo.ContentPanel} The removed panel
52057      */
52058     remove : function(target, panel){
52059         target = target.toLowerCase();
52060         return this.regions[target].remove(panel);
52061     },
52062
52063     /**
52064      * Searches all regions for a panel with the specified id
52065      * @param {String} panelId
52066      * @return {Roo.ContentPanel} The panel or null if it wasn't found
52067      */
52068     findPanel : function(panelId){
52069         var rs = this.regions;
52070         for(var target in rs){
52071             if(typeof rs[target] != "function"){
52072                 var p = rs[target].getPanel(panelId);
52073                 if(p){
52074                     return p;
52075                 }
52076             }
52077         }
52078         return null;
52079     },
52080
52081     /**
52082      * Searches all regions for a panel with the specified id and activates (shows) it.
52083      * @param {String/ContentPanel} panelId The panels id or the panel itself
52084      * @return {Roo.ContentPanel} The shown panel or null
52085      */
52086     showPanel : function(panelId) {
52087       var rs = this.regions;
52088       for(var target in rs){
52089          var r = rs[target];
52090          if(typeof r != "function"){
52091             if(r.hasPanel(panelId)){
52092                return r.showPanel(panelId);
52093             }
52094          }
52095       }
52096       return null;
52097    },
52098
52099    /**
52100      * Restores this layout's state using Roo.state.Manager or the state provided by the passed provider.
52101      * @param {Roo.state.Provider} provider (optional) An alternate state provider
52102      */
52103     restoreState : function(provider){
52104         if(!provider){
52105             provider = Roo.state.Manager;
52106         }
52107         var sm = new Roo.LayoutStateManager();
52108         sm.init(this, provider);
52109     },
52110
52111     /**
52112      * Adds a batch of multiple ContentPanels dynamically by passing a special regions config object.  This config
52113      * object should contain properties for each region to add ContentPanels to, and each property's value should be
52114      * a valid ContentPanel config object.  Example:
52115      * <pre><code>
52116 // Create the main layout
52117 var layout = new Roo.BorderLayout('main-ct', {
52118     west: {
52119         split:true,
52120         minSize: 175,
52121         titlebar: true
52122     },
52123     center: {
52124         title:'Components'
52125     }
52126 }, 'main-ct');
52127
52128 // Create and add multiple ContentPanels at once via configs
52129 layout.batchAdd({
52130    west: {
52131        id: 'source-files',
52132        autoCreate:true,
52133        title:'Ext Source Files',
52134        autoScroll:true,
52135        fitToFrame:true
52136    },
52137    center : {
52138        el: cview,
52139        autoScroll:true,
52140        fitToFrame:true,
52141        toolbar: tb,
52142        resizeEl:'cbody'
52143    }
52144 });
52145 </code></pre>
52146      * @param {Object} regions An object containing ContentPanel configs by region name
52147      */
52148     batchAdd : function(regions){
52149         this.beginUpdate();
52150         for(var rname in regions){
52151             var lr = this.regions[rname];
52152             if(lr){
52153                 this.addTypedPanels(lr, regions[rname]);
52154             }
52155         }
52156         this.endUpdate();
52157     },
52158
52159     // private
52160     addTypedPanels : function(lr, ps){
52161         if(typeof ps == 'string'){
52162             lr.add(new Roo.ContentPanel(ps));
52163         }
52164         else if(ps instanceof Array){
52165             for(var i =0, len = ps.length; i < len; i++){
52166                 this.addTypedPanels(lr, ps[i]);
52167             }
52168         }
52169         else if(!ps.events){ // raw config?
52170             var el = ps.el;
52171             delete ps.el; // prevent conflict
52172             lr.add(new Roo.ContentPanel(el || Roo.id(), ps));
52173         }
52174         else {  // panel object assumed!
52175             lr.add(ps);
52176         }
52177     },
52178     /**
52179      * Adds a xtype elements to the layout.
52180      * <pre><code>
52181
52182 layout.addxtype({
52183        xtype : 'ContentPanel',
52184        region: 'west',
52185        items: [ .... ]
52186    }
52187 );
52188
52189 layout.addxtype({
52190         xtype : 'NestedLayoutPanel',
52191         region: 'west',
52192         layout: {
52193            center: { },
52194            west: { }   
52195         },
52196         items : [ ... list of content panels or nested layout panels.. ]
52197    }
52198 );
52199 </code></pre>
52200      * @param {Object} cfg Xtype definition of item to add.
52201      */
52202     addxtype : function(cfg)
52203     {
52204         // basically accepts a pannel...
52205         // can accept a layout region..!?!?
52206         //Roo.log('Roo.BorderLayout add ' + cfg.xtype)
52207         
52208         if (!cfg.xtype.match(/Panel$/)) {
52209             return false;
52210         }
52211         var ret = false;
52212         
52213         if (typeof(cfg.region) == 'undefined') {
52214             Roo.log("Failed to add Panel, region was not set");
52215             Roo.log(cfg);
52216             return false;
52217         }
52218         var region = cfg.region;
52219         delete cfg.region;
52220         
52221           
52222         var xitems = [];
52223         if (cfg.items) {
52224             xitems = cfg.items;
52225             delete cfg.items;
52226         }
52227         var nb = false;
52228         
52229         switch(cfg.xtype) 
52230         {
52231             case 'ContentPanel':  // ContentPanel (el, cfg)
52232             case 'ScrollPanel':  // ContentPanel (el, cfg)
52233             case 'ViewPanel': 
52234                 if(cfg.autoCreate) {
52235                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52236                 } else {
52237                     var el = this.el.createChild();
52238                     ret = new Roo[cfg.xtype](el, cfg); // new panel!!!!!
52239                 }
52240                 
52241                 this.add(region, ret);
52242                 break;
52243             
52244             
52245             case 'TreePanel': // our new panel!
52246                 cfg.el = this.el.createChild();
52247                 ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52248                 this.add(region, ret);
52249                 break;
52250             
52251             case 'NestedLayoutPanel': 
52252                 // create a new Layout (which is  a Border Layout...
52253                 var el = this.el.createChild();
52254                 var clayout = cfg.layout;
52255                 delete cfg.layout;
52256                 clayout.items   = clayout.items  || [];
52257                 // replace this exitems with the clayout ones..
52258                 xitems = clayout.items;
52259                  
52260                 
52261                 if (region == 'center' && this.active && this.getRegion('center').panels.length < 1) {
52262                     cfg.background = false;
52263                 }
52264                 var layout = new Roo.BorderLayout(el, clayout);
52265                 
52266                 ret = new Roo[cfg.xtype](layout, cfg); // new panel!!!!!
52267                 //console.log('adding nested layout panel '  + cfg.toSource());
52268                 this.add(region, ret);
52269                 nb = {}; /// find first...
52270                 break;
52271                 
52272             case 'GridPanel': 
52273             
52274                 // needs grid and region
52275                 
52276                 //var el = this.getRegion(region).el.createChild();
52277                 var el = this.el.createChild();
52278                 // create the grid first...
52279                 
52280                 var grid = new Roo.grid[cfg.grid.xtype](el, cfg.grid);
52281                 delete cfg.grid;
52282                 if (region == 'center' && this.active ) {
52283                     cfg.background = false;
52284                 }
52285                 ret = new Roo[cfg.xtype](grid, cfg); // new panel!!!!!
52286                 
52287                 this.add(region, ret);
52288                 if (cfg.background) {
52289                     ret.on('activate', function(gp) {
52290                         if (!gp.grid.rendered) {
52291                             gp.grid.render();
52292                         }
52293                     });
52294                 } else {
52295                     grid.render();
52296                 }
52297                 break;
52298            
52299            
52300            
52301                 
52302                 
52303                 
52304             default:
52305                 if (typeof(Roo[cfg.xtype]) != 'undefined') {
52306                     
52307                     ret = new Roo[cfg.xtype](cfg); // new panel!!!!!
52308                     this.add(region, ret);
52309                 } else {
52310                 
52311                     alert("Can not add '" + cfg.xtype + "' to BorderLayout");
52312                     return null;
52313                 }
52314                 
52315              // GridPanel (grid, cfg)
52316             
52317         }
52318         this.beginUpdate();
52319         // add children..
52320         var region = '';
52321         var abn = {};
52322         Roo.each(xitems, function(i)  {
52323             region = nb && i.region ? i.region : false;
52324             
52325             var add = ret.addxtype(i);
52326            
52327             if (region) {
52328                 nb[region] = nb[region] == undefined ? 0 : nb[region]+1;
52329                 if (!i.background) {
52330                     abn[region] = nb[region] ;
52331                 }
52332             }
52333             
52334         });
52335         this.endUpdate();
52336
52337         // make the last non-background panel active..
52338         //if (nb) { Roo.log(abn); }
52339         if (nb) {
52340             
52341             for(var r in abn) {
52342                 region = this.getRegion(r);
52343                 if (region) {
52344                     // tried using nb[r], but it does not work..
52345                      
52346                     region.showPanel(abn[r]);
52347                    
52348                 }
52349             }
52350         }
52351         return ret;
52352         
52353     }
52354 });
52355
52356 /**
52357  * Shortcut for creating a new BorderLayout object and adding one or more ContentPanels to it in a single step, handling
52358  * the beginUpdate and endUpdate calls internally.  The key to this method is the <b>panels</b> property that can be
52359  * provided with each region config, which allows you to add ContentPanel configs in addition to the region configs
52360  * during creation.  The following code is equivalent to the constructor-based example at the beginning of this class:
52361  * <pre><code>
52362 // shorthand
52363 var CP = Roo.ContentPanel;
52364
52365 var layout = Roo.BorderLayout.create({
52366     north: {
52367         initialSize: 25,
52368         titlebar: false,
52369         panels: [new CP("north", "North")]
52370     },
52371     west: {
52372         split:true,
52373         initialSize: 200,
52374         minSize: 175,
52375         maxSize: 400,
52376         titlebar: true,
52377         collapsible: true,
52378         panels: [new CP("west", {title: "West"})]
52379     },
52380     east: {
52381         split:true,
52382         initialSize: 202,
52383         minSize: 175,
52384         maxSize: 400,
52385         titlebar: true,
52386         collapsible: true,
52387         panels: [new CP("autoTabs", {title: "Auto Tabs", closable: true})]
52388     },
52389     south: {
52390         split:true,
52391         initialSize: 100,
52392         minSize: 100,
52393         maxSize: 200,
52394         titlebar: true,
52395         collapsible: true,
52396         panels: [new CP("south", {title: "South", closable: true})]
52397     },
52398     center: {
52399         titlebar: true,
52400         autoScroll:true,
52401         resizeTabs: true,
52402         minTabWidth: 50,
52403         preferredTabWidth: 150,
52404         panels: [
52405             new CP("center1", {title: "Close Me", closable: true}),
52406             new CP("center2", {title: "Center Panel", closable: false})
52407         ]
52408     }
52409 }, document.body);
52410
52411 layout.getRegion("center").showPanel("center1");
52412 </code></pre>
52413  * @param config
52414  * @param targetEl
52415  */
52416 Roo.BorderLayout.create = function(config, targetEl){
52417     var layout = new Roo.BorderLayout(targetEl || document.body, config);
52418     layout.beginUpdate();
52419     var regions = Roo.BorderLayout.RegionFactory.validRegions;
52420     for(var j = 0, jlen = regions.length; j < jlen; j++){
52421         var lr = regions[j];
52422         if(layout.regions[lr] && config[lr].panels){
52423             var r = layout.regions[lr];
52424             var ps = config[lr].panels;
52425             layout.addTypedPanels(r, ps);
52426         }
52427     }
52428     layout.endUpdate();
52429     return layout;
52430 };
52431
52432 // private
52433 Roo.BorderLayout.RegionFactory = {
52434     // private
52435     validRegions : ["north","south","east","west","center"],
52436
52437     // private
52438     create : function(target, mgr, config){
52439         target = target.toLowerCase();
52440         if(config.lightweight || config.basic){
52441             return new Roo.BasicLayoutRegion(mgr, config, target);
52442         }
52443         switch(target){
52444             case "north":
52445                 return new Roo.NorthLayoutRegion(mgr, config);
52446             case "south":
52447                 return new Roo.SouthLayoutRegion(mgr, config);
52448             case "east":
52449                 return new Roo.EastLayoutRegion(mgr, config);
52450             case "west":
52451                 return new Roo.WestLayoutRegion(mgr, config);
52452             case "center":
52453                 return new Roo.CenterLayoutRegion(mgr, config);
52454         }
52455         throw 'Layout region "'+target+'" not supported.';
52456     }
52457 };/*
52458  * Based on:
52459  * Ext JS Library 1.1.1
52460  * Copyright(c) 2006-2007, Ext JS, LLC.
52461  *
52462  * Originally Released Under LGPL - original licence link has changed is not relivant.
52463  *
52464  * Fork - LGPL
52465  * <script type="text/javascript">
52466  */
52467  
52468 /**
52469  * @class Roo.BasicLayoutRegion
52470  * @extends Roo.util.Observable
52471  * This class represents a lightweight region in a layout manager. This region does not move dom nodes
52472  * and does not have a titlebar, tabs or any other features. All it does is size and position 
52473  * panels. To create a BasicLayoutRegion, add lightweight:true or basic:true to your regions config.
52474  */
52475 Roo.BasicLayoutRegion = function(mgr, config, pos, skipConfig){
52476     this.mgr = mgr;
52477     this.position  = pos;
52478     this.events = {
52479         /**
52480          * @scope Roo.BasicLayoutRegion
52481          */
52482         
52483         /**
52484          * @event beforeremove
52485          * Fires before a panel is removed (or closed). To cancel the removal set "e.cancel = true" on the event argument.
52486          * @param {Roo.LayoutRegion} this
52487          * @param {Roo.ContentPanel} panel The panel
52488          * @param {Object} e The cancel event object
52489          */
52490         "beforeremove" : true,
52491         /**
52492          * @event invalidated
52493          * Fires when the layout for this region is changed.
52494          * @param {Roo.LayoutRegion} this
52495          */
52496         "invalidated" : true,
52497         /**
52498          * @event visibilitychange
52499          * Fires when this region is shown or hidden 
52500          * @param {Roo.LayoutRegion} this
52501          * @param {Boolean} visibility true or false
52502          */
52503         "visibilitychange" : true,
52504         /**
52505          * @event paneladded
52506          * Fires when a panel is added. 
52507          * @param {Roo.LayoutRegion} this
52508          * @param {Roo.ContentPanel} panel The panel
52509          */
52510         "paneladded" : true,
52511         /**
52512          * @event panelremoved
52513          * Fires when a panel is removed. 
52514          * @param {Roo.LayoutRegion} this
52515          * @param {Roo.ContentPanel} panel The panel
52516          */
52517         "panelremoved" : true,
52518         /**
52519          * @event beforecollapse
52520          * Fires when this region before collapse.
52521          * @param {Roo.LayoutRegion} this
52522          */
52523         "beforecollapse" : true,
52524         /**
52525          * @event collapsed
52526          * Fires when this region is collapsed.
52527          * @param {Roo.LayoutRegion} this
52528          */
52529         "collapsed" : true,
52530         /**
52531          * @event expanded
52532          * Fires when this region is expanded.
52533          * @param {Roo.LayoutRegion} this
52534          */
52535         "expanded" : true,
52536         /**
52537          * @event slideshow
52538          * Fires when this region is slid into view.
52539          * @param {Roo.LayoutRegion} this
52540          */
52541         "slideshow" : true,
52542         /**
52543          * @event slidehide
52544          * Fires when this region slides out of view. 
52545          * @param {Roo.LayoutRegion} this
52546          */
52547         "slidehide" : true,
52548         /**
52549          * @event panelactivated
52550          * Fires when a panel is activated. 
52551          * @param {Roo.LayoutRegion} this
52552          * @param {Roo.ContentPanel} panel The activated panel
52553          */
52554         "panelactivated" : true,
52555         /**
52556          * @event resized
52557          * Fires when the user resizes this region. 
52558          * @param {Roo.LayoutRegion} this
52559          * @param {Number} newSize The new size (width for east/west, height for north/south)
52560          */
52561         "resized" : true
52562     };
52563     /** A collection of panels in this region. @type Roo.util.MixedCollection */
52564     this.panels = new Roo.util.MixedCollection();
52565     this.panels.getKey = this.getPanelId.createDelegate(this);
52566     this.box = null;
52567     this.activePanel = null;
52568     // ensure listeners are added...
52569     
52570     if (config.listeners || config.events) {
52571         Roo.BasicLayoutRegion.superclass.constructor.call(this, {
52572             listeners : config.listeners || {},
52573             events : config.events || {}
52574         });
52575     }
52576     
52577     if(skipConfig !== true){
52578         this.applyConfig(config);
52579     }
52580 };
52581
52582 Roo.extend(Roo.BasicLayoutRegion, Roo.util.Observable, {
52583     getPanelId : function(p){
52584         return p.getId();
52585     },
52586     
52587     applyConfig : function(config){
52588         this.margins = config.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52589         this.config = config;
52590         
52591     },
52592     
52593     /**
52594      * Resizes the region to the specified size. For vertical regions (west, east) this adjusts 
52595      * the width, for horizontal (north, south) the height.
52596      * @param {Number} newSize The new width or height
52597      */
52598     resizeTo : function(newSize){
52599         var el = this.el ? this.el :
52600                  (this.activePanel ? this.activePanel.getEl() : null);
52601         if(el){
52602             switch(this.position){
52603                 case "east":
52604                 case "west":
52605                     el.setWidth(newSize);
52606                     this.fireEvent("resized", this, newSize);
52607                 break;
52608                 case "north":
52609                 case "south":
52610                     el.setHeight(newSize);
52611                     this.fireEvent("resized", this, newSize);
52612                 break;                
52613             }
52614         }
52615     },
52616     
52617     getBox : function(){
52618         return this.activePanel ? this.activePanel.getEl().getBox(false, true) : null;
52619     },
52620     
52621     getMargins : function(){
52622         return this.margins;
52623     },
52624     
52625     updateBox : function(box){
52626         this.box = box;
52627         var el = this.activePanel.getEl();
52628         el.dom.style.left = box.x + "px";
52629         el.dom.style.top = box.y + "px";
52630         this.activePanel.setSize(box.width, box.height);
52631     },
52632     
52633     /**
52634      * Returns the container element for this region.
52635      * @return {Roo.Element}
52636      */
52637     getEl : function(){
52638         return this.activePanel;
52639     },
52640     
52641     /**
52642      * Returns true if this region is currently visible.
52643      * @return {Boolean}
52644      */
52645     isVisible : function(){
52646         return this.activePanel ? true : false;
52647     },
52648     
52649     setActivePanel : function(panel){
52650         panel = this.getPanel(panel);
52651         if(this.activePanel && this.activePanel != panel){
52652             this.activePanel.setActiveState(false);
52653             this.activePanel.getEl().setLeftTop(-10000,-10000);
52654         }
52655         this.activePanel = panel;
52656         panel.setActiveState(true);
52657         if(this.box){
52658             panel.setSize(this.box.width, this.box.height);
52659         }
52660         this.fireEvent("panelactivated", this, panel);
52661         this.fireEvent("invalidated");
52662     },
52663     
52664     /**
52665      * Show the specified panel.
52666      * @param {Number/String/ContentPanel} panelId The panels index, id or the panel itself
52667      * @return {Roo.ContentPanel} The shown panel or null
52668      */
52669     showPanel : function(panel){
52670         if(panel = this.getPanel(panel)){
52671             this.setActivePanel(panel);
52672         }
52673         return panel;
52674     },
52675     
52676     /**
52677      * Get the active panel for this region.
52678      * @return {Roo.ContentPanel} The active panel or null
52679      */
52680     getActivePanel : function(){
52681         return this.activePanel;
52682     },
52683     
52684     /**
52685      * Add the passed ContentPanel(s)
52686      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
52687      * @return {Roo.ContentPanel} The panel added (if only one was added)
52688      */
52689     add : function(panel){
52690         if(arguments.length > 1){
52691             for(var i = 0, len = arguments.length; i < len; i++) {
52692                 this.add(arguments[i]);
52693             }
52694             return null;
52695         }
52696         if(this.hasPanel(panel)){
52697             this.showPanel(panel);
52698             return panel;
52699         }
52700         var el = panel.getEl();
52701         if(el.dom.parentNode != this.mgr.el.dom){
52702             this.mgr.el.dom.appendChild(el.dom);
52703         }
52704         if(panel.setRegion){
52705             panel.setRegion(this);
52706         }
52707         this.panels.add(panel);
52708         el.setStyle("position", "absolute");
52709         if(!panel.background){
52710             this.setActivePanel(panel);
52711             if(this.config.initialSize && this.panels.getCount()==1){
52712                 this.resizeTo(this.config.initialSize);
52713             }
52714         }
52715         this.fireEvent("paneladded", this, panel);
52716         return panel;
52717     },
52718     
52719     /**
52720      * Returns true if the panel is in this region.
52721      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52722      * @return {Boolean}
52723      */
52724     hasPanel : function(panel){
52725         if(typeof panel == "object"){ // must be panel obj
52726             panel = panel.getId();
52727         }
52728         return this.getPanel(panel) ? true : false;
52729     },
52730     
52731     /**
52732      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
52733      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52734      * @param {Boolean} preservePanel Overrides the config preservePanel option
52735      * @return {Roo.ContentPanel} The panel that was removed
52736      */
52737     remove : function(panel, preservePanel){
52738         panel = this.getPanel(panel);
52739         if(!panel){
52740             return null;
52741         }
52742         var e = {};
52743         this.fireEvent("beforeremove", this, panel, e);
52744         if(e.cancel === true){
52745             return null;
52746         }
52747         var panelId = panel.getId();
52748         this.panels.removeKey(panelId);
52749         return panel;
52750     },
52751     
52752     /**
52753      * Returns the panel specified or null if it's not in this region.
52754      * @param {Number/String/ContentPanel} panel The panels index, id or the panel itself
52755      * @return {Roo.ContentPanel}
52756      */
52757     getPanel : function(id){
52758         if(typeof id == "object"){ // must be panel obj
52759             return id;
52760         }
52761         return this.panels.get(id);
52762     },
52763     
52764     /**
52765      * Returns this regions position (north/south/east/west/center).
52766      * @return {String} 
52767      */
52768     getPosition: function(){
52769         return this.position;    
52770     }
52771 });/*
52772  * Based on:
52773  * Ext JS Library 1.1.1
52774  * Copyright(c) 2006-2007, Ext JS, LLC.
52775  *
52776  * Originally Released Under LGPL - original licence link has changed is not relivant.
52777  *
52778  * Fork - LGPL
52779  * <script type="text/javascript">
52780  */
52781  
52782 /**
52783  * @class Roo.LayoutRegion
52784  * @extends Roo.BasicLayoutRegion
52785  * This class represents a region in a layout manager.
52786  * @cfg {Boolean}   collapsible     False to disable collapsing (defaults to true)
52787  * @cfg {Boolean}   collapsed       True to set the initial display to collapsed (defaults to false)
52788  * @cfg {Boolean}   floatable       False to disable floating (defaults to true)
52789  * @cfg {Object}    margins         Margins for the element (defaults to {top: 0, left: 0, right:0, bottom: 0})
52790  * @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})
52791  * @cfg {String}    tabPosition     (top|bottom) "top" or "bottom" (defaults to "bottom")
52792  * @cfg {String}    collapsedTitle  Optional string message to display in the collapsed block of a north or south region
52793  * @cfg {Boolean}   alwaysShowTabs  True to always display tabs even when there is only 1 panel (defaults to false)
52794  * @cfg {Boolean}   autoScroll      True to enable overflow scrolling (defaults to false)
52795  * @cfg {Boolean}   titlebar        True to display a title bar (defaults to true)
52796  * @cfg {String}    title           The title for the region (overrides panel titles)
52797  * @cfg {Boolean}   animate         True to animate expand/collapse (defaults to false)
52798  * @cfg {Boolean}   autoHide        False to disable auto hiding when the mouse leaves the "floated" region (defaults to true)
52799  * @cfg {Boolean}   preservePanels  True to preserve removed panels so they can be readded later (defaults to false)
52800  * @cfg {Boolean}   closeOnTab      True to place the close icon on the tabs instead of the region titlebar (defaults to false)
52801  * @cfg {Boolean}   hideTabs        True to hide the tab strip (defaults to false)
52802  * @cfg {Boolean}   resizeTabs      True to enable automatic tab resizing. This will resize the tabs so they are all the same size and fit within
52803  *                      the space available, similar to FireFox 1.5 tabs (defaults to false)
52804  * @cfg {Number}    minTabWidth     The minimum tab width (defaults to 40)
52805  * @cfg {Number}    preferredTabWidth The preferred tab width (defaults to 150)
52806  * @cfg {Boolean}   showPin         True to show a pin button
52807  * @cfg {Boolean}   hidden          True to start the region hidden (defaults to false)
52808  * @cfg {Boolean}   hideWhenEmpty   True to hide the region when it has no panels
52809  * @cfg {Boolean}   disableTabTips  True to disable tab tooltips
52810  * @cfg {Number}    width           For East/West panels
52811  * @cfg {Number}    height          For North/South panels
52812  * @cfg {Boolean}   split           To show the splitter
52813  * @cfg {Boolean}   toolbar         xtype configuration for a toolbar - shows on right of tabbar
52814  */
52815 Roo.LayoutRegion = function(mgr, config, pos){
52816     Roo.LayoutRegion.superclass.constructor.call(this, mgr, config, pos, true);
52817     var dh = Roo.DomHelper;
52818     /** This region's container element 
52819     * @type Roo.Element */
52820     this.el = dh.append(mgr.el.dom, {tag: "div", cls: "x-layout-panel x-layout-panel-" + this.position}, true);
52821     /** This region's title element 
52822     * @type Roo.Element */
52823
52824     this.titleEl = dh.append(this.el.dom, {tag: "div", unselectable: "on", cls: "x-unselectable x-layout-panel-hd x-layout-title-"+this.position, children:[
52825         {tag: "span", cls: "x-unselectable x-layout-panel-hd-text", unselectable: "on", html: "&#160;"},
52826         {tag: "div", cls: "x-unselectable x-layout-panel-hd-tools", unselectable: "on"}
52827     ]}, true);
52828     this.titleEl.enableDisplayMode();
52829     /** This region's title text element 
52830     * @type HTMLElement */
52831     this.titleTextEl = this.titleEl.dom.firstChild;
52832     this.tools = Roo.get(this.titleEl.dom.childNodes[1], true);
52833     this.closeBtn = this.createTool(this.tools.dom, "x-layout-close");
52834     this.closeBtn.enableDisplayMode();
52835     this.closeBtn.on("click", this.closeClicked, this);
52836     this.closeBtn.hide();
52837
52838     this.createBody(config);
52839     this.visible = true;
52840     this.collapsed = false;
52841
52842     if(config.hideWhenEmpty){
52843         this.hide();
52844         this.on("paneladded", this.validateVisibility, this);
52845         this.on("panelremoved", this.validateVisibility, this);
52846     }
52847     this.applyConfig(config);
52848 };
52849
52850 Roo.extend(Roo.LayoutRegion, Roo.BasicLayoutRegion, {
52851
52852     createBody : function(){
52853         /** This region's body element 
52854         * @type Roo.Element */
52855         this.bodyEl = this.el.createChild({tag: "div", cls: "x-layout-panel-body"});
52856     },
52857
52858     applyConfig : function(c){
52859         if(c.collapsible && this.position != "center" && !this.collapsedEl){
52860             var dh = Roo.DomHelper;
52861             if(c.titlebar !== false){
52862                 this.collapseBtn = this.createTool(this.tools.dom, "x-layout-collapse-"+this.position);
52863                 this.collapseBtn.on("click", this.collapse, this);
52864                 this.collapseBtn.enableDisplayMode();
52865
52866                 if(c.showPin === true || this.showPin){
52867                     this.stickBtn = this.createTool(this.tools.dom, "x-layout-stick");
52868                     this.stickBtn.enableDisplayMode();
52869                     this.stickBtn.on("click", this.expand, this);
52870                     this.stickBtn.hide();
52871                 }
52872             }
52873             /** This region's collapsed element
52874             * @type Roo.Element */
52875             this.collapsedEl = dh.append(this.mgr.el.dom, {cls: "x-layout-collapsed x-layout-collapsed-"+this.position, children:[
52876                 {cls: "x-layout-collapsed-tools", children:[{cls: "x-layout-ctools-inner"}]}
52877             ]}, true);
52878             if(c.floatable !== false){
52879                this.collapsedEl.addClassOnOver("x-layout-collapsed-over");
52880                this.collapsedEl.on("click", this.collapseClick, this);
52881             }
52882
52883             if(c.collapsedTitle && (this.position == "north" || this.position== "south")) {
52884                 this.collapsedTitleTextEl = dh.append(this.collapsedEl.dom, {tag: "div", cls: "x-unselectable x-layout-panel-hd-text",
52885                    id: "message", unselectable: "on", style:{"float":"left"}});
52886                this.collapsedTitleTextEl.innerHTML = c.collapsedTitle;
52887              }
52888             this.expandBtn = this.createTool(this.collapsedEl.dom.firstChild.firstChild, "x-layout-expand-"+this.position);
52889             this.expandBtn.on("click", this.expand, this);
52890         }
52891         if(this.collapseBtn){
52892             this.collapseBtn.setVisible(c.collapsible == true);
52893         }
52894         this.cmargins = c.cmargins || this.cmargins ||
52895                          (this.position == "west" || this.position == "east" ?
52896                              {top: 0, left: 2, right:2, bottom: 0} :
52897                              {top: 2, left: 0, right:0, bottom: 2});
52898         this.margins = c.margins || this.margins || {top: 0, left: 0, right:0, bottom: 0};
52899         this.bottomTabs = c.tabPosition != "top";
52900         this.autoScroll = c.autoScroll || false;
52901         if(this.autoScroll){
52902             this.bodyEl.setStyle("overflow", "auto");
52903         }else{
52904             this.bodyEl.setStyle("overflow", "hidden");
52905         }
52906         //if(c.titlebar !== false){
52907             if((!c.titlebar && !c.title) || c.titlebar === false){
52908                 this.titleEl.hide();
52909             }else{
52910                 this.titleEl.show();
52911                 if(c.title){
52912                     this.titleTextEl.innerHTML = c.title;
52913                 }
52914             }
52915         //}
52916         this.duration = c.duration || .30;
52917         this.slideDuration = c.slideDuration || .45;
52918         this.config = c;
52919         if(c.collapsed){
52920             this.collapse(true);
52921         }
52922         if(c.hidden){
52923             this.hide();
52924         }
52925     },
52926     /**
52927      * Returns true if this region is currently visible.
52928      * @return {Boolean}
52929      */
52930     isVisible : function(){
52931         return this.visible;
52932     },
52933
52934     /**
52935      * Updates the title for collapsed north/south regions (used with {@link #collapsedTitle} config option)
52936      * @param {String} title (optional) The title text (accepts HTML markup, defaults to the numeric character reference for a non-breaking space, "&amp;#160;")
52937      */
52938     setCollapsedTitle : function(title){
52939         title = title || "&#160;";
52940         if(this.collapsedTitleTextEl){
52941             this.collapsedTitleTextEl.innerHTML = title;
52942         }
52943     },
52944
52945     getBox : function(){
52946         var b;
52947         if(!this.collapsed){
52948             b = this.el.getBox(false, true);
52949         }else{
52950             b = this.collapsedEl.getBox(false, true);
52951         }
52952         return b;
52953     },
52954
52955     getMargins : function(){
52956         return this.collapsed ? this.cmargins : this.margins;
52957     },
52958
52959     highlight : function(){
52960         this.el.addClass("x-layout-panel-dragover");
52961     },
52962
52963     unhighlight : function(){
52964         this.el.removeClass("x-layout-panel-dragover");
52965     },
52966
52967     updateBox : function(box){
52968         this.box = box;
52969         if(!this.collapsed){
52970             this.el.dom.style.left = box.x + "px";
52971             this.el.dom.style.top = box.y + "px";
52972             this.updateBody(box.width, box.height);
52973         }else{
52974             this.collapsedEl.dom.style.left = box.x + "px";
52975             this.collapsedEl.dom.style.top = box.y + "px";
52976             this.collapsedEl.setSize(box.width, box.height);
52977         }
52978         if(this.tabs){
52979             this.tabs.autoSizeTabs();
52980         }
52981     },
52982
52983     updateBody : function(w, h){
52984         if(w !== null){
52985             this.el.setWidth(w);
52986             w -= this.el.getBorderWidth("rl");
52987             if(this.config.adjustments){
52988                 w += this.config.adjustments[0];
52989             }
52990         }
52991         if(h !== null){
52992             this.el.setHeight(h);
52993             h = this.titleEl && this.titleEl.isDisplayed() ? h - (this.titleEl.getHeight()||0) : h;
52994             h -= this.el.getBorderWidth("tb");
52995             if(this.config.adjustments){
52996                 h += this.config.adjustments[1];
52997             }
52998             this.bodyEl.setHeight(h);
52999             if(this.tabs){
53000                 h = this.tabs.syncHeight(h);
53001             }
53002         }
53003         if(this.panelSize){
53004             w = w !== null ? w : this.panelSize.width;
53005             h = h !== null ? h : this.panelSize.height;
53006         }
53007         if(this.activePanel){
53008             var el = this.activePanel.getEl();
53009             w = w !== null ? w : el.getWidth();
53010             h = h !== null ? h : el.getHeight();
53011             this.panelSize = {width: w, height: h};
53012             this.activePanel.setSize(w, h);
53013         }
53014         if(Roo.isIE && this.tabs){
53015             this.tabs.el.repaint();
53016         }
53017     },
53018
53019     /**
53020      * Returns the container element for this region.
53021      * @return {Roo.Element}
53022      */
53023     getEl : function(){
53024         return this.el;
53025     },
53026
53027     /**
53028      * Hides this region.
53029      */
53030     hide : function(){
53031         if(!this.collapsed){
53032             this.el.dom.style.left = "-2000px";
53033             this.el.hide();
53034         }else{
53035             this.collapsedEl.dom.style.left = "-2000px";
53036             this.collapsedEl.hide();
53037         }
53038         this.visible = false;
53039         this.fireEvent("visibilitychange", this, false);
53040     },
53041
53042     /**
53043      * Shows this region if it was previously hidden.
53044      */
53045     show : function(){
53046         if(!this.collapsed){
53047             this.el.show();
53048         }else{
53049             this.collapsedEl.show();
53050         }
53051         this.visible = true;
53052         this.fireEvent("visibilitychange", this, true);
53053     },
53054
53055     closeClicked : function(){
53056         if(this.activePanel){
53057             this.remove(this.activePanel);
53058         }
53059     },
53060
53061     collapseClick : function(e){
53062         if(this.isSlid){
53063            e.stopPropagation();
53064            this.slideIn();
53065         }else{
53066            e.stopPropagation();
53067            this.slideOut();
53068         }
53069     },
53070
53071     /**
53072      * Collapses this region.
53073      * @param {Boolean} skipAnim (optional) true to collapse the element without animation (if animate is true)
53074      */
53075     collapse : function(skipAnim, skipCheck){
53076         if(this.collapsed) {
53077             return;
53078         }
53079         
53080         if(skipCheck || this.fireEvent("beforecollapse", this) != false){
53081             
53082             this.collapsed = true;
53083             if(this.split){
53084                 this.split.el.hide();
53085             }
53086             if(this.config.animate && skipAnim !== true){
53087                 this.fireEvent("invalidated", this);
53088                 this.animateCollapse();
53089             }else{
53090                 this.el.setLocation(-20000,-20000);
53091                 this.el.hide();
53092                 this.collapsedEl.show();
53093                 this.fireEvent("collapsed", this);
53094                 this.fireEvent("invalidated", this);
53095             }
53096         }
53097         
53098     },
53099
53100     animateCollapse : function(){
53101         // overridden
53102     },
53103
53104     /**
53105      * Expands this region if it was previously collapsed.
53106      * @param {Roo.EventObject} e The event that triggered the expand (or null if calling manually)
53107      * @param {Boolean} skipAnim (optional) true to expand the element without animation (if animate is true)
53108      */
53109     expand : function(e, skipAnim){
53110         if(e) {
53111             e.stopPropagation();
53112         }
53113         if(!this.collapsed || this.el.hasActiveFx()) {
53114             return;
53115         }
53116         if(this.isSlid){
53117             this.afterSlideIn();
53118             skipAnim = true;
53119         }
53120         this.collapsed = false;
53121         if(this.config.animate && skipAnim !== true){
53122             this.animateExpand();
53123         }else{
53124             this.el.show();
53125             if(this.split){
53126                 this.split.el.show();
53127             }
53128             this.collapsedEl.setLocation(-2000,-2000);
53129             this.collapsedEl.hide();
53130             this.fireEvent("invalidated", this);
53131             this.fireEvent("expanded", this);
53132         }
53133     },
53134
53135     animateExpand : function(){
53136         // overridden
53137     },
53138
53139     initTabs : function()
53140     {
53141         this.bodyEl.setStyle("overflow", "hidden");
53142         var ts = new Roo.TabPanel(
53143                 this.bodyEl.dom,
53144                 {
53145                     tabPosition: this.bottomTabs ? 'bottom' : 'top',
53146                     disableTooltips: this.config.disableTabTips,
53147                     toolbar : this.config.toolbar
53148                 }
53149         );
53150         if(this.config.hideTabs){
53151             ts.stripWrap.setDisplayed(false);
53152         }
53153         this.tabs = ts;
53154         ts.resizeTabs = this.config.resizeTabs === true;
53155         ts.minTabWidth = this.config.minTabWidth || 40;
53156         ts.maxTabWidth = this.config.maxTabWidth || 250;
53157         ts.preferredTabWidth = this.config.preferredTabWidth || 150;
53158         ts.monitorResize = false;
53159         ts.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53160         ts.bodyEl.addClass('x-layout-tabs-body');
53161         this.panels.each(this.initPanelAsTab, this);
53162     },
53163
53164     initPanelAsTab : function(panel){
53165         var ti = this.tabs.addTab(panel.getEl().id, panel.getTitle(), null,
53166                     this.config.closeOnTab && panel.isClosable());
53167         if(panel.tabTip !== undefined){
53168             ti.setTooltip(panel.tabTip);
53169         }
53170         ti.on("activate", function(){
53171               this.setActivePanel(panel);
53172         }, this);
53173         if(this.config.closeOnTab){
53174             ti.on("beforeclose", function(t, e){
53175                 e.cancel = true;
53176                 this.remove(panel);
53177             }, this);
53178         }
53179         return ti;
53180     },
53181
53182     updatePanelTitle : function(panel, title){
53183         if(this.activePanel == panel){
53184             this.updateTitle(title);
53185         }
53186         if(this.tabs){
53187             var ti = this.tabs.getTab(panel.getEl().id);
53188             ti.setText(title);
53189             if(panel.tabTip !== undefined){
53190                 ti.setTooltip(panel.tabTip);
53191             }
53192         }
53193     },
53194
53195     updateTitle : function(title){
53196         if(this.titleTextEl && !this.config.title){
53197             this.titleTextEl.innerHTML = (typeof title != "undefined" && title.length > 0 ? title : "&#160;");
53198         }
53199     },
53200
53201     setActivePanel : function(panel){
53202         panel = this.getPanel(panel);
53203         if(this.activePanel && this.activePanel != panel){
53204             this.activePanel.setActiveState(false);
53205         }
53206         this.activePanel = panel;
53207         panel.setActiveState(true);
53208         if(this.panelSize){
53209             panel.setSize(this.panelSize.width, this.panelSize.height);
53210         }
53211         if(this.closeBtn){
53212             this.closeBtn.setVisible(!this.config.closeOnTab && !this.isSlid && panel.isClosable());
53213         }
53214         this.updateTitle(panel.getTitle());
53215         if(this.tabs){
53216             this.fireEvent("invalidated", this);
53217         }
53218         this.fireEvent("panelactivated", this, panel);
53219     },
53220
53221     /**
53222      * Shows the specified panel.
53223      * @param {Number/String/ContentPanel} panelId The panel's index, id or the panel itself
53224      * @return {Roo.ContentPanel} The shown panel, or null if a panel could not be found from panelId
53225      */
53226     showPanel : function(panel)
53227     {
53228         panel = this.getPanel(panel);
53229         if(panel){
53230             if(this.tabs){
53231                 var tab = this.tabs.getTab(panel.getEl().id);
53232                 if(tab.isHidden()){
53233                     this.tabs.unhideTab(tab.id);
53234                 }
53235                 tab.activate();
53236             }else{
53237                 this.setActivePanel(panel);
53238             }
53239         }
53240         return panel;
53241     },
53242
53243     /**
53244      * Get the active panel for this region.
53245      * @return {Roo.ContentPanel} The active panel or null
53246      */
53247     getActivePanel : function(){
53248         return this.activePanel;
53249     },
53250
53251     validateVisibility : function(){
53252         if(this.panels.getCount() < 1){
53253             this.updateTitle("&#160;");
53254             this.closeBtn.hide();
53255             this.hide();
53256         }else{
53257             if(!this.isVisible()){
53258                 this.show();
53259             }
53260         }
53261     },
53262
53263     /**
53264      * Adds the passed ContentPanel(s) to this region.
53265      * @param {ContentPanel...} panel The ContentPanel(s) to add (you can pass more than one)
53266      * @return {Roo.ContentPanel} The panel added (if only one was added; null otherwise)
53267      */
53268     add : function(panel){
53269         if(arguments.length > 1){
53270             for(var i = 0, len = arguments.length; i < len; i++) {
53271                 this.add(arguments[i]);
53272             }
53273             return null;
53274         }
53275         if(this.hasPanel(panel)){
53276             this.showPanel(panel);
53277             return panel;
53278         }
53279         panel.setRegion(this);
53280         this.panels.add(panel);
53281         if(this.panels.getCount() == 1 && !this.config.alwaysShowTabs){
53282             this.bodyEl.dom.appendChild(panel.getEl().dom);
53283             if(panel.background !== true){
53284                 this.setActivePanel(panel);
53285             }
53286             this.fireEvent("paneladded", this, panel);
53287             return panel;
53288         }
53289         if(!this.tabs){
53290             this.initTabs();
53291         }else{
53292             this.initPanelAsTab(panel);
53293         }
53294         if(panel.background !== true){
53295             this.tabs.activate(panel.getEl().id);
53296         }
53297         this.fireEvent("paneladded", this, panel);
53298         return panel;
53299     },
53300
53301     /**
53302      * Hides the tab for the specified panel.
53303      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53304      */
53305     hidePanel : function(panel){
53306         if(this.tabs && (panel = this.getPanel(panel))){
53307             this.tabs.hideTab(panel.getEl().id);
53308         }
53309     },
53310
53311     /**
53312      * Unhides the tab for a previously hidden panel.
53313      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53314      */
53315     unhidePanel : function(panel){
53316         if(this.tabs && (panel = this.getPanel(panel))){
53317             this.tabs.unhideTab(panel.getEl().id);
53318         }
53319     },
53320
53321     clearPanels : function(){
53322         while(this.panels.getCount() > 0){
53323              this.remove(this.panels.first());
53324         }
53325     },
53326
53327     /**
53328      * Removes the specified panel. If preservePanel is not true (either here or in the config), the panel is destroyed.
53329      * @param {Number/String/ContentPanel} panel The panel's index, id or the panel itself
53330      * @param {Boolean} preservePanel Overrides the config preservePanel option
53331      * @return {Roo.ContentPanel} The panel that was removed
53332      */
53333     remove : function(panel, preservePanel){
53334         panel = this.getPanel(panel);
53335         if(!panel){
53336             return null;
53337         }
53338         var e = {};
53339         this.fireEvent("beforeremove", this, panel, e);
53340         if(e.cancel === true){
53341             return null;
53342         }
53343         preservePanel = (typeof preservePanel != "undefined" ? preservePanel : (this.config.preservePanels === true || panel.preserve === true));
53344         var panelId = panel.getId();
53345         this.panels.removeKey(panelId);
53346         if(preservePanel){
53347             document.body.appendChild(panel.getEl().dom);
53348         }
53349         if(this.tabs){
53350             this.tabs.removeTab(panel.getEl().id);
53351         }else if (!preservePanel){
53352             this.bodyEl.dom.removeChild(panel.getEl().dom);
53353         }
53354         if(this.panels.getCount() == 1 && this.tabs && !this.config.alwaysShowTabs){
53355             var p = this.panels.first();
53356             var tempEl = document.createElement("div"); // temp holder to keep IE from deleting the node
53357             tempEl.appendChild(p.getEl().dom);
53358             this.bodyEl.update("");
53359             this.bodyEl.dom.appendChild(p.getEl().dom);
53360             tempEl = null;
53361             this.updateTitle(p.getTitle());
53362             this.tabs = null;
53363             this.bodyEl.setStyle("overflow", this.config.autoScroll ? "auto" : "hidden");
53364             this.setActivePanel(p);
53365         }
53366         panel.setRegion(null);
53367         if(this.activePanel == panel){
53368             this.activePanel = null;
53369         }
53370         if(this.config.autoDestroy !== false && preservePanel !== true){
53371             try{panel.destroy();}catch(e){}
53372         }
53373         this.fireEvent("panelremoved", this, panel);
53374         return panel;
53375     },
53376
53377     /**
53378      * Returns the TabPanel component used by this region
53379      * @return {Roo.TabPanel}
53380      */
53381     getTabs : function(){
53382         return this.tabs;
53383     },
53384
53385     createTool : function(parentEl, className){
53386         var btn = Roo.DomHelper.append(parentEl, {tag: "div", cls: "x-layout-tools-button",
53387             children: [{tag: "div", cls: "x-layout-tools-button-inner " + className, html: "&#160;"}]}, true);
53388         btn.addClassOnOver("x-layout-tools-button-over");
53389         return btn;
53390     }
53391 });/*
53392  * Based on:
53393  * Ext JS Library 1.1.1
53394  * Copyright(c) 2006-2007, Ext JS, LLC.
53395  *
53396  * Originally Released Under LGPL - original licence link has changed is not relivant.
53397  *
53398  * Fork - LGPL
53399  * <script type="text/javascript">
53400  */
53401  
53402
53403
53404 /**
53405  * @class Roo.SplitLayoutRegion
53406  * @extends Roo.LayoutRegion
53407  * Adds a splitbar and other (private) useful functionality to a {@link Roo.LayoutRegion}.
53408  */
53409 Roo.SplitLayoutRegion = function(mgr, config, pos, cursor){
53410     this.cursor = cursor;
53411     Roo.SplitLayoutRegion.superclass.constructor.call(this, mgr, config, pos);
53412 };
53413
53414 Roo.extend(Roo.SplitLayoutRegion, Roo.LayoutRegion, {
53415     splitTip : "Drag to resize.",
53416     collapsibleSplitTip : "Drag to resize. Double click to hide.",
53417     useSplitTips : false,
53418
53419     applyConfig : function(config){
53420         Roo.SplitLayoutRegion.superclass.applyConfig.call(this, config);
53421         if(config.split){
53422             if(!this.split){
53423                 var splitEl = Roo.DomHelper.append(this.mgr.el.dom, 
53424                         {tag: "div", id: this.el.id + "-split", cls: "x-layout-split x-layout-split-"+this.position, html: "&#160;"});
53425                 /** The SplitBar for this region 
53426                 * @type Roo.SplitBar */
53427                 this.split = new Roo.SplitBar(splitEl, this.el, this.orientation);
53428                 this.split.on("moved", this.onSplitMove, this);
53429                 this.split.useShim = config.useShim === true;
53430                 this.split.getMaximumSize = this[this.position == 'north' || this.position == 'south' ? 'getVMaxSize' : 'getHMaxSize'].createDelegate(this);
53431                 if(this.useSplitTips){
53432                     this.split.el.dom.title = config.collapsible ? this.collapsibleSplitTip : this.splitTip;
53433                 }
53434                 if(config.collapsible){
53435                     this.split.el.on("dblclick", this.collapse,  this);
53436                 }
53437             }
53438             if(typeof config.minSize != "undefined"){
53439                 this.split.minSize = config.minSize;
53440             }
53441             if(typeof config.maxSize != "undefined"){
53442                 this.split.maxSize = config.maxSize;
53443             }
53444             if(config.hideWhenEmpty || config.hidden || config.collapsed){
53445                 this.hideSplitter();
53446             }
53447         }
53448     },
53449
53450     getHMaxSize : function(){
53451          var cmax = this.config.maxSize || 10000;
53452          var center = this.mgr.getRegion("center");
53453          return Math.min(cmax, (this.el.getWidth()+center.getEl().getWidth())-center.getMinWidth());
53454     },
53455
53456     getVMaxSize : function(){
53457          var cmax = this.config.maxSize || 10000;
53458          var center = this.mgr.getRegion("center");
53459          return Math.min(cmax, (this.el.getHeight()+center.getEl().getHeight())-center.getMinHeight());
53460     },
53461
53462     onSplitMove : function(split, newSize){
53463         this.fireEvent("resized", this, newSize);
53464     },
53465     
53466     /** 
53467      * Returns the {@link Roo.SplitBar} for this region.
53468      * @return {Roo.SplitBar}
53469      */
53470     getSplitBar : function(){
53471         return this.split;
53472     },
53473     
53474     hide : function(){
53475         this.hideSplitter();
53476         Roo.SplitLayoutRegion.superclass.hide.call(this);
53477     },
53478
53479     hideSplitter : function(){
53480         if(this.split){
53481             this.split.el.setLocation(-2000,-2000);
53482             this.split.el.hide();
53483         }
53484     },
53485
53486     show : function(){
53487         if(this.split){
53488             this.split.el.show();
53489         }
53490         Roo.SplitLayoutRegion.superclass.show.call(this);
53491     },
53492     
53493     beforeSlide: function(){
53494         if(Roo.isGecko){// firefox overflow auto bug workaround
53495             this.bodyEl.clip();
53496             if(this.tabs) {
53497                 this.tabs.bodyEl.clip();
53498             }
53499             if(this.activePanel){
53500                 this.activePanel.getEl().clip();
53501                 
53502                 if(this.activePanel.beforeSlide){
53503                     this.activePanel.beforeSlide();
53504                 }
53505             }
53506         }
53507     },
53508     
53509     afterSlide : function(){
53510         if(Roo.isGecko){// firefox overflow auto bug workaround
53511             this.bodyEl.unclip();
53512             if(this.tabs) {
53513                 this.tabs.bodyEl.unclip();
53514             }
53515             if(this.activePanel){
53516                 this.activePanel.getEl().unclip();
53517                 if(this.activePanel.afterSlide){
53518                     this.activePanel.afterSlide();
53519                 }
53520             }
53521         }
53522     },
53523
53524     initAutoHide : function(){
53525         if(this.autoHide !== false){
53526             if(!this.autoHideHd){
53527                 var st = new Roo.util.DelayedTask(this.slideIn, this);
53528                 this.autoHideHd = {
53529                     "mouseout": function(e){
53530                         if(!e.within(this.el, true)){
53531                             st.delay(500);
53532                         }
53533                     },
53534                     "mouseover" : function(e){
53535                         st.cancel();
53536                     },
53537                     scope : this
53538                 };
53539             }
53540             this.el.on(this.autoHideHd);
53541         }
53542     },
53543
53544     clearAutoHide : function(){
53545         if(this.autoHide !== false){
53546             this.el.un("mouseout", this.autoHideHd.mouseout);
53547             this.el.un("mouseover", this.autoHideHd.mouseover);
53548         }
53549     },
53550
53551     clearMonitor : function(){
53552         Roo.get(document).un("click", this.slideInIf, this);
53553     },
53554
53555     // these names are backwards but not changed for compat
53556     slideOut : function(){
53557         if(this.isSlid || this.el.hasActiveFx()){
53558             return;
53559         }
53560         this.isSlid = true;
53561         if(this.collapseBtn){
53562             this.collapseBtn.hide();
53563         }
53564         this.closeBtnState = this.closeBtn.getStyle('display');
53565         this.closeBtn.hide();
53566         if(this.stickBtn){
53567             this.stickBtn.show();
53568         }
53569         this.el.show();
53570         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor());
53571         this.beforeSlide();
53572         this.el.setStyle("z-index", 10001);
53573         this.el.slideIn(this.getSlideAnchor(), {
53574             callback: function(){
53575                 this.afterSlide();
53576                 this.initAutoHide();
53577                 Roo.get(document).on("click", this.slideInIf, this);
53578                 this.fireEvent("slideshow", this);
53579             },
53580             scope: this,
53581             block: true
53582         });
53583     },
53584
53585     afterSlideIn : function(){
53586         this.clearAutoHide();
53587         this.isSlid = false;
53588         this.clearMonitor();
53589         this.el.setStyle("z-index", "");
53590         if(this.collapseBtn){
53591             this.collapseBtn.show();
53592         }
53593         this.closeBtn.setStyle('display', this.closeBtnState);
53594         if(this.stickBtn){
53595             this.stickBtn.hide();
53596         }
53597         this.fireEvent("slidehide", this);
53598     },
53599
53600     slideIn : function(cb){
53601         if(!this.isSlid || this.el.hasActiveFx()){
53602             Roo.callback(cb);
53603             return;
53604         }
53605         this.isSlid = false;
53606         this.beforeSlide();
53607         this.el.slideOut(this.getSlideAnchor(), {
53608             callback: function(){
53609                 this.el.setLeftTop(-10000, -10000);
53610                 this.afterSlide();
53611                 this.afterSlideIn();
53612                 Roo.callback(cb);
53613             },
53614             scope: this,
53615             block: true
53616         });
53617     },
53618     
53619     slideInIf : function(e){
53620         if(!e.within(this.el)){
53621             this.slideIn();
53622         }
53623     },
53624
53625     animateCollapse : function(){
53626         this.beforeSlide();
53627         this.el.setStyle("z-index", 20000);
53628         var anchor = this.getSlideAnchor();
53629         this.el.slideOut(anchor, {
53630             callback : function(){
53631                 this.el.setStyle("z-index", "");
53632                 this.collapsedEl.slideIn(anchor, {duration:.3});
53633                 this.afterSlide();
53634                 this.el.setLocation(-10000,-10000);
53635                 this.el.hide();
53636                 this.fireEvent("collapsed", this);
53637             },
53638             scope: this,
53639             block: true
53640         });
53641     },
53642
53643     animateExpand : function(){
53644         this.beforeSlide();
53645         this.el.alignTo(this.collapsedEl, this.getCollapseAnchor(), this.getExpandAdj());
53646         this.el.setStyle("z-index", 20000);
53647         this.collapsedEl.hide({
53648             duration:.1
53649         });
53650         this.el.slideIn(this.getSlideAnchor(), {
53651             callback : function(){
53652                 this.el.setStyle("z-index", "");
53653                 this.afterSlide();
53654                 if(this.split){
53655                     this.split.el.show();
53656                 }
53657                 this.fireEvent("invalidated", this);
53658                 this.fireEvent("expanded", this);
53659             },
53660             scope: this,
53661             block: true
53662         });
53663     },
53664
53665     anchors : {
53666         "west" : "left",
53667         "east" : "right",
53668         "north" : "top",
53669         "south" : "bottom"
53670     },
53671
53672     sanchors : {
53673         "west" : "l",
53674         "east" : "r",
53675         "north" : "t",
53676         "south" : "b"
53677     },
53678
53679     canchors : {
53680         "west" : "tl-tr",
53681         "east" : "tr-tl",
53682         "north" : "tl-bl",
53683         "south" : "bl-tl"
53684     },
53685
53686     getAnchor : function(){
53687         return this.anchors[this.position];
53688     },
53689
53690     getCollapseAnchor : function(){
53691         return this.canchors[this.position];
53692     },
53693
53694     getSlideAnchor : function(){
53695         return this.sanchors[this.position];
53696     },
53697
53698     getAlignAdj : function(){
53699         var cm = this.cmargins;
53700         switch(this.position){
53701             case "west":
53702                 return [0, 0];
53703             break;
53704             case "east":
53705                 return [0, 0];
53706             break;
53707             case "north":
53708                 return [0, 0];
53709             break;
53710             case "south":
53711                 return [0, 0];
53712             break;
53713         }
53714     },
53715
53716     getExpandAdj : function(){
53717         var c = this.collapsedEl, cm = this.cmargins;
53718         switch(this.position){
53719             case "west":
53720                 return [-(cm.right+c.getWidth()+cm.left), 0];
53721             break;
53722             case "east":
53723                 return [cm.right+c.getWidth()+cm.left, 0];
53724             break;
53725             case "north":
53726                 return [0, -(cm.top+cm.bottom+c.getHeight())];
53727             break;
53728             case "south":
53729                 return [0, cm.top+cm.bottom+c.getHeight()];
53730             break;
53731         }
53732     }
53733 });/*
53734  * Based on:
53735  * Ext JS Library 1.1.1
53736  * Copyright(c) 2006-2007, Ext JS, LLC.
53737  *
53738  * Originally Released Under LGPL - original licence link has changed is not relivant.
53739  *
53740  * Fork - LGPL
53741  * <script type="text/javascript">
53742  */
53743 /*
53744  * These classes are private internal classes
53745  */
53746 Roo.CenterLayoutRegion = function(mgr, config){
53747     Roo.LayoutRegion.call(this, mgr, config, "center");
53748     this.visible = true;
53749     this.minWidth = config.minWidth || 20;
53750     this.minHeight = config.minHeight || 20;
53751 };
53752
53753 Roo.extend(Roo.CenterLayoutRegion, Roo.LayoutRegion, {
53754     hide : function(){
53755         // center panel can't be hidden
53756     },
53757     
53758     show : function(){
53759         // center panel can't be hidden
53760     },
53761     
53762     getMinWidth: function(){
53763         return this.minWidth;
53764     },
53765     
53766     getMinHeight: function(){
53767         return this.minHeight;
53768     }
53769 });
53770
53771
53772 Roo.NorthLayoutRegion = function(mgr, config){
53773     Roo.LayoutRegion.call(this, mgr, config, "north", "n-resize");
53774     if(this.split){
53775         this.split.placement = Roo.SplitBar.TOP;
53776         this.split.orientation = Roo.SplitBar.VERTICAL;
53777         this.split.el.addClass("x-layout-split-v");
53778     }
53779     var size = config.initialSize || config.height;
53780     if(typeof size != "undefined"){
53781         this.el.setHeight(size);
53782     }
53783 };
53784 Roo.extend(Roo.NorthLayoutRegion, Roo.SplitLayoutRegion, {
53785     orientation: Roo.SplitBar.VERTICAL,
53786     getBox : function(){
53787         if(this.collapsed){
53788             return this.collapsedEl.getBox();
53789         }
53790         var box = this.el.getBox();
53791         if(this.split){
53792             box.height += this.split.el.getHeight();
53793         }
53794         return box;
53795     },
53796     
53797     updateBox : function(box){
53798         if(this.split && !this.collapsed){
53799             box.height -= this.split.el.getHeight();
53800             this.split.el.setLeft(box.x);
53801             this.split.el.setTop(box.y+box.height);
53802             this.split.el.setWidth(box.width);
53803         }
53804         if(this.collapsed){
53805             this.updateBody(box.width, null);
53806         }
53807         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53808     }
53809 });
53810
53811 Roo.SouthLayoutRegion = function(mgr, config){
53812     Roo.SplitLayoutRegion.call(this, mgr, config, "south", "s-resize");
53813     if(this.split){
53814         this.split.placement = Roo.SplitBar.BOTTOM;
53815         this.split.orientation = Roo.SplitBar.VERTICAL;
53816         this.split.el.addClass("x-layout-split-v");
53817     }
53818     var size = config.initialSize || config.height;
53819     if(typeof size != "undefined"){
53820         this.el.setHeight(size);
53821     }
53822 };
53823 Roo.extend(Roo.SouthLayoutRegion, Roo.SplitLayoutRegion, {
53824     orientation: Roo.SplitBar.VERTICAL,
53825     getBox : function(){
53826         if(this.collapsed){
53827             return this.collapsedEl.getBox();
53828         }
53829         var box = this.el.getBox();
53830         if(this.split){
53831             var sh = this.split.el.getHeight();
53832             box.height += sh;
53833             box.y -= sh;
53834         }
53835         return box;
53836     },
53837     
53838     updateBox : function(box){
53839         if(this.split && !this.collapsed){
53840             var sh = this.split.el.getHeight();
53841             box.height -= sh;
53842             box.y += sh;
53843             this.split.el.setLeft(box.x);
53844             this.split.el.setTop(box.y-sh);
53845             this.split.el.setWidth(box.width);
53846         }
53847         if(this.collapsed){
53848             this.updateBody(box.width, null);
53849         }
53850         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53851     }
53852 });
53853
53854 Roo.EastLayoutRegion = function(mgr, config){
53855     Roo.SplitLayoutRegion.call(this, mgr, config, "east", "e-resize");
53856     if(this.split){
53857         this.split.placement = Roo.SplitBar.RIGHT;
53858         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53859         this.split.el.addClass("x-layout-split-h");
53860     }
53861     var size = config.initialSize || config.width;
53862     if(typeof size != "undefined"){
53863         this.el.setWidth(size);
53864     }
53865 };
53866 Roo.extend(Roo.EastLayoutRegion, Roo.SplitLayoutRegion, {
53867     orientation: Roo.SplitBar.HORIZONTAL,
53868     getBox : function(){
53869         if(this.collapsed){
53870             return this.collapsedEl.getBox();
53871         }
53872         var box = this.el.getBox();
53873         if(this.split){
53874             var sw = this.split.el.getWidth();
53875             box.width += sw;
53876             box.x -= sw;
53877         }
53878         return box;
53879     },
53880
53881     updateBox : function(box){
53882         if(this.split && !this.collapsed){
53883             var sw = this.split.el.getWidth();
53884             box.width -= sw;
53885             this.split.el.setLeft(box.x);
53886             this.split.el.setTop(box.y);
53887             this.split.el.setHeight(box.height);
53888             box.x += sw;
53889         }
53890         if(this.collapsed){
53891             this.updateBody(null, box.height);
53892         }
53893         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53894     }
53895 });
53896
53897 Roo.WestLayoutRegion = function(mgr, config){
53898     Roo.SplitLayoutRegion.call(this, mgr, config, "west", "w-resize");
53899     if(this.split){
53900         this.split.placement = Roo.SplitBar.LEFT;
53901         this.split.orientation = Roo.SplitBar.HORIZONTAL;
53902         this.split.el.addClass("x-layout-split-h");
53903     }
53904     var size = config.initialSize || config.width;
53905     if(typeof size != "undefined"){
53906         this.el.setWidth(size);
53907     }
53908 };
53909 Roo.extend(Roo.WestLayoutRegion, Roo.SplitLayoutRegion, {
53910     orientation: Roo.SplitBar.HORIZONTAL,
53911     getBox : function(){
53912         if(this.collapsed){
53913             return this.collapsedEl.getBox();
53914         }
53915         var box = this.el.getBox();
53916         if(this.split){
53917             box.width += this.split.el.getWidth();
53918         }
53919         return box;
53920     },
53921     
53922     updateBox : function(box){
53923         if(this.split && !this.collapsed){
53924             var sw = this.split.el.getWidth();
53925             box.width -= sw;
53926             this.split.el.setLeft(box.x+box.width);
53927             this.split.el.setTop(box.y);
53928             this.split.el.setHeight(box.height);
53929         }
53930         if(this.collapsed){
53931             this.updateBody(null, box.height);
53932         }
53933         Roo.LayoutRegion.prototype.updateBox.call(this, box);
53934     }
53935 });
53936 /*
53937  * Based on:
53938  * Ext JS Library 1.1.1
53939  * Copyright(c) 2006-2007, Ext JS, LLC.
53940  *
53941  * Originally Released Under LGPL - original licence link has changed is not relivant.
53942  *
53943  * Fork - LGPL
53944  * <script type="text/javascript">
53945  */
53946  
53947  
53948 /*
53949  * Private internal class for reading and applying state
53950  */
53951 Roo.LayoutStateManager = function(layout){
53952      // default empty state
53953      this.state = {
53954         north: {},
53955         south: {},
53956         east: {},
53957         west: {}       
53958     };
53959 };
53960
53961 Roo.LayoutStateManager.prototype = {
53962     init : function(layout, provider){
53963         this.provider = provider;
53964         var state = provider.get(layout.id+"-layout-state");
53965         if(state){
53966             var wasUpdating = layout.isUpdating();
53967             if(!wasUpdating){
53968                 layout.beginUpdate();
53969             }
53970             for(var key in state){
53971                 if(typeof state[key] != "function"){
53972                     var rstate = state[key];
53973                     var r = layout.getRegion(key);
53974                     if(r && rstate){
53975                         if(rstate.size){
53976                             r.resizeTo(rstate.size);
53977                         }
53978                         if(rstate.collapsed == true){
53979                             r.collapse(true);
53980                         }else{
53981                             r.expand(null, true);
53982                         }
53983                     }
53984                 }
53985             }
53986             if(!wasUpdating){
53987                 layout.endUpdate();
53988             }
53989             this.state = state; 
53990         }
53991         this.layout = layout;
53992         layout.on("regionresized", this.onRegionResized, this);
53993         layout.on("regioncollapsed", this.onRegionCollapsed, this);
53994         layout.on("regionexpanded", this.onRegionExpanded, this);
53995     },
53996     
53997     storeState : function(){
53998         this.provider.set(this.layout.id+"-layout-state", this.state);
53999     },
54000     
54001     onRegionResized : function(region, newSize){
54002         this.state[region.getPosition()].size = newSize;
54003         this.storeState();
54004     },
54005     
54006     onRegionCollapsed : function(region){
54007         this.state[region.getPosition()].collapsed = true;
54008         this.storeState();
54009     },
54010     
54011     onRegionExpanded : function(region){
54012         this.state[region.getPosition()].collapsed = false;
54013         this.storeState();
54014     }
54015 };/*
54016  * Based on:
54017  * Ext JS Library 1.1.1
54018  * Copyright(c) 2006-2007, Ext JS, LLC.
54019  *
54020  * Originally Released Under LGPL - original licence link has changed is not relivant.
54021  *
54022  * Fork - LGPL
54023  * <script type="text/javascript">
54024  */
54025 /**
54026  * @class Roo.ContentPanel
54027  * @extends Roo.util.Observable
54028  * A basic ContentPanel element.
54029  * @cfg {Boolean}   fitToFrame    True for this panel to adjust its size to fit when the region resizes  (defaults to false)
54030  * @cfg {Boolean}   fitContainer   When using {@link #fitToFrame} and {@link #resizeEl}, you can also fit the parent container  (defaults to false)
54031  * @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
54032  * @cfg {Boolean}   closable      True if the panel can be closed/removed
54033  * @cfg {Boolean}   background    True if the panel should not be activated when it is added (defaults to false)
54034  * @cfg {String/HTMLElement/Element} resizeEl An element to resize if {@link #fitToFrame} is true (instead of this panel's element)
54035  * @cfg {Toolbar}   toolbar       A toolbar for this panel
54036  * @cfg {Boolean} autoScroll    True to scroll overflow in this panel (use with {@link #fitToFrame})
54037  * @cfg {String} title          The title for this panel
54038  * @cfg {Array} adjustments     Values to <b>add</b> to the width/height when doing a {@link #fitToFrame} (default is [0, 0])
54039  * @cfg {String} url            Calls {@link #setUrl} with this value
54040  * @cfg {String} region         (center|north|south|east|west) which region to put this panel on (when used with xtype constructors)
54041  * @cfg {String/Object} params  When used with {@link #url}, calls {@link #setUrl} with this value
54042  * @cfg {Boolean} loadOnce      When used with {@link #url}, calls {@link #setUrl} with this value
54043  * @cfg {String}    content        Raw content to fill content panel with (uses setContent on construction.)
54044
54045  * @constructor
54046  * Create a new ContentPanel.
54047  * @param {String/HTMLElement/Roo.Element} el The container element for this panel
54048  * @param {String/Object} config A string to set only the title or a config object
54049  * @param {String} content (optional) Set the HTML content for this panel
54050  * @param {String} region (optional) Used by xtype constructors to add to regions. (values center,east,west,south,north)
54051  */
54052 Roo.ContentPanel = function(el, config, content){
54053     
54054      
54055     /*
54056     if(el.autoCreate || el.xtype){ // xtype is available if this is called from factory
54057         config = el;
54058         el = Roo.id();
54059     }
54060     if (config && config.parentLayout) { 
54061         el = config.parentLayout.el.createChild(); 
54062     }
54063     */
54064     if(el.autoCreate){ // xtype is available if this is called from factory
54065         config = el;
54066         el = Roo.id();
54067     }
54068     this.el = Roo.get(el);
54069     if(!this.el && config && config.autoCreate){
54070         if(typeof config.autoCreate == "object"){
54071             if(!config.autoCreate.id){
54072                 config.autoCreate.id = config.id||el;
54073             }
54074             this.el = Roo.DomHelper.append(document.body,
54075                         config.autoCreate, true);
54076         }else{
54077             this.el = Roo.DomHelper.append(document.body,
54078                         {tag: "div", cls: "x-layout-inactive-content", id: config.id||el}, true);
54079         }
54080     }
54081     this.closable = false;
54082     this.loaded = false;
54083     this.active = false;
54084     if(typeof config == "string"){
54085         this.title = config;
54086     }else{
54087         Roo.apply(this, config);
54088     }
54089     
54090     if (this.toolbar && !this.toolbar.el && this.toolbar.xtype) {
54091         this.wrapEl = this.el.wrap();
54092         this.toolbar.container = this.el.insertSibling(false, 'before');
54093         this.toolbar = new Roo.Toolbar(this.toolbar);
54094     }
54095     
54096     // xtype created footer. - not sure if will work as we normally have to render first..
54097     if (this.footer && !this.footer.el && this.footer.xtype) {
54098         if (!this.wrapEl) {
54099             this.wrapEl = this.el.wrap();
54100         }
54101     
54102         this.footer.container = this.wrapEl.createChild();
54103          
54104         this.footer = Roo.factory(this.footer, Roo);
54105         
54106     }
54107     
54108     if(this.resizeEl){
54109         this.resizeEl = Roo.get(this.resizeEl, true);
54110     }else{
54111         this.resizeEl = this.el;
54112     }
54113     // handle view.xtype
54114     
54115  
54116     
54117     
54118     this.addEvents({
54119         /**
54120          * @event activate
54121          * Fires when this panel is activated. 
54122          * @param {Roo.ContentPanel} this
54123          */
54124         "activate" : true,
54125         /**
54126          * @event deactivate
54127          * Fires when this panel is activated. 
54128          * @param {Roo.ContentPanel} this
54129          */
54130         "deactivate" : true,
54131
54132         /**
54133          * @event resize
54134          * Fires when this panel is resized if fitToFrame is true.
54135          * @param {Roo.ContentPanel} this
54136          * @param {Number} width The width after any component adjustments
54137          * @param {Number} height The height after any component adjustments
54138          */
54139         "resize" : true,
54140         
54141          /**
54142          * @event render
54143          * Fires when this tab is created
54144          * @param {Roo.ContentPanel} this
54145          */
54146         "render" : true
54147          
54148         
54149     });
54150     
54151
54152     
54153     
54154     if(this.autoScroll){
54155         this.resizeEl.setStyle("overflow", "auto");
54156     } else {
54157         // fix randome scrolling
54158         this.el.on('scroll', function() {
54159             Roo.log('fix random scolling');
54160             this.scrollTo('top',0); 
54161         });
54162     }
54163     content = content || this.content;
54164     if(content){
54165         this.setContent(content);
54166     }
54167     if(config && config.url){
54168         this.setUrl(this.url, this.params, this.loadOnce);
54169     }
54170     
54171     
54172     
54173     Roo.ContentPanel.superclass.constructor.call(this);
54174     
54175     if (this.view && typeof(this.view.xtype) != 'undefined') {
54176         this.view.el = this.el.appendChild(document.createElement("div"));
54177         this.view = Roo.factory(this.view); 
54178         this.view.render  &&  this.view.render(false, '');  
54179     }
54180     
54181     
54182     this.fireEvent('render', this);
54183 };
54184
54185 Roo.extend(Roo.ContentPanel, Roo.util.Observable, {
54186     tabTip:'',
54187     setRegion : function(region){
54188         this.region = region;
54189         if(region){
54190            this.el.replaceClass("x-layout-inactive-content", "x-layout-active-content");
54191         }else{
54192            this.el.replaceClass("x-layout-active-content", "x-layout-inactive-content");
54193         } 
54194     },
54195     
54196     /**
54197      * Returns the toolbar for this Panel if one was configured. 
54198      * @return {Roo.Toolbar} 
54199      */
54200     getToolbar : function(){
54201         return this.toolbar;
54202     },
54203     
54204     setActiveState : function(active){
54205         this.active = active;
54206         if(!active){
54207             this.fireEvent("deactivate", this);
54208         }else{
54209             this.fireEvent("activate", this);
54210         }
54211     },
54212     /**
54213      * Updates this panel's element
54214      * @param {String} content The new content
54215      * @param {Boolean} loadScripts (optional) true to look for and process scripts
54216     */
54217     setContent : function(content, loadScripts){
54218         this.el.update(content, loadScripts);
54219     },
54220
54221     ignoreResize : function(w, h){
54222         if(this.lastSize && this.lastSize.width == w && this.lastSize.height == h){
54223             return true;
54224         }else{
54225             this.lastSize = {width: w, height: h};
54226             return false;
54227         }
54228     },
54229     /**
54230      * Get the {@link Roo.UpdateManager} for this panel. Enables you to perform Ajax updates.
54231      * @return {Roo.UpdateManager} The UpdateManager
54232      */
54233     getUpdateManager : function(){
54234         return this.el.getUpdateManager();
54235     },
54236      /**
54237      * Loads this content panel immediately with content from XHR. Note: to delay loading until the panel is activated, use {@link #setUrl}.
54238      * @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:
54239 <pre><code>
54240 panel.load({
54241     url: "your-url.php",
54242     params: {param1: "foo", param2: "bar"}, // or a URL encoded string
54243     callback: yourFunction,
54244     scope: yourObject, //(optional scope)
54245     discardUrl: false,
54246     nocache: false,
54247     text: "Loading...",
54248     timeout: 30,
54249     scripts: false
54250 });
54251 </code></pre>
54252      * The only required property is <i>url</i>. The optional properties <i>nocache</i>, <i>text</i> and <i>scripts</i>
54253      * 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.
54254      * @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}
54255      * @param {Function} callback (optional) Callback when transaction is complete -- called with signature (oElement, bSuccess, oResponse)
54256      * @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.
54257      * @return {Roo.ContentPanel} this
54258      */
54259     load : function(){
54260         var um = this.el.getUpdateManager();
54261         um.update.apply(um, arguments);
54262         return this;
54263     },
54264
54265
54266     /**
54267      * 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.
54268      * @param {String/Function} url The URL to load the content from or a function to call to get the URL
54269      * @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)
54270      * @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)
54271      * @return {Roo.UpdateManager} The UpdateManager
54272      */
54273     setUrl : function(url, params, loadOnce){
54274         if(this.refreshDelegate){
54275             this.removeListener("activate", this.refreshDelegate);
54276         }
54277         this.refreshDelegate = this._handleRefresh.createDelegate(this, [url, params, loadOnce]);
54278         this.on("activate", this.refreshDelegate);
54279         return this.el.getUpdateManager();
54280     },
54281     
54282     _handleRefresh : function(url, params, loadOnce){
54283         if(!loadOnce || !this.loaded){
54284             var updater = this.el.getUpdateManager();
54285             updater.update(url, params, this._setLoaded.createDelegate(this));
54286         }
54287     },
54288     
54289     _setLoaded : function(){
54290         this.loaded = true;
54291     }, 
54292     
54293     /**
54294      * Returns this panel's id
54295      * @return {String} 
54296      */
54297     getId : function(){
54298         return this.el.id;
54299     },
54300     
54301     /** 
54302      * Returns this panel's element - used by regiosn to add.
54303      * @return {Roo.Element} 
54304      */
54305     getEl : function(){
54306         return this.wrapEl || this.el;
54307     },
54308     
54309     adjustForComponents : function(width, height)
54310     {
54311         //Roo.log('adjustForComponents ');
54312         if(this.resizeEl != this.el){
54313             width -= this.el.getFrameWidth('lr');
54314             height -= this.el.getFrameWidth('tb');
54315         }
54316         if(this.toolbar){
54317             var te = this.toolbar.getEl();
54318             height -= te.getHeight();
54319             te.setWidth(width);
54320         }
54321         if(this.footer){
54322             var te = this.footer.getEl();
54323             //Roo.log("footer:" + te.getHeight());
54324             
54325             height -= te.getHeight();
54326             te.setWidth(width);
54327         }
54328         
54329         
54330         if(this.adjustments){
54331             width += this.adjustments[0];
54332             height += this.adjustments[1];
54333         }
54334         return {"width": width, "height": height};
54335     },
54336     
54337     setSize : function(width, height){
54338         if(this.fitToFrame && !this.ignoreResize(width, height)){
54339             if(this.fitContainer && this.resizeEl != this.el){
54340                 this.el.setSize(width, height);
54341             }
54342             var size = this.adjustForComponents(width, height);
54343             this.resizeEl.setSize(this.autoWidth ? "auto" : size.width, this.autoHeight ? "auto" : size.height);
54344             this.fireEvent('resize', this, size.width, size.height);
54345         }
54346     },
54347     
54348     /**
54349      * Returns this panel's title
54350      * @return {String} 
54351      */
54352     getTitle : function(){
54353         return this.title;
54354     },
54355     
54356     /**
54357      * Set this panel's title
54358      * @param {String} title
54359      */
54360     setTitle : function(title){
54361         this.title = title;
54362         if(this.region){
54363             this.region.updatePanelTitle(this, title);
54364         }
54365     },
54366     
54367     /**
54368      * Returns true is this panel was configured to be closable
54369      * @return {Boolean} 
54370      */
54371     isClosable : function(){
54372         return this.closable;
54373     },
54374     
54375     beforeSlide : function(){
54376         this.el.clip();
54377         this.resizeEl.clip();
54378     },
54379     
54380     afterSlide : function(){
54381         this.el.unclip();
54382         this.resizeEl.unclip();
54383     },
54384     
54385     /**
54386      *   Force a content refresh from the URL specified in the {@link #setUrl} method.
54387      *   Will fail silently if the {@link #setUrl} method has not been called.
54388      *   This does not activate the panel, just updates its content.
54389      */
54390     refresh : function(){
54391         if(this.refreshDelegate){
54392            this.loaded = false;
54393            this.refreshDelegate();
54394         }
54395     },
54396     
54397     /**
54398      * Destroys this panel
54399      */
54400     destroy : function(){
54401         this.el.removeAllListeners();
54402         var tempEl = document.createElement("span");
54403         tempEl.appendChild(this.el.dom);
54404         tempEl.innerHTML = "";
54405         this.el.remove();
54406         this.el = null;
54407     },
54408     
54409     /**
54410      * form - if the content panel contains a form - this is a reference to it.
54411      * @type {Roo.form.Form}
54412      */
54413     form : false,
54414     /**
54415      * view - if the content panel contains a view (Roo.DatePicker / Roo.View / Roo.JsonView)
54416      *    This contains a reference to it.
54417      * @type {Roo.View}
54418      */
54419     view : false,
54420     
54421       /**
54422      * Adds a xtype elements to the panel - currently only supports Forms, View, JsonView.
54423      * <pre><code>
54424
54425 layout.addxtype({
54426        xtype : 'Form',
54427        items: [ .... ]
54428    }
54429 );
54430
54431 </code></pre>
54432      * @param {Object} cfg Xtype definition of item to add.
54433      */
54434     
54435     addxtype : function(cfg) {
54436         // add form..
54437         if (cfg.xtype.match(/^Form$/)) {
54438             
54439             var el;
54440             //if (this.footer) {
54441             //    el = this.footer.container.insertSibling(false, 'before');
54442             //} else {
54443                 el = this.el.createChild();
54444             //}
54445
54446             this.form = new  Roo.form.Form(cfg);
54447             
54448             
54449             if ( this.form.allItems.length) {
54450                 this.form.render(el.dom);
54451             }
54452             return this.form;
54453         }
54454         // should only have one of theses..
54455         if ([ 'View', 'JsonView', 'DatePicker'].indexOf(cfg.xtype) > -1) {
54456             // views.. should not be just added - used named prop 'view''
54457             
54458             cfg.el = this.el.appendChild(document.createElement("div"));
54459             // factory?
54460             
54461             var ret = new Roo.factory(cfg);
54462              
54463              ret.render && ret.render(false, ''); // render blank..
54464             this.view = ret;
54465             return ret;
54466         }
54467         return false;
54468     }
54469 });
54470
54471 /**
54472  * @class Roo.GridPanel
54473  * @extends Roo.ContentPanel
54474  * @constructor
54475  * Create a new GridPanel.
54476  * @param {Roo.grid.Grid} grid The grid for this panel
54477  * @param {String/Object} config A string to set only the panel's title, or a config object
54478  */
54479 Roo.GridPanel = function(grid, config){
54480     
54481   
54482     this.wrapper = Roo.DomHelper.append(document.body, // wrapper for IE7 strict & safari scroll issue
54483         {tag: "div", cls: "x-layout-grid-wrapper x-layout-inactive-content"}, true);
54484         
54485     this.wrapper.dom.appendChild(grid.getGridEl().dom);
54486     
54487     Roo.GridPanel.superclass.constructor.call(this, this.wrapper, config);
54488     
54489     if(this.toolbar){
54490         this.toolbar.el.insertBefore(this.wrapper.dom.firstChild);
54491     }
54492     // xtype created footer. - not sure if will work as we normally have to render first..
54493     if (this.footer && !this.footer.el && this.footer.xtype) {
54494         
54495         this.footer.container = this.grid.getView().getFooterPanel(true);
54496         this.footer.dataSource = this.grid.dataSource;
54497         this.footer = Roo.factory(this.footer, Roo);
54498         
54499     }
54500     
54501     grid.monitorWindowResize = false; // turn off autosizing
54502     grid.autoHeight = false;
54503     grid.autoWidth = false;
54504     this.grid = grid;
54505     this.grid.getGridEl().replaceClass("x-layout-inactive-content", "x-layout-component-panel");
54506 };
54507
54508 Roo.extend(Roo.GridPanel, Roo.ContentPanel, {
54509     getId : function(){
54510         return this.grid.id;
54511     },
54512     
54513     /**
54514      * Returns the grid for this panel
54515      * @return {Roo.grid.Grid} 
54516      */
54517     getGrid : function(){
54518         return this.grid;    
54519     },
54520     
54521     setSize : function(width, height){
54522         if(!this.ignoreResize(width, height)){
54523             var grid = this.grid;
54524             var size = this.adjustForComponents(width, height);
54525             grid.getGridEl().setSize(size.width, size.height);
54526             grid.autoSize();
54527         }
54528     },
54529     
54530     beforeSlide : function(){
54531         this.grid.getView().scroller.clip();
54532     },
54533     
54534     afterSlide : function(){
54535         this.grid.getView().scroller.unclip();
54536     },
54537     
54538     destroy : function(){
54539         this.grid.destroy();
54540         delete this.grid;
54541         Roo.GridPanel.superclass.destroy.call(this); 
54542     }
54543 });
54544
54545
54546 /**
54547  * @class Roo.NestedLayoutPanel
54548  * @extends Roo.ContentPanel
54549  * @constructor
54550  * Create a new NestedLayoutPanel.
54551  * 
54552  * 
54553  * @param {Roo.BorderLayout} layout The layout for this panel
54554  * @param {String/Object} config A string to set only the title or a config object
54555  */
54556 Roo.NestedLayoutPanel = function(layout, config)
54557 {
54558     // construct with only one argument..
54559     /* FIXME - implement nicer consturctors
54560     if (layout.layout) {
54561         config = layout;
54562         layout = config.layout;
54563         delete config.layout;
54564     }
54565     if (layout.xtype && !layout.getEl) {
54566         // then layout needs constructing..
54567         layout = Roo.factory(layout, Roo);
54568     }
54569     */
54570     
54571     
54572     Roo.NestedLayoutPanel.superclass.constructor.call(this, layout.getEl(), config);
54573     
54574     layout.monitorWindowResize = false; // turn off autosizing
54575     this.layout = layout;
54576     this.layout.getEl().addClass("x-layout-nested-layout");
54577     
54578     
54579     
54580     
54581 };
54582
54583 Roo.extend(Roo.NestedLayoutPanel, Roo.ContentPanel, {
54584
54585     setSize : function(width, height){
54586         if(!this.ignoreResize(width, height)){
54587             var size = this.adjustForComponents(width, height);
54588             var el = this.layout.getEl();
54589             el.setSize(size.width, size.height);
54590             var touch = el.dom.offsetWidth;
54591             this.layout.layout();
54592             // ie requires a double layout on the first pass
54593             if(Roo.isIE && !this.initialized){
54594                 this.initialized = true;
54595                 this.layout.layout();
54596             }
54597         }
54598     },
54599     
54600     // activate all subpanels if not currently active..
54601     
54602     setActiveState : function(active){
54603         this.active = active;
54604         if(!active){
54605             this.fireEvent("deactivate", this);
54606             return;
54607         }
54608         
54609         this.fireEvent("activate", this);
54610         // not sure if this should happen before or after..
54611         if (!this.layout) {
54612             return; // should not happen..
54613         }
54614         var reg = false;
54615         for (var r in this.layout.regions) {
54616             reg = this.layout.getRegion(r);
54617             if (reg.getActivePanel()) {
54618                 //reg.showPanel(reg.getActivePanel()); // force it to activate.. 
54619                 reg.setActivePanel(reg.getActivePanel());
54620                 continue;
54621             }
54622             if (!reg.panels.length) {
54623                 continue;
54624             }
54625             reg.showPanel(reg.getPanel(0));
54626         }
54627         
54628         
54629         
54630         
54631     },
54632     
54633     /**
54634      * Returns the nested BorderLayout for this panel
54635      * @return {Roo.BorderLayout} 
54636      */
54637     getLayout : function(){
54638         return this.layout;
54639     },
54640     
54641      /**
54642      * Adds a xtype elements to the layout of the nested panel
54643      * <pre><code>
54644
54645 panel.addxtype({
54646        xtype : 'ContentPanel',
54647        region: 'west',
54648        items: [ .... ]
54649    }
54650 );
54651
54652 panel.addxtype({
54653         xtype : 'NestedLayoutPanel',
54654         region: 'west',
54655         layout: {
54656            center: { },
54657            west: { }   
54658         },
54659         items : [ ... list of content panels or nested layout panels.. ]
54660    }
54661 );
54662 </code></pre>
54663      * @param {Object} cfg Xtype definition of item to add.
54664      */
54665     addxtype : function(cfg) {
54666         return this.layout.addxtype(cfg);
54667     
54668     }
54669 });
54670
54671 Roo.ScrollPanel = function(el, config, content){
54672     config = config || {};
54673     config.fitToFrame = true;
54674     Roo.ScrollPanel.superclass.constructor.call(this, el, config, content);
54675     
54676     this.el.dom.style.overflow = "hidden";
54677     var wrap = this.el.wrap({cls: "x-scroller x-layout-inactive-content"});
54678     this.el.removeClass("x-layout-inactive-content");
54679     this.el.on("mousewheel", this.onWheel, this);
54680
54681     var up = wrap.createChild({cls: "x-scroller-up", html: "&#160;"}, this.el.dom);
54682     var down = wrap.createChild({cls: "x-scroller-down", html: "&#160;"});
54683     up.unselectable(); down.unselectable();
54684     up.on("click", this.scrollUp, this);
54685     down.on("click", this.scrollDown, this);
54686     up.addClassOnOver("x-scroller-btn-over");
54687     down.addClassOnOver("x-scroller-btn-over");
54688     up.addClassOnClick("x-scroller-btn-click");
54689     down.addClassOnClick("x-scroller-btn-click");
54690     this.adjustments = [0, -(up.getHeight() + down.getHeight())];
54691
54692     this.resizeEl = this.el;
54693     this.el = wrap; this.up = up; this.down = down;
54694 };
54695
54696 Roo.extend(Roo.ScrollPanel, Roo.ContentPanel, {
54697     increment : 100,
54698     wheelIncrement : 5,
54699     scrollUp : function(){
54700         this.resizeEl.scroll("up", this.increment, {callback: this.afterScroll, scope: this});
54701     },
54702
54703     scrollDown : function(){
54704         this.resizeEl.scroll("down", this.increment, {callback: this.afterScroll, scope: this});
54705     },
54706
54707     afterScroll : function(){
54708         var el = this.resizeEl;
54709         var t = el.dom.scrollTop, h = el.dom.scrollHeight, ch = el.dom.clientHeight;
54710         this.up[t == 0 ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54711         this.down[h - t <= ch ? "addClass" : "removeClass"]("x-scroller-btn-disabled");
54712     },
54713
54714     setSize : function(){
54715         Roo.ScrollPanel.superclass.setSize.apply(this, arguments);
54716         this.afterScroll();
54717     },
54718
54719     onWheel : function(e){
54720         var d = e.getWheelDelta();
54721         this.resizeEl.dom.scrollTop -= (d*this.wheelIncrement);
54722         this.afterScroll();
54723         e.stopEvent();
54724     },
54725
54726     setContent : function(content, loadScripts){
54727         this.resizeEl.update(content, loadScripts);
54728     }
54729
54730 });
54731
54732
54733
54734
54735
54736
54737
54738
54739
54740 /**
54741  * @class Roo.TreePanel
54742  * @extends Roo.ContentPanel
54743  * @constructor
54744  * Create a new TreePanel. - defaults to fit/scoll contents.
54745  * @param {String/Object} config A string to set only the panel's title, or a config object
54746  * @cfg {Roo.tree.TreePanel} tree The tree TreePanel, with config etc.
54747  */
54748 Roo.TreePanel = function(config){
54749     var el = config.el;
54750     var tree = config.tree;
54751     delete config.tree; 
54752     delete config.el; // hopefull!
54753     
54754     // wrapper for IE7 strict & safari scroll issue
54755     
54756     var treeEl = el.createChild();
54757     config.resizeEl = treeEl;
54758     
54759     
54760     
54761     Roo.TreePanel.superclass.constructor.call(this, el, config);
54762  
54763  
54764     this.tree = new Roo.tree.TreePanel(treeEl , tree);
54765     //console.log(tree);
54766     this.on('activate', function()
54767     {
54768         if (this.tree.rendered) {
54769             return;
54770         }
54771         //console.log('render tree');
54772         this.tree.render();
54773     });
54774     // this should not be needed.. - it's actually the 'el' that resizes?
54775     // actuall it breaks the containerScroll - dragging nodes auto scroll at top
54776     
54777     //this.on('resize',  function (cp, w, h) {
54778     //        this.tree.innerCt.setWidth(w);
54779     //        this.tree.innerCt.setHeight(h);
54780     //        //this.tree.innerCt.setStyle('overflow-y', 'auto');
54781     //});
54782
54783         
54784     
54785 };
54786
54787 Roo.extend(Roo.TreePanel, Roo.ContentPanel, {   
54788     fitToFrame : true,
54789     autoScroll : true
54790 });
54791
54792
54793
54794
54795
54796
54797
54798
54799
54800
54801
54802 /*
54803  * Based on:
54804  * Ext JS Library 1.1.1
54805  * Copyright(c) 2006-2007, Ext JS, LLC.
54806  *
54807  * Originally Released Under LGPL - original licence link has changed is not relivant.
54808  *
54809  * Fork - LGPL
54810  * <script type="text/javascript">
54811  */
54812  
54813
54814 /**
54815  * @class Roo.ReaderLayout
54816  * @extends Roo.BorderLayout
54817  * This is a pre-built layout that represents a classic, 5-pane application.  It consists of a header, a primary
54818  * center region containing two nested regions (a top one for a list view and one for item preview below),
54819  * and regions on either side that can be used for navigation, application commands, informational displays, etc.
54820  * The setup and configuration work exactly the same as it does for a {@link Roo.BorderLayout} - this class simply
54821  * expedites the setup of the overall layout and regions for this common application style.
54822  * Example:
54823  <pre><code>
54824 var reader = new Roo.ReaderLayout();
54825 var CP = Roo.ContentPanel;  // shortcut for adding
54826
54827 reader.beginUpdate();
54828 reader.add("north", new CP("north", "North"));
54829 reader.add("west", new CP("west", {title: "West"}));
54830 reader.add("east", new CP("east", {title: "East"}));
54831
54832 reader.regions.listView.add(new CP("listView", "List"));
54833 reader.regions.preview.add(new CP("preview", "Preview"));
54834 reader.endUpdate();
54835 </code></pre>
54836 * @constructor
54837 * Create a new ReaderLayout
54838 * @param {Object} config Configuration options
54839 * @param {String/HTMLElement/Element} container (optional) The container this layout is bound to (defaults to
54840 * document.body if omitted)
54841 */
54842 Roo.ReaderLayout = function(config, renderTo){
54843     var c = config || {size:{}};
54844     Roo.ReaderLayout.superclass.constructor.call(this, renderTo || document.body, {
54845         north: c.north !== false ? Roo.apply({
54846             split:false,
54847             initialSize: 32,
54848             titlebar: false
54849         }, c.north) : false,
54850         west: c.west !== false ? Roo.apply({
54851             split:true,
54852             initialSize: 200,
54853             minSize: 175,
54854             maxSize: 400,
54855             titlebar: true,
54856             collapsible: true,
54857             animate: true,
54858             margins:{left:5,right:0,bottom:5,top:5},
54859             cmargins:{left:5,right:5,bottom:5,top:5}
54860         }, c.west) : false,
54861         east: c.east !== false ? Roo.apply({
54862             split:true,
54863             initialSize: 200,
54864             minSize: 175,
54865             maxSize: 400,
54866             titlebar: true,
54867             collapsible: true,
54868             animate: true,
54869             margins:{left:0,right:5,bottom:5,top:5},
54870             cmargins:{left:5,right:5,bottom:5,top:5}
54871         }, c.east) : false,
54872         center: Roo.apply({
54873             tabPosition: 'top',
54874             autoScroll:false,
54875             closeOnTab: true,
54876             titlebar:false,
54877             margins:{left:c.west!==false ? 0 : 5,right:c.east!==false ? 0 : 5,bottom:5,top:2}
54878         }, c.center)
54879     });
54880
54881     this.el.addClass('x-reader');
54882
54883     this.beginUpdate();
54884
54885     var inner = new Roo.BorderLayout(Roo.get(document.body).createChild(), {
54886         south: c.preview !== false ? Roo.apply({
54887             split:true,
54888             initialSize: 200,
54889             minSize: 100,
54890             autoScroll:true,
54891             collapsible:true,
54892             titlebar: true,
54893             cmargins:{top:5,left:0, right:0, bottom:0}
54894         }, c.preview) : false,
54895         center: Roo.apply({
54896             autoScroll:false,
54897             titlebar:false,
54898             minHeight:200
54899         }, c.listView)
54900     });
54901     this.add('center', new Roo.NestedLayoutPanel(inner,
54902             Roo.apply({title: c.mainTitle || '',tabTip:''},c.innerPanelCfg)));
54903
54904     this.endUpdate();
54905
54906     this.regions.preview = inner.getRegion('south');
54907     this.regions.listView = inner.getRegion('center');
54908 };
54909
54910 Roo.extend(Roo.ReaderLayout, Roo.BorderLayout);/*
54911  * Based on:
54912  * Ext JS Library 1.1.1
54913  * Copyright(c) 2006-2007, Ext JS, LLC.
54914  *
54915  * Originally Released Under LGPL - original licence link has changed is not relivant.
54916  *
54917  * Fork - LGPL
54918  * <script type="text/javascript">
54919  */
54920  
54921 /**
54922  * @class Roo.grid.Grid
54923  * @extends Roo.util.Observable
54924  * This class represents the primary interface of a component based grid control.
54925  * <br><br>Usage:<pre><code>
54926  var grid = new Roo.grid.Grid("my-container-id", {
54927      ds: myDataStore,
54928      cm: myColModel,
54929      selModel: mySelectionModel,
54930      autoSizeColumns: true,
54931      monitorWindowResize: false,
54932      trackMouseOver: true
54933  });
54934  // set any options
54935  grid.render();
54936  * </code></pre>
54937  * <b>Common Problems:</b><br/>
54938  * - Grid does not resize properly when going smaller: Setting overflow hidden on the container
54939  * element will correct this<br/>
54940  * - If you get el.style[camel]= NaNpx or -2px or something related, be certain you have given your container element
54941  * dimensions. The grid adapts to your container's size, if your container has no size defined then the results
54942  * are unpredictable.<br/>
54943  * - Do not render the grid into an element with display:none. Try using visibility:hidden. Otherwise there is no way for the
54944  * grid to calculate dimensions/offsets.<br/>
54945   * @constructor
54946  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
54947  * The container MUST have some type of size defined for the grid to fill. The container will be
54948  * automatically set to position relative if it isn't already.
54949  * @param {Object} config A config object that sets properties on this grid.
54950  */
54951 Roo.grid.Grid = function(container, config){
54952         // initialize the container
54953         this.container = Roo.get(container);
54954         this.container.update("");
54955         this.container.setStyle("overflow", "hidden");
54956     this.container.addClass('x-grid-container');
54957
54958     this.id = this.container.id;
54959
54960     Roo.apply(this, config);
54961     // check and correct shorthanded configs
54962     if(this.ds){
54963         this.dataSource = this.ds;
54964         delete this.ds;
54965     }
54966     if(this.cm){
54967         this.colModel = this.cm;
54968         delete this.cm;
54969     }
54970     if(this.sm){
54971         this.selModel = this.sm;
54972         delete this.sm;
54973     }
54974
54975     if (this.selModel) {
54976         this.selModel = Roo.factory(this.selModel, Roo.grid);
54977         this.sm = this.selModel;
54978         this.sm.xmodule = this.xmodule || false;
54979     }
54980     if (typeof(this.colModel.config) == 'undefined') {
54981         this.colModel = new Roo.grid.ColumnModel(this.colModel);
54982         this.cm = this.colModel;
54983         this.cm.xmodule = this.xmodule || false;
54984     }
54985     if (this.dataSource) {
54986         this.dataSource= Roo.factory(this.dataSource, Roo.data);
54987         this.ds = this.dataSource;
54988         this.ds.xmodule = this.xmodule || false;
54989          
54990     }
54991     
54992     
54993     
54994     if(this.width){
54995         this.container.setWidth(this.width);
54996     }
54997
54998     if(this.height){
54999         this.container.setHeight(this.height);
55000     }
55001     /** @private */
55002         this.addEvents({
55003         // raw events
55004         /**
55005          * @event click
55006          * The raw click event for the entire grid.
55007          * @param {Roo.EventObject} e
55008          */
55009         "click" : true,
55010         /**
55011          * @event dblclick
55012          * The raw dblclick event for the entire grid.
55013          * @param {Roo.EventObject} e
55014          */
55015         "dblclick" : true,
55016         /**
55017          * @event contextmenu
55018          * The raw contextmenu event for the entire grid.
55019          * @param {Roo.EventObject} e
55020          */
55021         "contextmenu" : true,
55022         /**
55023          * @event mousedown
55024          * The raw mousedown event for the entire grid.
55025          * @param {Roo.EventObject} e
55026          */
55027         "mousedown" : true,
55028         /**
55029          * @event mouseup
55030          * The raw mouseup event for the entire grid.
55031          * @param {Roo.EventObject} e
55032          */
55033         "mouseup" : true,
55034         /**
55035          * @event mouseover
55036          * The raw mouseover event for the entire grid.
55037          * @param {Roo.EventObject} e
55038          */
55039         "mouseover" : true,
55040         /**
55041          * @event mouseout
55042          * The raw mouseout event for the entire grid.
55043          * @param {Roo.EventObject} e
55044          */
55045         "mouseout" : true,
55046         /**
55047          * @event keypress
55048          * The raw keypress event for the entire grid.
55049          * @param {Roo.EventObject} e
55050          */
55051         "keypress" : true,
55052         /**
55053          * @event keydown
55054          * The raw keydown event for the entire grid.
55055          * @param {Roo.EventObject} e
55056          */
55057         "keydown" : true,
55058
55059         // custom events
55060
55061         /**
55062          * @event cellclick
55063          * Fires when a cell is clicked
55064          * @param {Grid} this
55065          * @param {Number} rowIndex
55066          * @param {Number} columnIndex
55067          * @param {Roo.EventObject} e
55068          */
55069         "cellclick" : true,
55070         /**
55071          * @event celldblclick
55072          * Fires when a cell is double clicked
55073          * @param {Grid} this
55074          * @param {Number} rowIndex
55075          * @param {Number} columnIndex
55076          * @param {Roo.EventObject} e
55077          */
55078         "celldblclick" : true,
55079         /**
55080          * @event rowclick
55081          * Fires when a row is clicked
55082          * @param {Grid} this
55083          * @param {Number} rowIndex
55084          * @param {Roo.EventObject} e
55085          */
55086         "rowclick" : true,
55087         /**
55088          * @event rowdblclick
55089          * Fires when a row is double clicked
55090          * @param {Grid} this
55091          * @param {Number} rowIndex
55092          * @param {Roo.EventObject} e
55093          */
55094         "rowdblclick" : true,
55095         /**
55096          * @event headerclick
55097          * Fires when a header is clicked
55098          * @param {Grid} this
55099          * @param {Number} columnIndex
55100          * @param {Roo.EventObject} e
55101          */
55102         "headerclick" : true,
55103         /**
55104          * @event headerdblclick
55105          * Fires when a header cell is double clicked
55106          * @param {Grid} this
55107          * @param {Number} columnIndex
55108          * @param {Roo.EventObject} e
55109          */
55110         "headerdblclick" : true,
55111         /**
55112          * @event rowcontextmenu
55113          * Fires when a row is right clicked
55114          * @param {Grid} this
55115          * @param {Number} rowIndex
55116          * @param {Roo.EventObject} e
55117          */
55118         "rowcontextmenu" : true,
55119         /**
55120          * @event cellcontextmenu
55121          * Fires when a cell is right clicked
55122          * @param {Grid} this
55123          * @param {Number} rowIndex
55124          * @param {Number} cellIndex
55125          * @param {Roo.EventObject} e
55126          */
55127          "cellcontextmenu" : true,
55128         /**
55129          * @event headercontextmenu
55130          * Fires when a header is right clicked
55131          * @param {Grid} this
55132          * @param {Number} columnIndex
55133          * @param {Roo.EventObject} e
55134          */
55135         "headercontextmenu" : true,
55136         /**
55137          * @event bodyscroll
55138          * Fires when the body element is scrolled
55139          * @param {Number} scrollLeft
55140          * @param {Number} scrollTop
55141          */
55142         "bodyscroll" : true,
55143         /**
55144          * @event columnresize
55145          * Fires when the user resizes a column
55146          * @param {Number} columnIndex
55147          * @param {Number} newSize
55148          */
55149         "columnresize" : true,
55150         /**
55151          * @event columnmove
55152          * Fires when the user moves a column
55153          * @param {Number} oldIndex
55154          * @param {Number} newIndex
55155          */
55156         "columnmove" : true,
55157         /**
55158          * @event startdrag
55159          * Fires when row(s) start being dragged
55160          * @param {Grid} this
55161          * @param {Roo.GridDD} dd The drag drop object
55162          * @param {event} e The raw browser event
55163          */
55164         "startdrag" : true,
55165         /**
55166          * @event enddrag
55167          * Fires when a drag operation is complete
55168          * @param {Grid} this
55169          * @param {Roo.GridDD} dd The drag drop object
55170          * @param {event} e The raw browser event
55171          */
55172         "enddrag" : true,
55173         /**
55174          * @event dragdrop
55175          * Fires when dragged row(s) are dropped on a valid DD target
55176          * @param {Grid} this
55177          * @param {Roo.GridDD} dd The drag drop object
55178          * @param {String} targetId The target drag drop object
55179          * @param {event} e The raw browser event
55180          */
55181         "dragdrop" : true,
55182         /**
55183          * @event dragover
55184          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
55185          * @param {Grid} this
55186          * @param {Roo.GridDD} dd The drag drop object
55187          * @param {String} targetId The target drag drop object
55188          * @param {event} e The raw browser event
55189          */
55190         "dragover" : true,
55191         /**
55192          * @event dragenter
55193          *  Fires when the dragged row(s) first cross another DD target while being dragged
55194          * @param {Grid} this
55195          * @param {Roo.GridDD} dd The drag drop object
55196          * @param {String} targetId The target drag drop object
55197          * @param {event} e The raw browser event
55198          */
55199         "dragenter" : true,
55200         /**
55201          * @event dragout
55202          * Fires when the dragged row(s) leave another DD target while being dragged
55203          * @param {Grid} this
55204          * @param {Roo.GridDD} dd The drag drop object
55205          * @param {String} targetId The target drag drop object
55206          * @param {event} e The raw browser event
55207          */
55208         "dragout" : true,
55209         /**
55210          * @event rowclass
55211          * Fires when a row is rendered, so you can change add a style to it.
55212          * @param {GridView} gridview   The grid view
55213          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
55214          */
55215         'rowclass' : true,
55216
55217         /**
55218          * @event render
55219          * Fires when the grid is rendered
55220          * @param {Grid} grid
55221          */
55222         'render' : true
55223     });
55224
55225     Roo.grid.Grid.superclass.constructor.call(this);
55226 };
55227 Roo.extend(Roo.grid.Grid, Roo.util.Observable, {
55228     
55229     /**
55230      * @cfg {String} ddGroup - drag drop group.
55231      */
55232
55233     /**
55234      * @cfg {Number} minColumnWidth The minimum width a column can be resized to. Default is 25.
55235      */
55236     minColumnWidth : 25,
55237
55238     /**
55239      * @cfg {Boolean} autoSizeColumns True to automatically resize the columns to fit their content
55240      * <b>on initial render.</b> It is more efficient to explicitly size the columns
55241      * through the ColumnModel's {@link Roo.grid.ColumnModel#width} config option.  Default is false.
55242      */
55243     autoSizeColumns : false,
55244
55245     /**
55246      * @cfg {Boolean} autoSizeHeaders True to measure headers with column data when auto sizing columns. Default is true.
55247      */
55248     autoSizeHeaders : true,
55249
55250     /**
55251      * @cfg {Boolean} monitorWindowResize True to autoSize the grid when the window resizes. Default is true.
55252      */
55253     monitorWindowResize : true,
55254
55255     /**
55256      * @cfg {Boolean} maxRowsToMeasure If autoSizeColumns is on, maxRowsToMeasure can be used to limit the number of
55257      * rows measured to get a columns size. Default is 0 (all rows).
55258      */
55259     maxRowsToMeasure : 0,
55260
55261     /**
55262      * @cfg {Boolean} trackMouseOver True to highlight rows when the mouse is over. Default is true.
55263      */
55264     trackMouseOver : true,
55265
55266     /**
55267     * @cfg {Boolean} enableDrag  True to enable drag of rows. Default is false. (double check if this is needed?)
55268     */
55269     
55270     /**
55271     * @cfg {Boolean} enableDragDrop True to enable drag and drop of rows. Default is false.
55272     */
55273     enableDragDrop : false,
55274     
55275     /**
55276     * @cfg {Boolean} enableColumnMove True to enable drag and drop reorder of columns. Default is true.
55277     */
55278     enableColumnMove : true,
55279     
55280     /**
55281     * @cfg {Boolean} enableColumnHide True to enable hiding of columns with the header context menu. Default is true.
55282     */
55283     enableColumnHide : true,
55284     
55285     /**
55286     * @cfg {Boolean} enableRowHeightSync True to manually sync row heights across locked and not locked rows. Default is false.
55287     */
55288     enableRowHeightSync : false,
55289     
55290     /**
55291     * @cfg {Boolean} stripeRows True to stripe the rows.  Default is true.
55292     */
55293     stripeRows : true,
55294     
55295     /**
55296     * @cfg {Boolean} autoHeight True to fit the height of the grid container to the height of the data. Default is false.
55297     */
55298     autoHeight : false,
55299
55300     /**
55301      * @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.
55302      */
55303     autoExpandColumn : false,
55304
55305     /**
55306     * @cfg {Number} autoExpandMin The minimum width the autoExpandColumn can have (if enabled).
55307     * Default is 50.
55308     */
55309     autoExpandMin : 50,
55310
55311     /**
55312     * @cfg {Number} autoExpandMax The maximum width the autoExpandColumn can have (if enabled). Default is 1000.
55313     */
55314     autoExpandMax : 1000,
55315
55316     /**
55317     * @cfg {Object} view The {@link Roo.grid.GridView} used by the grid. This can be set before a call to render().
55318     */
55319     view : null,
55320
55321     /**
55322     * @cfg {Object} loadMask An {@link Roo.LoadMask} config or true to mask the grid while loading. Default is false.
55323     */
55324     loadMask : false,
55325     /**
55326     * @cfg {Roo.dd.DropTarget} dropTarget An {@link Roo.dd.DropTarget} config
55327     */
55328     dropTarget: false,
55329     
55330    
55331     
55332     // private
55333     rendered : false,
55334
55335     /**
55336     * @cfg {Boolean} autoWidth True to set the grid's width to the default total width of the grid's columns instead
55337     * of a fixed width. Default is false.
55338     */
55339     /**
55340     * @cfg {Number} maxHeight Sets the maximum height of the grid - ignored if autoHeight is not on.
55341     */
55342     /**
55343      * Called once after all setup has been completed and the grid is ready to be rendered.
55344      * @return {Roo.grid.Grid} this
55345      */
55346     render : function()
55347     {
55348         var c = this.container;
55349         // try to detect autoHeight/width mode
55350         if((!c.dom.offsetHeight || c.dom.offsetHeight < 20) || c.getStyle("height") == "auto"){
55351             this.autoHeight = true;
55352         }
55353         var view = this.getView();
55354         view.init(this);
55355
55356         c.on("click", this.onClick, this);
55357         c.on("dblclick", this.onDblClick, this);
55358         c.on("contextmenu", this.onContextMenu, this);
55359         c.on("keydown", this.onKeyDown, this);
55360         if (Roo.isTouch) {
55361             c.on("touchstart", this.onTouchStart, this);
55362         }
55363
55364         this.relayEvents(c, ["mousedown","mouseup","mouseover","mouseout","keypress"]);
55365
55366         this.getSelectionModel().init(this);
55367
55368         view.render();
55369
55370         if(this.loadMask){
55371             this.loadMask = new Roo.LoadMask(this.container,
55372                     Roo.apply({store:this.dataSource}, this.loadMask));
55373         }
55374         
55375         
55376         if (this.toolbar && this.toolbar.xtype) {
55377             this.toolbar.container = this.getView().getHeaderPanel(true);
55378             this.toolbar = new Roo.Toolbar(this.toolbar);
55379         }
55380         if (this.footer && this.footer.xtype) {
55381             this.footer.dataSource = this.getDataSource();
55382             this.footer.container = this.getView().getFooterPanel(true);
55383             this.footer = Roo.factory(this.footer, Roo);
55384         }
55385         if (this.dropTarget && this.dropTarget.xtype) {
55386             delete this.dropTarget.xtype;
55387             this.dropTarget =  new Roo.dd.DropTarget(this.getView().mainBody, this.dropTarget);
55388         }
55389         
55390         
55391         this.rendered = true;
55392         this.fireEvent('render', this);
55393         return this;
55394     },
55395
55396         /**
55397          * Reconfigures the grid to use a different Store and Column Model.
55398          * The View will be bound to the new objects and refreshed.
55399          * @param {Roo.data.Store} dataSource The new {@link Roo.data.Store} object
55400          * @param {Roo.grid.ColumnModel} The new {@link Roo.grid.ColumnModel} object
55401          */
55402     reconfigure : function(dataSource, colModel){
55403         if(this.loadMask){
55404             this.loadMask.destroy();
55405             this.loadMask = new Roo.LoadMask(this.container,
55406                     Roo.apply({store:dataSource}, this.loadMask));
55407         }
55408         this.view.bind(dataSource, colModel);
55409         this.dataSource = dataSource;
55410         this.colModel = colModel;
55411         this.view.refresh(true);
55412     },
55413
55414     // private
55415     onKeyDown : function(e){
55416         this.fireEvent("keydown", e);
55417     },
55418
55419     /**
55420      * Destroy this grid.
55421      * @param {Boolean} removeEl True to remove the element
55422      */
55423     destroy : function(removeEl, keepListeners){
55424         if(this.loadMask){
55425             this.loadMask.destroy();
55426         }
55427         var c = this.container;
55428         c.removeAllListeners();
55429         this.view.destroy();
55430         this.colModel.purgeListeners();
55431         if(!keepListeners){
55432             this.purgeListeners();
55433         }
55434         c.update("");
55435         if(removeEl === true){
55436             c.remove();
55437         }
55438     },
55439
55440     // private
55441     processEvent : function(name, e){
55442         // does this fire select???
55443         //Roo.log('grid:processEvent '  + name);
55444         
55445         if (name != 'touchstart' ) {
55446             this.fireEvent(name, e);    
55447         }
55448         
55449         var t = e.getTarget();
55450         var v = this.view;
55451         var header = v.findHeaderIndex(t);
55452         if(header !== false){
55453             var ename = name == 'touchstart' ? 'click' : name;
55454              
55455             this.fireEvent("header" + ename, this, header, e);
55456         }else{
55457             var row = v.findRowIndex(t);
55458             var cell = v.findCellIndex(t);
55459             if (name == 'touchstart') {
55460                 // first touch is always a click.
55461                 // hopefull this happens after selection is updated.?
55462                 name = false;
55463                 
55464                 if (typeof(this.selModel.getSelectedCell) != 'undefined') {
55465                     var cs = this.selModel.getSelectedCell();
55466                     if (row == cs[0] && cell == cs[1]){
55467                         name = 'dblclick';
55468                     }
55469                 }
55470                 if (typeof(this.selModel.getSelections) != 'undefined') {
55471                     var cs = this.selModel.getSelections();
55472                     var ds = this.dataSource;
55473                     if (cs.length == 1 && ds.getAt(row) == cs[0]){
55474                         name = 'dblclick';
55475                     }
55476                 }
55477                 if (!name) {
55478                     return;
55479                 }
55480             }
55481             
55482             
55483             if(row !== false){
55484                 this.fireEvent("row" + name, this, row, e);
55485                 if(cell !== false){
55486                     this.fireEvent("cell" + name, this, row, cell, e);
55487                 }
55488             }
55489         }
55490     },
55491
55492     // private
55493     onClick : function(e){
55494         this.processEvent("click", e);
55495     },
55496    // private
55497     onTouchStart : function(e){
55498         this.processEvent("touchstart", e);
55499     },
55500
55501     // private
55502     onContextMenu : function(e, t){
55503         this.processEvent("contextmenu", e);
55504     },
55505
55506     // private
55507     onDblClick : function(e){
55508         this.processEvent("dblclick", e);
55509     },
55510
55511     // private
55512     walkCells : function(row, col, step, fn, scope){
55513         var cm = this.colModel, clen = cm.getColumnCount();
55514         var ds = this.dataSource, rlen = ds.getCount(), first = true;
55515         if(step < 0){
55516             if(col < 0){
55517                 row--;
55518                 first = false;
55519             }
55520             while(row >= 0){
55521                 if(!first){
55522                     col = clen-1;
55523                 }
55524                 first = false;
55525                 while(col >= 0){
55526                     if(fn.call(scope || this, row, col, cm) === true){
55527                         return [row, col];
55528                     }
55529                     col--;
55530                 }
55531                 row--;
55532             }
55533         } else {
55534             if(col >= clen){
55535                 row++;
55536                 first = false;
55537             }
55538             while(row < rlen){
55539                 if(!first){
55540                     col = 0;
55541                 }
55542                 first = false;
55543                 while(col < clen){
55544                     if(fn.call(scope || this, row, col, cm) === true){
55545                         return [row, col];
55546                     }
55547                     col++;
55548                 }
55549                 row++;
55550             }
55551         }
55552         return null;
55553     },
55554
55555     // private
55556     getSelections : function(){
55557         return this.selModel.getSelections();
55558     },
55559
55560     /**
55561      * Causes the grid to manually recalculate its dimensions. Generally this is done automatically,
55562      * but if manual update is required this method will initiate it.
55563      */
55564     autoSize : function(){
55565         if(this.rendered){
55566             this.view.layout();
55567             if(this.view.adjustForScroll){
55568                 this.view.adjustForScroll();
55569             }
55570         }
55571     },
55572
55573     /**
55574      * Returns the grid's underlying element.
55575      * @return {Element} The element
55576      */
55577     getGridEl : function(){
55578         return this.container;
55579     },
55580
55581     // private for compatibility, overridden by editor grid
55582     stopEditing : function(){},
55583
55584     /**
55585      * Returns the grid's SelectionModel.
55586      * @return {SelectionModel}
55587      */
55588     getSelectionModel : function(){
55589         if(!this.selModel){
55590             this.selModel = new Roo.grid.RowSelectionModel();
55591         }
55592         return this.selModel;
55593     },
55594
55595     /**
55596      * Returns the grid's DataSource.
55597      * @return {DataSource}
55598      */
55599     getDataSource : function(){
55600         return this.dataSource;
55601     },
55602
55603     /**
55604      * Returns the grid's ColumnModel.
55605      * @return {ColumnModel}
55606      */
55607     getColumnModel : function(){
55608         return this.colModel;
55609     },
55610
55611     /**
55612      * Returns the grid's GridView object.
55613      * @return {GridView}
55614      */
55615     getView : function(){
55616         if(!this.view){
55617             this.view = new Roo.grid.GridView(this.viewConfig);
55618         }
55619         return this.view;
55620     },
55621     /**
55622      * Called to get grid's drag proxy text, by default returns this.ddText.
55623      * @return {String}
55624      */
55625     getDragDropText : function(){
55626         var count = this.selModel.getCount();
55627         return String.format(this.ddText, count, count == 1 ? '' : 's');
55628     }
55629 });
55630 /**
55631  * Configures the text is the drag proxy (defaults to "%0 selected row(s)").
55632  * %0 is replaced with the number of selected rows.
55633  * @type String
55634  */
55635 Roo.grid.Grid.prototype.ddText = "{0} selected row{1}";/*
55636  * Based on:
55637  * Ext JS Library 1.1.1
55638  * Copyright(c) 2006-2007, Ext JS, LLC.
55639  *
55640  * Originally Released Under LGPL - original licence link has changed is not relivant.
55641  *
55642  * Fork - LGPL
55643  * <script type="text/javascript">
55644  */
55645  
55646 Roo.grid.AbstractGridView = function(){
55647         this.grid = null;
55648         
55649         this.events = {
55650             "beforerowremoved" : true,
55651             "beforerowsinserted" : true,
55652             "beforerefresh" : true,
55653             "rowremoved" : true,
55654             "rowsinserted" : true,
55655             "rowupdated" : true,
55656             "refresh" : true
55657         };
55658     Roo.grid.AbstractGridView.superclass.constructor.call(this);
55659 };
55660
55661 Roo.extend(Roo.grid.AbstractGridView, Roo.util.Observable, {
55662     rowClass : "x-grid-row",
55663     cellClass : "x-grid-cell",
55664     tdClass : "x-grid-td",
55665     hdClass : "x-grid-hd",
55666     splitClass : "x-grid-hd-split",
55667     
55668     init: function(grid){
55669         this.grid = grid;
55670                 var cid = this.grid.getGridEl().id;
55671         this.colSelector = "#" + cid + " ." + this.cellClass + "-";
55672         this.tdSelector = "#" + cid + " ." + this.tdClass + "-";
55673         this.hdSelector = "#" + cid + " ." + this.hdClass + "-";
55674         this.splitSelector = "#" + cid + " ." + this.splitClass + "-";
55675         },
55676         
55677     getColumnRenderers : function(){
55678         var renderers = [];
55679         var cm = this.grid.colModel;
55680         var colCount = cm.getColumnCount();
55681         for(var i = 0; i < colCount; i++){
55682             renderers[i] = cm.getRenderer(i);
55683         }
55684         return renderers;
55685     },
55686     
55687     getColumnIds : function(){
55688         var ids = [];
55689         var cm = this.grid.colModel;
55690         var colCount = cm.getColumnCount();
55691         for(var i = 0; i < colCount; i++){
55692             ids[i] = cm.getColumnId(i);
55693         }
55694         return ids;
55695     },
55696     
55697     getDataIndexes : function(){
55698         if(!this.indexMap){
55699             this.indexMap = this.buildIndexMap();
55700         }
55701         return this.indexMap.colToData;
55702     },
55703     
55704     getColumnIndexByDataIndex : function(dataIndex){
55705         if(!this.indexMap){
55706             this.indexMap = this.buildIndexMap();
55707         }
55708         return this.indexMap.dataToCol[dataIndex];
55709     },
55710     
55711     /**
55712      * Set a css style for a column dynamically. 
55713      * @param {Number} colIndex The index of the column
55714      * @param {String} name The css property name
55715      * @param {String} value The css value
55716      */
55717     setCSSStyle : function(colIndex, name, value){
55718         var selector = "#" + this.grid.id + " .x-grid-col-" + colIndex;
55719         Roo.util.CSS.updateRule(selector, name, value);
55720     },
55721     
55722     generateRules : function(cm){
55723         var ruleBuf = [], rulesId = this.grid.id + '-cssrules';
55724         Roo.util.CSS.removeStyleSheet(rulesId);
55725         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
55726             var cid = cm.getColumnId(i);
55727             ruleBuf.push(this.colSelector, cid, " {\n", cm.config[i].css, "}\n",
55728                          this.tdSelector, cid, " {\n}\n",
55729                          this.hdSelector, cid, " {\n}\n",
55730                          this.splitSelector, cid, " {\n}\n");
55731         }
55732         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
55733     }
55734 });/*
55735  * Based on:
55736  * Ext JS Library 1.1.1
55737  * Copyright(c) 2006-2007, Ext JS, LLC.
55738  *
55739  * Originally Released Under LGPL - original licence link has changed is not relivant.
55740  *
55741  * Fork - LGPL
55742  * <script type="text/javascript">
55743  */
55744
55745 // private
55746 // This is a support class used internally by the Grid components
55747 Roo.grid.HeaderDragZone = function(grid, hd, hd2){
55748     this.grid = grid;
55749     this.view = grid.getView();
55750     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55751     Roo.grid.HeaderDragZone.superclass.constructor.call(this, hd);
55752     if(hd2){
55753         this.setHandleElId(Roo.id(hd));
55754         this.setOuterHandleElId(Roo.id(hd2));
55755     }
55756     this.scroll = false;
55757 };
55758 Roo.extend(Roo.grid.HeaderDragZone, Roo.dd.DragZone, {
55759     maxDragWidth: 120,
55760     getDragData : function(e){
55761         var t = Roo.lib.Event.getTarget(e);
55762         var h = this.view.findHeaderCell(t);
55763         if(h){
55764             return {ddel: h.firstChild, header:h};
55765         }
55766         return false;
55767     },
55768
55769     onInitDrag : function(e){
55770         this.view.headersDisabled = true;
55771         var clone = this.dragData.ddel.cloneNode(true);
55772         clone.id = Roo.id();
55773         clone.style.width = Math.min(this.dragData.header.offsetWidth,this.maxDragWidth) + "px";
55774         this.proxy.update(clone);
55775         return true;
55776     },
55777
55778     afterValidDrop : function(){
55779         var v = this.view;
55780         setTimeout(function(){
55781             v.headersDisabled = false;
55782         }, 50);
55783     },
55784
55785     afterInvalidDrop : function(){
55786         var v = this.view;
55787         setTimeout(function(){
55788             v.headersDisabled = false;
55789         }, 50);
55790     }
55791 });
55792 /*
55793  * Based on:
55794  * Ext JS Library 1.1.1
55795  * Copyright(c) 2006-2007, Ext JS, LLC.
55796  *
55797  * Originally Released Under LGPL - original licence link has changed is not relivant.
55798  *
55799  * Fork - LGPL
55800  * <script type="text/javascript">
55801  */
55802 // private
55803 // This is a support class used internally by the Grid components
55804 Roo.grid.HeaderDropZone = function(grid, hd, hd2){
55805     this.grid = grid;
55806     this.view = grid.getView();
55807     // split the proxies so they don't interfere with mouse events
55808     this.proxyTop = Roo.DomHelper.append(document.body, {
55809         cls:"col-move-top", html:"&#160;"
55810     }, true);
55811     this.proxyBottom = Roo.DomHelper.append(document.body, {
55812         cls:"col-move-bottom", html:"&#160;"
55813     }, true);
55814     this.proxyTop.hide = this.proxyBottom.hide = function(){
55815         this.setLeftTop(-100,-100);
55816         this.setStyle("visibility", "hidden");
55817     };
55818     this.ddGroup = "gridHeader" + this.grid.getGridEl().id;
55819     // temporarily disabled
55820     //Roo.dd.ScrollManager.register(this.view.scroller.dom);
55821     Roo.grid.HeaderDropZone.superclass.constructor.call(this, grid.getGridEl().dom);
55822 };
55823 Roo.extend(Roo.grid.HeaderDropZone, Roo.dd.DropZone, {
55824     proxyOffsets : [-4, -9],
55825     fly: Roo.Element.fly,
55826
55827     getTargetFromEvent : function(e){
55828         var t = Roo.lib.Event.getTarget(e);
55829         var cindex = this.view.findCellIndex(t);
55830         if(cindex !== false){
55831             return this.view.getHeaderCell(cindex);
55832         }
55833         return null;
55834     },
55835
55836     nextVisible : function(h){
55837         var v = this.view, cm = this.grid.colModel;
55838         h = h.nextSibling;
55839         while(h){
55840             if(!cm.isHidden(v.getCellIndex(h))){
55841                 return h;
55842             }
55843             h = h.nextSibling;
55844         }
55845         return null;
55846     },
55847
55848     prevVisible : function(h){
55849         var v = this.view, cm = this.grid.colModel;
55850         h = h.prevSibling;
55851         while(h){
55852             if(!cm.isHidden(v.getCellIndex(h))){
55853                 return h;
55854             }
55855             h = h.prevSibling;
55856         }
55857         return null;
55858     },
55859
55860     positionIndicator : function(h, n, e){
55861         var x = Roo.lib.Event.getPageX(e);
55862         var r = Roo.lib.Dom.getRegion(n.firstChild);
55863         var px, pt, py = r.top + this.proxyOffsets[1];
55864         if((r.right - x) <= (r.right-r.left)/2){
55865             px = r.right+this.view.borderWidth;
55866             pt = "after";
55867         }else{
55868             px = r.left;
55869             pt = "before";
55870         }
55871         var oldIndex = this.view.getCellIndex(h);
55872         var newIndex = this.view.getCellIndex(n);
55873
55874         if(this.grid.colModel.isFixed(newIndex)){
55875             return false;
55876         }
55877
55878         var locked = this.grid.colModel.isLocked(newIndex);
55879
55880         if(pt == "after"){
55881             newIndex++;
55882         }
55883         if(oldIndex < newIndex){
55884             newIndex--;
55885         }
55886         if(oldIndex == newIndex && (locked == this.grid.colModel.isLocked(oldIndex))){
55887             return false;
55888         }
55889         px +=  this.proxyOffsets[0];
55890         this.proxyTop.setLeftTop(px, py);
55891         this.proxyTop.show();
55892         if(!this.bottomOffset){
55893             this.bottomOffset = this.view.mainHd.getHeight();
55894         }
55895         this.proxyBottom.setLeftTop(px, py+this.proxyTop.dom.offsetHeight+this.bottomOffset);
55896         this.proxyBottom.show();
55897         return pt;
55898     },
55899
55900     onNodeEnter : function(n, dd, e, data){
55901         if(data.header != n){
55902             this.positionIndicator(data.header, n, e);
55903         }
55904     },
55905
55906     onNodeOver : function(n, dd, e, data){
55907         var result = false;
55908         if(data.header != n){
55909             result = this.positionIndicator(data.header, n, e);
55910         }
55911         if(!result){
55912             this.proxyTop.hide();
55913             this.proxyBottom.hide();
55914         }
55915         return result ? this.dropAllowed : this.dropNotAllowed;
55916     },
55917
55918     onNodeOut : function(n, dd, e, data){
55919         this.proxyTop.hide();
55920         this.proxyBottom.hide();
55921     },
55922
55923     onNodeDrop : function(n, dd, e, data){
55924         var h = data.header;
55925         if(h != n){
55926             var cm = this.grid.colModel;
55927             var x = Roo.lib.Event.getPageX(e);
55928             var r = Roo.lib.Dom.getRegion(n.firstChild);
55929             var pt = (r.right - x) <= ((r.right-r.left)/2) ? "after" : "before";
55930             var oldIndex = this.view.getCellIndex(h);
55931             var newIndex = this.view.getCellIndex(n);
55932             var locked = cm.isLocked(newIndex);
55933             if(pt == "after"){
55934                 newIndex++;
55935             }
55936             if(oldIndex < newIndex){
55937                 newIndex--;
55938             }
55939             if(oldIndex == newIndex && (locked == cm.isLocked(oldIndex))){
55940                 return false;
55941             }
55942             cm.setLocked(oldIndex, locked, true);
55943             cm.moveColumn(oldIndex, newIndex);
55944             this.grid.fireEvent("columnmove", oldIndex, newIndex);
55945             return true;
55946         }
55947         return false;
55948     }
55949 });
55950 /*
55951  * Based on:
55952  * Ext JS Library 1.1.1
55953  * Copyright(c) 2006-2007, Ext JS, LLC.
55954  *
55955  * Originally Released Under LGPL - original licence link has changed is not relivant.
55956  *
55957  * Fork - LGPL
55958  * <script type="text/javascript">
55959  */
55960   
55961 /**
55962  * @class Roo.grid.GridView
55963  * @extends Roo.util.Observable
55964  *
55965  * @constructor
55966  * @param {Object} config
55967  */
55968 Roo.grid.GridView = function(config){
55969     Roo.grid.GridView.superclass.constructor.call(this);
55970     this.el = null;
55971
55972     Roo.apply(this, config);
55973 };
55974
55975 Roo.extend(Roo.grid.GridView, Roo.grid.AbstractGridView, {
55976
55977     unselectable :  'unselectable="on"',
55978     unselectableCls :  'x-unselectable',
55979     
55980     
55981     rowClass : "x-grid-row",
55982
55983     cellClass : "x-grid-col",
55984
55985     tdClass : "x-grid-td",
55986
55987     hdClass : "x-grid-hd",
55988
55989     splitClass : "x-grid-split",
55990
55991     sortClasses : ["sort-asc", "sort-desc"],
55992
55993     enableMoveAnim : false,
55994
55995     hlColor: "C3DAF9",
55996
55997     dh : Roo.DomHelper,
55998
55999     fly : Roo.Element.fly,
56000
56001     css : Roo.util.CSS,
56002
56003     borderWidth: 1,
56004
56005     splitOffset: 3,
56006
56007     scrollIncrement : 22,
56008
56009     cellRE: /(?:.*?)x-grid-(?:hd|cell|csplit)-(?:[\d]+)-([\d]+)(?:.*?)/,
56010
56011     findRE: /\s?(?:x-grid-hd|x-grid-col|x-grid-csplit)\s/,
56012
56013     bind : function(ds, cm){
56014         if(this.ds){
56015             this.ds.un("load", this.onLoad, this);
56016             this.ds.un("datachanged", this.onDataChange, this);
56017             this.ds.un("add", this.onAdd, this);
56018             this.ds.un("remove", this.onRemove, this);
56019             this.ds.un("update", this.onUpdate, this);
56020             this.ds.un("clear", this.onClear, this);
56021         }
56022         if(ds){
56023             ds.on("load", this.onLoad, this);
56024             ds.on("datachanged", this.onDataChange, this);
56025             ds.on("add", this.onAdd, this);
56026             ds.on("remove", this.onRemove, this);
56027             ds.on("update", this.onUpdate, this);
56028             ds.on("clear", this.onClear, this);
56029         }
56030         this.ds = ds;
56031
56032         if(this.cm){
56033             this.cm.un("widthchange", this.onColWidthChange, this);
56034             this.cm.un("headerchange", this.onHeaderChange, this);
56035             this.cm.un("hiddenchange", this.onHiddenChange, this);
56036             this.cm.un("columnmoved", this.onColumnMove, this);
56037             this.cm.un("columnlockchange", this.onColumnLock, this);
56038         }
56039         if(cm){
56040             this.generateRules(cm);
56041             cm.on("widthchange", this.onColWidthChange, this);
56042             cm.on("headerchange", this.onHeaderChange, this);
56043             cm.on("hiddenchange", this.onHiddenChange, this);
56044             cm.on("columnmoved", this.onColumnMove, this);
56045             cm.on("columnlockchange", this.onColumnLock, this);
56046         }
56047         this.cm = cm;
56048     },
56049
56050     init: function(grid){
56051         Roo.grid.GridView.superclass.init.call(this, grid);
56052
56053         this.bind(grid.dataSource, grid.colModel);
56054
56055         grid.on("headerclick", this.handleHeaderClick, this);
56056
56057         if(grid.trackMouseOver){
56058             grid.on("mouseover", this.onRowOver, this);
56059             grid.on("mouseout", this.onRowOut, this);
56060         }
56061         grid.cancelTextSelection = function(){};
56062         this.gridId = grid.id;
56063
56064         var tpls = this.templates || {};
56065
56066         if(!tpls.master){
56067             tpls.master = new Roo.Template(
56068                '<div class="x-grid" hidefocus="true">',
56069                 '<a href="#" class="x-grid-focus" tabIndex="-1"></a>',
56070                   '<div class="x-grid-topbar"></div>',
56071                   '<div class="x-grid-scroller"><div></div></div>',
56072                   '<div class="x-grid-locked">',
56073                       '<div class="x-grid-header">{lockedHeader}</div>',
56074                       '<div class="x-grid-body">{lockedBody}</div>',
56075                   "</div>",
56076                   '<div class="x-grid-viewport">',
56077                       '<div class="x-grid-header">{header}</div>',
56078                       '<div class="x-grid-body">{body}</div>',
56079                   "</div>",
56080                   '<div class="x-grid-bottombar"></div>',
56081                  
56082                   '<div class="x-grid-resize-proxy">&#160;</div>',
56083                "</div>"
56084             );
56085             tpls.master.disableformats = true;
56086         }
56087
56088         if(!tpls.header){
56089             tpls.header = new Roo.Template(
56090                '<table border="0" cellspacing="0" cellpadding="0">',
56091                '<tbody><tr class="x-grid-hd-row">{cells}</tr></tbody>',
56092                "</table>{splits}"
56093             );
56094             tpls.header.disableformats = true;
56095         }
56096         tpls.header.compile();
56097
56098         if(!tpls.hcell){
56099             tpls.hcell = new Roo.Template(
56100                 '<td class="x-grid-hd x-grid-td-{id} {cellId}"><div title="{title}" class="x-grid-hd-inner x-grid-hd-{id}">',
56101                 '<div class="x-grid-hd-text ' + this.unselectableCls +  '" ' + this.unselectable +'>{value}<img class="x-grid-sort-icon" src="', Roo.BLANK_IMAGE_URL, '" /></div>',
56102                 "</div></td>"
56103              );
56104              tpls.hcell.disableFormats = true;
56105         }
56106         tpls.hcell.compile();
56107
56108         if(!tpls.hsplit){
56109             tpls.hsplit = new Roo.Template('<div class="x-grid-split {splitId} x-grid-split-{id}" style="{style} ' +
56110                                             this.unselectableCls +  '" ' + this.unselectable +'>&#160;</div>');
56111             tpls.hsplit.disableFormats = true;
56112         }
56113         tpls.hsplit.compile();
56114
56115         if(!tpls.body){
56116             tpls.body = new Roo.Template(
56117                '<table border="0" cellspacing="0" cellpadding="0">',
56118                "<tbody>{rows}</tbody>",
56119                "</table>"
56120             );
56121             tpls.body.disableFormats = true;
56122         }
56123         tpls.body.compile();
56124
56125         if(!tpls.row){
56126             tpls.row = new Roo.Template('<tr class="x-grid-row {alt}">{cells}</tr>');
56127             tpls.row.disableFormats = true;
56128         }
56129         tpls.row.compile();
56130
56131         if(!tpls.cell){
56132             tpls.cell = new Roo.Template(
56133                 '<td class="x-grid-col x-grid-td-{id} {cellId} {css}" tabIndex="0">',
56134                 '<div class="x-grid-col-{id} x-grid-cell-inner"><div class="x-grid-cell-text ' +
56135                     this.unselectableCls +  '" ' + this.unselectable +'" {attr}>{value}</div></div>',
56136                 "</td>"
56137             );
56138             tpls.cell.disableFormats = true;
56139         }
56140         tpls.cell.compile();
56141
56142         this.templates = tpls;
56143     },
56144
56145     // remap these for backwards compat
56146     onColWidthChange : function(){
56147         this.updateColumns.apply(this, arguments);
56148     },
56149     onHeaderChange : function(){
56150         this.updateHeaders.apply(this, arguments);
56151     }, 
56152     onHiddenChange : function(){
56153         this.handleHiddenChange.apply(this, arguments);
56154     },
56155     onColumnMove : function(){
56156         this.handleColumnMove.apply(this, arguments);
56157     },
56158     onColumnLock : function(){
56159         this.handleLockChange.apply(this, arguments);
56160     },
56161
56162     onDataChange : function(){
56163         this.refresh();
56164         this.updateHeaderSortState();
56165     },
56166
56167     onClear : function(){
56168         this.refresh();
56169     },
56170
56171     onUpdate : function(ds, record){
56172         this.refreshRow(record);
56173     },
56174
56175     refreshRow : function(record){
56176         var ds = this.ds, index;
56177         if(typeof record == 'number'){
56178             index = record;
56179             record = ds.getAt(index);
56180         }else{
56181             index = ds.indexOf(record);
56182         }
56183         this.insertRows(ds, index, index, true);
56184         this.onRemove(ds, record, index+1, true);
56185         this.syncRowHeights(index, index);
56186         this.layout();
56187         this.fireEvent("rowupdated", this, index, record);
56188     },
56189
56190     onAdd : function(ds, records, index){
56191         this.insertRows(ds, index, index + (records.length-1));
56192     },
56193
56194     onRemove : function(ds, record, index, isUpdate){
56195         if(isUpdate !== true){
56196             this.fireEvent("beforerowremoved", this, index, record);
56197         }
56198         var bt = this.getBodyTable(), lt = this.getLockedTable();
56199         if(bt.rows[index]){
56200             bt.firstChild.removeChild(bt.rows[index]);
56201         }
56202         if(lt.rows[index]){
56203             lt.firstChild.removeChild(lt.rows[index]);
56204         }
56205         if(isUpdate !== true){
56206             this.stripeRows(index);
56207             this.syncRowHeights(index, index);
56208             this.layout();
56209             this.fireEvent("rowremoved", this, index, record);
56210         }
56211     },
56212
56213     onLoad : function(){
56214         this.scrollToTop();
56215     },
56216
56217     /**
56218      * Scrolls the grid to the top
56219      */
56220     scrollToTop : function(){
56221         if(this.scroller){
56222             this.scroller.dom.scrollTop = 0;
56223             this.syncScroll();
56224         }
56225     },
56226
56227     /**
56228      * Gets a panel in the header of the grid that can be used for toolbars etc.
56229      * After modifying the contents of this panel a call to grid.autoSize() may be
56230      * required to register any changes in size.
56231      * @param {Boolean} doShow By default the header is hidden. Pass true to show the panel
56232      * @return Roo.Element
56233      */
56234     getHeaderPanel : function(doShow){
56235         if(doShow){
56236             this.headerPanel.show();
56237         }
56238         return this.headerPanel;
56239     },
56240
56241     /**
56242      * Gets a panel in the footer of the grid that can be used for toolbars etc.
56243      * After modifying the contents of this panel a call to grid.autoSize() may be
56244      * required to register any changes in size.
56245      * @param {Boolean} doShow By default the footer is hidden. Pass true to show the panel
56246      * @return Roo.Element
56247      */
56248     getFooterPanel : function(doShow){
56249         if(doShow){
56250             this.footerPanel.show();
56251         }
56252         return this.footerPanel;
56253     },
56254
56255     initElements : function(){
56256         var E = Roo.Element;
56257         var el = this.grid.getGridEl().dom.firstChild;
56258         var cs = el.childNodes;
56259
56260         this.el = new E(el);
56261         
56262          this.focusEl = new E(el.firstChild);
56263         this.focusEl.swallowEvent("click", true);
56264         
56265         this.headerPanel = new E(cs[1]);
56266         this.headerPanel.enableDisplayMode("block");
56267
56268         this.scroller = new E(cs[2]);
56269         this.scrollSizer = new E(this.scroller.dom.firstChild);
56270
56271         this.lockedWrap = new E(cs[3]);
56272         this.lockedHd = new E(this.lockedWrap.dom.firstChild);
56273         this.lockedBody = new E(this.lockedWrap.dom.childNodes[1]);
56274
56275         this.mainWrap = new E(cs[4]);
56276         this.mainHd = new E(this.mainWrap.dom.firstChild);
56277         this.mainBody = new E(this.mainWrap.dom.childNodes[1]);
56278
56279         this.footerPanel = new E(cs[5]);
56280         this.footerPanel.enableDisplayMode("block");
56281
56282         this.resizeProxy = new E(cs[6]);
56283
56284         this.headerSelector = String.format(
56285            '#{0} td.x-grid-hd, #{1} td.x-grid-hd',
56286            this.lockedHd.id, this.mainHd.id
56287         );
56288
56289         this.splitterSelector = String.format(
56290            '#{0} div.x-grid-split, #{1} div.x-grid-split',
56291            this.idToCssName(this.lockedHd.id), this.idToCssName(this.mainHd.id)
56292         );
56293     },
56294     idToCssName : function(s)
56295     {
56296         return s.replace(/[^a-z0-9]+/ig, '-');
56297     },
56298
56299     getHeaderCell : function(index){
56300         return Roo.DomQuery.select(this.headerSelector)[index];
56301     },
56302
56303     getHeaderCellMeasure : function(index){
56304         return this.getHeaderCell(index).firstChild;
56305     },
56306
56307     getHeaderCellText : function(index){
56308         return this.getHeaderCell(index).firstChild.firstChild;
56309     },
56310
56311     getLockedTable : function(){
56312         return this.lockedBody.dom.firstChild;
56313     },
56314
56315     getBodyTable : function(){
56316         return this.mainBody.dom.firstChild;
56317     },
56318
56319     getLockedRow : function(index){
56320         return this.getLockedTable().rows[index];
56321     },
56322
56323     getRow : function(index){
56324         return this.getBodyTable().rows[index];
56325     },
56326
56327     getRowComposite : function(index){
56328         if(!this.rowEl){
56329             this.rowEl = new Roo.CompositeElementLite();
56330         }
56331         var els = [], lrow, mrow;
56332         if(lrow = this.getLockedRow(index)){
56333             els.push(lrow);
56334         }
56335         if(mrow = this.getRow(index)){
56336             els.push(mrow);
56337         }
56338         this.rowEl.elements = els;
56339         return this.rowEl;
56340     },
56341     /**
56342      * Gets the 'td' of the cell
56343      * 
56344      * @param {Integer} rowIndex row to select
56345      * @param {Integer} colIndex column to select
56346      * 
56347      * @return {Object} 
56348      */
56349     getCell : function(rowIndex, colIndex){
56350         var locked = this.cm.getLockedCount();
56351         var source;
56352         if(colIndex < locked){
56353             source = this.lockedBody.dom.firstChild;
56354         }else{
56355             source = this.mainBody.dom.firstChild;
56356             colIndex -= locked;
56357         }
56358         return source.rows[rowIndex].childNodes[colIndex];
56359     },
56360
56361     getCellText : function(rowIndex, colIndex){
56362         return this.getCell(rowIndex, colIndex).firstChild.firstChild;
56363     },
56364
56365     getCellBox : function(cell){
56366         var b = this.fly(cell).getBox();
56367         if(Roo.isOpera){ // opera fails to report the Y
56368             b.y = cell.offsetTop + this.mainBody.getY();
56369         }
56370         return b;
56371     },
56372
56373     getCellIndex : function(cell){
56374         var id = String(cell.className).match(this.cellRE);
56375         if(id){
56376             return parseInt(id[1], 10);
56377         }
56378         return 0;
56379     },
56380
56381     findHeaderIndex : function(n){
56382         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56383         return r ? this.getCellIndex(r) : false;
56384     },
56385
56386     findHeaderCell : function(n){
56387         var r = Roo.fly(n).findParent("td." + this.hdClass, 6);
56388         return r ? r : false;
56389     },
56390
56391     findRowIndex : function(n){
56392         if(!n){
56393             return false;
56394         }
56395         var r = Roo.fly(n).findParent("tr." + this.rowClass, 6);
56396         return r ? r.rowIndex : false;
56397     },
56398
56399     findCellIndex : function(node){
56400         var stop = this.el.dom;
56401         while(node && node != stop){
56402             if(this.findRE.test(node.className)){
56403                 return this.getCellIndex(node);
56404             }
56405             node = node.parentNode;
56406         }
56407         return false;
56408     },
56409
56410     getColumnId : function(index){
56411         return this.cm.getColumnId(index);
56412     },
56413
56414     getSplitters : function()
56415     {
56416         if(this.splitterSelector){
56417            return Roo.DomQuery.select(this.splitterSelector);
56418         }else{
56419             return null;
56420       }
56421     },
56422
56423     getSplitter : function(index){
56424         return this.getSplitters()[index];
56425     },
56426
56427     onRowOver : function(e, t){
56428         var row;
56429         if((row = this.findRowIndex(t)) !== false){
56430             this.getRowComposite(row).addClass("x-grid-row-over");
56431         }
56432     },
56433
56434     onRowOut : function(e, t){
56435         var row;
56436         if((row = this.findRowIndex(t)) !== false && row !== this.findRowIndex(e.getRelatedTarget())){
56437             this.getRowComposite(row).removeClass("x-grid-row-over");
56438         }
56439     },
56440
56441     renderHeaders : function(){
56442         var cm = this.cm;
56443         var ct = this.templates.hcell, ht = this.templates.header, st = this.templates.hsplit;
56444         var cb = [], lb = [], sb = [], lsb = [], p = {};
56445         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56446             p.cellId = "x-grid-hd-0-" + i;
56447             p.splitId = "x-grid-csplit-0-" + i;
56448             p.id = cm.getColumnId(i);
56449             p.value = cm.getColumnHeader(i) || "";
56450             p.title = cm.getColumnTooltip(i) || (''+p.value).match(/\</)  ? '' :  p.value  || "";
56451             p.style = (this.grid.enableColumnResize === false || !cm.isResizable(i) || cm.isFixed(i)) ? 'cursor:default' : '';
56452             if(!cm.isLocked(i)){
56453                 cb[cb.length] = ct.apply(p);
56454                 sb[sb.length] = st.apply(p);
56455             }else{
56456                 lb[lb.length] = ct.apply(p);
56457                 lsb[lsb.length] = st.apply(p);
56458             }
56459         }
56460         return [ht.apply({cells: lb.join(""), splits:lsb.join("")}),
56461                 ht.apply({cells: cb.join(""), splits:sb.join("")})];
56462     },
56463
56464     updateHeaders : function(){
56465         var html = this.renderHeaders();
56466         this.lockedHd.update(html[0]);
56467         this.mainHd.update(html[1]);
56468     },
56469
56470     /**
56471      * Focuses the specified row.
56472      * @param {Number} row The row index
56473      */
56474     focusRow : function(row)
56475     {
56476         //Roo.log('GridView.focusRow');
56477         var x = this.scroller.dom.scrollLeft;
56478         this.focusCell(row, 0, false);
56479         this.scroller.dom.scrollLeft = x;
56480     },
56481
56482     /**
56483      * Focuses the specified cell.
56484      * @param {Number} row The row index
56485      * @param {Number} col The column index
56486      * @param {Boolean} hscroll false to disable horizontal scrolling
56487      */
56488     focusCell : function(row, col, hscroll)
56489     {
56490         //Roo.log('GridView.focusCell');
56491         var el = this.ensureVisible(row, col, hscroll);
56492         this.focusEl.alignTo(el, "tl-tl");
56493         if(Roo.isGecko){
56494             this.focusEl.focus();
56495         }else{
56496             this.focusEl.focus.defer(1, this.focusEl);
56497         }
56498     },
56499
56500     /**
56501      * Scrolls the specified cell into view
56502      * @param {Number} row The row index
56503      * @param {Number} col The column index
56504      * @param {Boolean} hscroll false to disable horizontal scrolling
56505      */
56506     ensureVisible : function(row, col, hscroll)
56507     {
56508         //Roo.log('GridView.ensureVisible,' + row + ',' + col);
56509         //return null; //disable for testing.
56510         if(typeof row != "number"){
56511             row = row.rowIndex;
56512         }
56513         if(row < 0 && row >= this.ds.getCount()){
56514             return  null;
56515         }
56516         col = (col !== undefined ? col : 0);
56517         var cm = this.grid.colModel;
56518         while(cm.isHidden(col)){
56519             col++;
56520         }
56521
56522         var el = this.getCell(row, col);
56523         if(!el){
56524             return null;
56525         }
56526         var c = this.scroller.dom;
56527
56528         var ctop = parseInt(el.offsetTop, 10);
56529         var cleft = parseInt(el.offsetLeft, 10);
56530         var cbot = ctop + el.offsetHeight;
56531         var cright = cleft + el.offsetWidth;
56532         
56533         var ch = c.clientHeight - this.mainHd.dom.offsetHeight;
56534         var stop = parseInt(c.scrollTop, 10);
56535         var sleft = parseInt(c.scrollLeft, 10);
56536         var sbot = stop + ch;
56537         var sright = sleft + c.clientWidth;
56538         /*
56539         Roo.log('GridView.ensureVisible:' +
56540                 ' ctop:' + ctop +
56541                 ' c.clientHeight:' + c.clientHeight +
56542                 ' this.mainHd.dom.offsetHeight:' + this.mainHd.dom.offsetHeight +
56543                 ' stop:' + stop +
56544                 ' cbot:' + cbot +
56545                 ' sbot:' + sbot +
56546                 ' ch:' + ch  
56547                 );
56548         */
56549         if(ctop < stop){
56550              c.scrollTop = ctop;
56551             //Roo.log("set scrolltop to ctop DISABLE?");
56552         }else if(cbot > sbot){
56553             //Roo.log("set scrolltop to cbot-ch");
56554             c.scrollTop = cbot-ch;
56555         }
56556         
56557         if(hscroll !== false){
56558             if(cleft < sleft){
56559                 c.scrollLeft = cleft;
56560             }else if(cright > sright){
56561                 c.scrollLeft = cright-c.clientWidth;
56562             }
56563         }
56564          
56565         return el;
56566     },
56567
56568     updateColumns : function(){
56569         this.grid.stopEditing();
56570         var cm = this.grid.colModel, colIds = this.getColumnIds();
56571         //var totalWidth = cm.getTotalWidth();
56572         var pos = 0;
56573         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56574             //if(cm.isHidden(i)) continue;
56575             var w = cm.getColumnWidth(i);
56576             this.css.updateRule(this.colSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56577             this.css.updateRule(this.hdSelector+this.idToCssName(colIds[i]), "width", (w - this.borderWidth) + "px");
56578         }
56579         this.updateSplitters();
56580     },
56581
56582     generateRules : function(cm){
56583         var ruleBuf = [], rulesId = this.idToCssName(this.grid.id)+ '-cssrules';
56584         Roo.util.CSS.removeStyleSheet(rulesId);
56585         for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56586             var cid = cm.getColumnId(i);
56587             var align = '';
56588             if(cm.config[i].align){
56589                 align = 'text-align:'+cm.config[i].align+';';
56590             }
56591             var hidden = '';
56592             if(cm.isHidden(i)){
56593                 hidden = 'display:none;';
56594             }
56595             var width = "width:" + (cm.getColumnWidth(i) - this.borderWidth) + "px;";
56596             ruleBuf.push(
56597                     this.colSelector, cid, " {\n", cm.config[i].css, align, width, "\n}\n",
56598                     this.hdSelector, cid, " {\n", align, width, "}\n",
56599                     this.tdSelector, cid, " {\n",hidden,"\n}\n",
56600                     this.splitSelector, cid, " {\n", hidden , "\n}\n");
56601         }
56602         return Roo.util.CSS.createStyleSheet(ruleBuf.join(""), rulesId);
56603     },
56604
56605     updateSplitters : function(){
56606         var cm = this.cm, s = this.getSplitters();
56607         if(s){ // splitters not created yet
56608             var pos = 0, locked = true;
56609             for(var i = 0, len = cm.getColumnCount(); i < len; i++){
56610                 if(cm.isHidden(i)) {
56611                     continue;
56612                 }
56613                 var w = cm.getColumnWidth(i); // make sure it's a number
56614                 if(!cm.isLocked(i) && locked){
56615                     pos = 0;
56616                     locked = false;
56617                 }
56618                 pos += w;
56619                 s[i].style.left = (pos-this.splitOffset) + "px";
56620             }
56621         }
56622     },
56623
56624     handleHiddenChange : function(colModel, colIndex, hidden){
56625         if(hidden){
56626             this.hideColumn(colIndex);
56627         }else{
56628             this.unhideColumn(colIndex);
56629         }
56630     },
56631
56632     hideColumn : function(colIndex){
56633         var cid = this.getColumnId(colIndex);
56634         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "none");
56635         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "none");
56636         if(Roo.isSafari){
56637             this.updateHeaders();
56638         }
56639         this.updateSplitters();
56640         this.layout();
56641     },
56642
56643     unhideColumn : function(colIndex){
56644         var cid = this.getColumnId(colIndex);
56645         this.css.updateRule(this.tdSelector+this.idToCssName(cid), "display", "");
56646         this.css.updateRule(this.splitSelector+this.idToCssName(cid), "display", "");
56647
56648         if(Roo.isSafari){
56649             this.updateHeaders();
56650         }
56651         this.updateSplitters();
56652         this.layout();
56653     },
56654
56655     insertRows : function(dm, firstRow, lastRow, isUpdate){
56656         if(firstRow == 0 && lastRow == dm.getCount()-1){
56657             this.refresh();
56658         }else{
56659             if(!isUpdate){
56660                 this.fireEvent("beforerowsinserted", this, firstRow, lastRow);
56661             }
56662             var s = this.getScrollState();
56663             var markup = this.renderRows(firstRow, lastRow);
56664             this.bufferRows(markup[0], this.getLockedTable(), firstRow);
56665             this.bufferRows(markup[1], this.getBodyTable(), firstRow);
56666             this.restoreScroll(s);
56667             if(!isUpdate){
56668                 this.fireEvent("rowsinserted", this, firstRow, lastRow);
56669                 this.syncRowHeights(firstRow, lastRow);
56670                 this.stripeRows(firstRow);
56671                 this.layout();
56672             }
56673         }
56674     },
56675
56676     bufferRows : function(markup, target, index){
56677         var before = null, trows = target.rows, tbody = target.tBodies[0];
56678         if(index < trows.length){
56679             before = trows[index];
56680         }
56681         var b = document.createElement("div");
56682         b.innerHTML = "<table><tbody>"+markup+"</tbody></table>";
56683         var rows = b.firstChild.rows;
56684         for(var i = 0, len = rows.length; i < len; i++){
56685             if(before){
56686                 tbody.insertBefore(rows[0], before);
56687             }else{
56688                 tbody.appendChild(rows[0]);
56689             }
56690         }
56691         b.innerHTML = "";
56692         b = null;
56693     },
56694
56695     deleteRows : function(dm, firstRow, lastRow){
56696         if(dm.getRowCount()<1){
56697             this.fireEvent("beforerefresh", this);
56698             this.mainBody.update("");
56699             this.lockedBody.update("");
56700             this.fireEvent("refresh", this);
56701         }else{
56702             this.fireEvent("beforerowsdeleted", this, firstRow, lastRow);
56703             var bt = this.getBodyTable();
56704             var tbody = bt.firstChild;
56705             var rows = bt.rows;
56706             for(var rowIndex = firstRow; rowIndex <= lastRow; rowIndex++){
56707                 tbody.removeChild(rows[firstRow]);
56708             }
56709             this.stripeRows(firstRow);
56710             this.fireEvent("rowsdeleted", this, firstRow, lastRow);
56711         }
56712     },
56713
56714     updateRows : function(dataSource, firstRow, lastRow){
56715         var s = this.getScrollState();
56716         this.refresh();
56717         this.restoreScroll(s);
56718     },
56719
56720     handleSort : function(dataSource, sortColumnIndex, sortDir, noRefresh){
56721         if(!noRefresh){
56722            this.refresh();
56723         }
56724         this.updateHeaderSortState();
56725     },
56726
56727     getScrollState : function(){
56728         
56729         var sb = this.scroller.dom;
56730         return {left: sb.scrollLeft, top: sb.scrollTop};
56731     },
56732
56733     stripeRows : function(startRow){
56734         if(!this.grid.stripeRows || this.ds.getCount() < 1){
56735             return;
56736         }
56737         startRow = startRow || 0;
56738         var rows = this.getBodyTable().rows;
56739         var lrows = this.getLockedTable().rows;
56740         var cls = ' x-grid-row-alt ';
56741         for(var i = startRow, len = rows.length; i < len; i++){
56742             var row = rows[i], lrow = lrows[i];
56743             var isAlt = ((i+1) % 2 == 0);
56744             var hasAlt = (' '+row.className + ' ').indexOf(cls) != -1;
56745             if(isAlt == hasAlt){
56746                 continue;
56747             }
56748             if(isAlt){
56749                 row.className += " x-grid-row-alt";
56750             }else{
56751                 row.className = row.className.replace("x-grid-row-alt", "");
56752             }
56753             if(lrow){
56754                 lrow.className = row.className;
56755             }
56756         }
56757     },
56758
56759     restoreScroll : function(state){
56760         //Roo.log('GridView.restoreScroll');
56761         var sb = this.scroller.dom;
56762         sb.scrollLeft = state.left;
56763         sb.scrollTop = state.top;
56764         this.syncScroll();
56765     },
56766
56767     syncScroll : function(){
56768         //Roo.log('GridView.syncScroll');
56769         var sb = this.scroller.dom;
56770         var sh = this.mainHd.dom;
56771         var bs = this.mainBody.dom;
56772         var lv = this.lockedBody.dom;
56773         sh.scrollLeft = bs.scrollLeft = sb.scrollLeft;
56774         lv.scrollTop = bs.scrollTop = sb.scrollTop;
56775     },
56776
56777     handleScroll : function(e){
56778         this.syncScroll();
56779         var sb = this.scroller.dom;
56780         this.grid.fireEvent("bodyscroll", sb.scrollLeft, sb.scrollTop);
56781         e.stopEvent();
56782     },
56783
56784     handleWheel : function(e){
56785         var d = e.getWheelDelta();
56786         this.scroller.dom.scrollTop -= d*22;
56787         // set this here to prevent jumpy scrolling on large tables
56788         this.lockedBody.dom.scrollTop = this.mainBody.dom.scrollTop = this.scroller.dom.scrollTop;
56789         e.stopEvent();
56790     },
56791
56792     renderRows : function(startRow, endRow){
56793         // pull in all the crap needed to render rows
56794         var g = this.grid, cm = g.colModel, ds = g.dataSource, stripe = g.stripeRows;
56795         var colCount = cm.getColumnCount();
56796
56797         if(ds.getCount() < 1){
56798             return ["", ""];
56799         }
56800
56801         // build a map for all the columns
56802         var cs = [];
56803         for(var i = 0; i < colCount; i++){
56804             var name = cm.getDataIndex(i);
56805             cs[i] = {
56806                 name : typeof name == 'undefined' ? ds.fields.get(i).name : name,
56807                 renderer : cm.getRenderer(i),
56808                 id : cm.getColumnId(i),
56809                 locked : cm.isLocked(i),
56810                 has_editor : cm.isCellEditable(i)
56811             };
56812         }
56813
56814         startRow = startRow || 0;
56815         endRow = typeof endRow == "undefined"? ds.getCount()-1 : endRow;
56816
56817         // records to render
56818         var rs = ds.getRange(startRow, endRow);
56819
56820         return this.doRender(cs, rs, ds, startRow, colCount, stripe);
56821     },
56822
56823     // As much as I hate to duplicate code, this was branched because FireFox really hates
56824     // [].join("") on strings. The performance difference was substantial enough to
56825     // branch this function
56826     doRender : Roo.isGecko ?
56827             function(cs, rs, ds, startRow, colCount, stripe){
56828                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56829                 // buffers
56830                 var buf = "", lbuf = "", cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56831                 
56832                 var hasListener = this.grid.hasListener('rowclass');
56833                 var rowcfg = {};
56834                 for(var j = 0, len = rs.length; j < len; j++){
56835                     r = rs[j]; cb = ""; lcb = ""; rowIndex = (j+startRow);
56836                     for(var i = 0; i < colCount; i++){
56837                         c = cs[i];
56838                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56839                         p.id = c.id;
56840                         p.css = p.attr = "";
56841                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56842                         if(p.value == undefined || p.value === "") {
56843                             p.value = "&#160;";
56844                         }
56845                         if(c.has_editor){
56846                             p.css += ' x-grid-editable-cell';
56847                         }
56848                         if(c.dirty && typeof r.modified[c.name] !== 'undefined'){
56849                             p.css +=  ' x-grid-dirty-cell';
56850                         }
56851                         var markup = ct.apply(p);
56852                         if(!c.locked){
56853                             cb+= markup;
56854                         }else{
56855                             lcb+= markup;
56856                         }
56857                     }
56858                     var alt = [];
56859                     if(stripe && ((rowIndex+1) % 2 == 0)){
56860                         alt.push("x-grid-row-alt")
56861                     }
56862                     if(r.dirty){
56863                         alt.push(  " x-grid-dirty-row");
56864                     }
56865                     rp.cells = lcb;
56866                     if(this.getRowClass){
56867                         alt.push(this.getRowClass(r, rowIndex));
56868                     }
56869                     if (hasListener) {
56870                         rowcfg = {
56871                              
56872                             record: r,
56873                             rowIndex : rowIndex,
56874                             rowClass : ''
56875                         };
56876                         this.grid.fireEvent('rowclass', this, rowcfg);
56877                         alt.push(rowcfg.rowClass);
56878                     }
56879                     rp.alt = alt.join(" ");
56880                     lbuf+= rt.apply(rp);
56881                     rp.cells = cb;
56882                     buf+=  rt.apply(rp);
56883                 }
56884                 return [lbuf, buf];
56885             } :
56886             function(cs, rs, ds, startRow, colCount, stripe){
56887                 var ts = this.templates, ct = ts.cell, rt = ts.row;
56888                 // buffers
56889                 var buf = [], lbuf = [], cb, lcb, c, p = {}, rp = {}, r, rowIndex;
56890                 var hasListener = this.grid.hasListener('rowclass');
56891  
56892                 var rowcfg = {};
56893                 for(var j = 0, len = rs.length; j < len; j++){
56894                     r = rs[j]; cb = []; lcb = []; rowIndex = (j+startRow);
56895                     for(var i = 0; i < colCount; i++){
56896                         c = cs[i];
56897                         p.cellId = "x-grid-cell-" + rowIndex + "-" + i;
56898                         p.id = c.id;
56899                         p.css = p.attr = "";
56900                         p.value = c.renderer(r.data[c.name], p, r, rowIndex, i, ds);
56901                         if(p.value == undefined || p.value === "") {
56902                             p.value = "&#160;";
56903                         }
56904                         //Roo.log(c);
56905                          if(c.has_editor){
56906                             p.css += ' x-grid-editable-cell';
56907                         }
56908                         if(r.dirty && typeof r.modified[c.name] !== 'undefined'){
56909                             p.css += ' x-grid-dirty-cell' 
56910                         }
56911                         
56912                         var markup = ct.apply(p);
56913                         if(!c.locked){
56914                             cb[cb.length] = markup;
56915                         }else{
56916                             lcb[lcb.length] = markup;
56917                         }
56918                     }
56919                     var alt = [];
56920                     if(stripe && ((rowIndex+1) % 2 == 0)){
56921                         alt.push( "x-grid-row-alt");
56922                     }
56923                     if(r.dirty){
56924                         alt.push(" x-grid-dirty-row");
56925                     }
56926                     rp.cells = lcb;
56927                     if(this.getRowClass){
56928                         alt.push( this.getRowClass(r, rowIndex));
56929                     }
56930                     if (hasListener) {
56931                         rowcfg = {
56932                              
56933                             record: r,
56934                             rowIndex : rowIndex,
56935                             rowClass : ''
56936                         };
56937                         this.grid.fireEvent('rowclass', this, rowcfg);
56938                         alt.push(rowcfg.rowClass);
56939                     }
56940                     
56941                     rp.alt = alt.join(" ");
56942                     rp.cells = lcb.join("");
56943                     lbuf[lbuf.length] = rt.apply(rp);
56944                     rp.cells = cb.join("");
56945                     buf[buf.length] =  rt.apply(rp);
56946                 }
56947                 return [lbuf.join(""), buf.join("")];
56948             },
56949
56950     renderBody : function(){
56951         var markup = this.renderRows();
56952         var bt = this.templates.body;
56953         return [bt.apply({rows: markup[0]}), bt.apply({rows: markup[1]})];
56954     },
56955
56956     /**
56957      * Refreshes the grid
56958      * @param {Boolean} headersToo
56959      */
56960     refresh : function(headersToo){
56961         this.fireEvent("beforerefresh", this);
56962         this.grid.stopEditing();
56963         var result = this.renderBody();
56964         this.lockedBody.update(result[0]);
56965         this.mainBody.update(result[1]);
56966         if(headersToo === true){
56967             this.updateHeaders();
56968             this.updateColumns();
56969             this.updateSplitters();
56970             this.updateHeaderSortState();
56971         }
56972         this.syncRowHeights();
56973         this.layout();
56974         this.fireEvent("refresh", this);
56975     },
56976
56977     handleColumnMove : function(cm, oldIndex, newIndex){
56978         this.indexMap = null;
56979         var s = this.getScrollState();
56980         this.refresh(true);
56981         this.restoreScroll(s);
56982         this.afterMove(newIndex);
56983     },
56984
56985     afterMove : function(colIndex){
56986         if(this.enableMoveAnim && Roo.enableFx){
56987             this.fly(this.getHeaderCell(colIndex).firstChild).highlight(this.hlColor);
56988         }
56989         // if multisort - fix sortOrder, and reload..
56990         if (this.grid.dataSource.multiSort) {
56991             // the we can call sort again..
56992             var dm = this.grid.dataSource;
56993             var cm = this.grid.colModel;
56994             var so = [];
56995             for(var i = 0; i < cm.config.length; i++ ) {
56996                 
56997                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined')) {
56998                     continue; // dont' bother, it's not in sort list or being set.
56999                 }
57000                 
57001                 so.push(cm.config[i].dataIndex);
57002             };
57003             dm.sortOrder = so;
57004             dm.load(dm.lastOptions);
57005             
57006             
57007         }
57008         
57009     },
57010
57011     updateCell : function(dm, rowIndex, dataIndex){
57012         var colIndex = this.getColumnIndexByDataIndex(dataIndex);
57013         if(typeof colIndex == "undefined"){ // not present in grid
57014             return;
57015         }
57016         var cm = this.grid.colModel;
57017         var cell = this.getCell(rowIndex, colIndex);
57018         var cellText = this.getCellText(rowIndex, colIndex);
57019
57020         var p = {
57021             cellId : "x-grid-cell-" + rowIndex + "-" + colIndex,
57022             id : cm.getColumnId(colIndex),
57023             css: colIndex == cm.getColumnCount()-1 ? "x-grid-col-last" : ""
57024         };
57025         var renderer = cm.getRenderer(colIndex);
57026         var val = renderer(dm.getValueAt(rowIndex, dataIndex), p, rowIndex, colIndex, dm);
57027         if(typeof val == "undefined" || val === "") {
57028             val = "&#160;";
57029         }
57030         cellText.innerHTML = val;
57031         cell.className = this.cellClass + " " + this.idToCssName(p.cellId) + " " + p.css;
57032         this.syncRowHeights(rowIndex, rowIndex);
57033     },
57034
57035     calcColumnWidth : function(colIndex, maxRowsToMeasure){
57036         var maxWidth = 0;
57037         if(this.grid.autoSizeHeaders){
57038             var h = this.getHeaderCellMeasure(colIndex);
57039             maxWidth = Math.max(maxWidth, h.scrollWidth);
57040         }
57041         var tb, index;
57042         if(this.cm.isLocked(colIndex)){
57043             tb = this.getLockedTable();
57044             index = colIndex;
57045         }else{
57046             tb = this.getBodyTable();
57047             index = colIndex - this.cm.getLockedCount();
57048         }
57049         if(tb && tb.rows){
57050             var rows = tb.rows;
57051             var stopIndex = Math.min(maxRowsToMeasure || rows.length, rows.length);
57052             for(var i = 0; i < stopIndex; i++){
57053                 var cell = rows[i].childNodes[index].firstChild;
57054                 maxWidth = Math.max(maxWidth, cell.scrollWidth);
57055             }
57056         }
57057         return maxWidth + /*margin for error in IE*/ 5;
57058     },
57059     /**
57060      * Autofit a column to its content.
57061      * @param {Number} colIndex
57062      * @param {Boolean} forceMinSize true to force the column to go smaller if possible
57063      */
57064      autoSizeColumn : function(colIndex, forceMinSize, suppressEvent){
57065          if(this.cm.isHidden(colIndex)){
57066              return; // can't calc a hidden column
57067          }
57068         if(forceMinSize){
57069             var cid = this.cm.getColumnId(colIndex);
57070             this.css.updateRule(this.colSelector +this.idToCssName( cid), "width", this.grid.minColumnWidth + "px");
57071            if(this.grid.autoSizeHeaders){
57072                this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", this.grid.minColumnWidth + "px");
57073            }
57074         }
57075         var newWidth = this.calcColumnWidth(colIndex);
57076         this.cm.setColumnWidth(colIndex,
57077             Math.max(this.grid.minColumnWidth, newWidth), suppressEvent);
57078         if(!suppressEvent){
57079             this.grid.fireEvent("columnresize", colIndex, newWidth);
57080         }
57081     },
57082
57083     /**
57084      * Autofits all columns to their content and then expands to fit any extra space in the grid
57085      */
57086      autoSizeColumns : function(){
57087         var cm = this.grid.colModel;
57088         var colCount = cm.getColumnCount();
57089         for(var i = 0; i < colCount; i++){
57090             this.autoSizeColumn(i, true, true);
57091         }
57092         if(cm.getTotalWidth() < this.scroller.dom.clientWidth){
57093             this.fitColumns();
57094         }else{
57095             this.updateColumns();
57096             this.layout();
57097         }
57098     },
57099
57100     /**
57101      * Autofits all columns to the grid's width proportionate with their current size
57102      * @param {Boolean} reserveScrollSpace Reserve space for a scrollbar
57103      */
57104     fitColumns : function(reserveScrollSpace){
57105         var cm = this.grid.colModel;
57106         var colCount = cm.getColumnCount();
57107         var cols = [];
57108         var width = 0;
57109         var i, w;
57110         for (i = 0; i < colCount; i++){
57111             if(!cm.isHidden(i) && !cm.isFixed(i)){
57112                 w = cm.getColumnWidth(i);
57113                 cols.push(i);
57114                 cols.push(w);
57115                 width += w;
57116             }
57117         }
57118         var avail = Math.min(this.scroller.dom.clientWidth, this.el.getWidth());
57119         if(reserveScrollSpace){
57120             avail -= 17;
57121         }
57122         var frac = (avail - cm.getTotalWidth())/width;
57123         while (cols.length){
57124             w = cols.pop();
57125             i = cols.pop();
57126             cm.setColumnWidth(i, Math.floor(w + w*frac), true);
57127         }
57128         this.updateColumns();
57129         this.layout();
57130     },
57131
57132     onRowSelect : function(rowIndex){
57133         var row = this.getRowComposite(rowIndex);
57134         row.addClass("x-grid-row-selected");
57135     },
57136
57137     onRowDeselect : function(rowIndex){
57138         var row = this.getRowComposite(rowIndex);
57139         row.removeClass("x-grid-row-selected");
57140     },
57141
57142     onCellSelect : function(row, col){
57143         var cell = this.getCell(row, col);
57144         if(cell){
57145             Roo.fly(cell).addClass("x-grid-cell-selected");
57146         }
57147     },
57148
57149     onCellDeselect : function(row, col){
57150         var cell = this.getCell(row, col);
57151         if(cell){
57152             Roo.fly(cell).removeClass("x-grid-cell-selected");
57153         }
57154     },
57155
57156     updateHeaderSortState : function(){
57157         
57158         // sort state can be single { field: xxx, direction : yyy}
57159         // or   { xxx=>ASC , yyy : DESC ..... }
57160         
57161         var mstate = {};
57162         if (!this.ds.multiSort) { 
57163             var state = this.ds.getSortState();
57164             if(!state){
57165                 return;
57166             }
57167             mstate[state.field] = state.direction;
57168             // FIXME... - this is not used here.. but might be elsewhere..
57169             this.sortState = state;
57170             
57171         } else {
57172             mstate = this.ds.sortToggle;
57173         }
57174         //remove existing sort classes..
57175         
57176         var sc = this.sortClasses;
57177         var hds = this.el.select(this.headerSelector).removeClass(sc);
57178         
57179         for(var f in mstate) {
57180         
57181             var sortColumn = this.cm.findColumnIndex(f);
57182             
57183             if(sortColumn != -1){
57184                 var sortDir = mstate[f];        
57185                 hds.item(sortColumn).addClass(sc[sortDir == "DESC" ? 1 : 0]);
57186             }
57187         }
57188         
57189          
57190         
57191     },
57192
57193
57194     handleHeaderClick : function(g, index,e){
57195         
57196         Roo.log("header click");
57197         
57198         if (Roo.isTouch) {
57199             // touch events on header are handled by context
57200             this.handleHdCtx(g,index,e);
57201             return;
57202         }
57203         
57204         
57205         if(this.headersDisabled){
57206             return;
57207         }
57208         var dm = g.dataSource, cm = g.colModel;
57209         if(!cm.isSortable(index)){
57210             return;
57211         }
57212         g.stopEditing();
57213         
57214         if (dm.multiSort) {
57215             // update the sortOrder
57216             var so = [];
57217             for(var i = 0; i < cm.config.length; i++ ) {
57218                 
57219                 if ((typeof(dm.sortToggle[cm.config[i].dataIndex]) == 'undefined') && (index != i)) {
57220                     continue; // dont' bother, it's not in sort list or being set.
57221                 }
57222                 
57223                 so.push(cm.config[i].dataIndex);
57224             };
57225             dm.sortOrder = so;
57226         }
57227         
57228         
57229         dm.sort(cm.getDataIndex(index));
57230     },
57231
57232
57233     destroy : function(){
57234         if(this.colMenu){
57235             this.colMenu.removeAll();
57236             Roo.menu.MenuMgr.unregister(this.colMenu);
57237             this.colMenu.getEl().remove();
57238             delete this.colMenu;
57239         }
57240         if(this.hmenu){
57241             this.hmenu.removeAll();
57242             Roo.menu.MenuMgr.unregister(this.hmenu);
57243             this.hmenu.getEl().remove();
57244             delete this.hmenu;
57245         }
57246         if(this.grid.enableColumnMove){
57247             var dds = Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57248             if(dds){
57249                 for(var dd in dds){
57250                     if(!dds[dd].config.isTarget && dds[dd].dragElId){
57251                         var elid = dds[dd].dragElId;
57252                         dds[dd].unreg();
57253                         Roo.get(elid).remove();
57254                     } else if(dds[dd].config.isTarget){
57255                         dds[dd].proxyTop.remove();
57256                         dds[dd].proxyBottom.remove();
57257                         dds[dd].unreg();
57258                     }
57259                     if(Roo.dd.DDM.locationCache[dd]){
57260                         delete Roo.dd.DDM.locationCache[dd];
57261                     }
57262                 }
57263                 delete Roo.dd.DDM.ids['gridHeader' + this.grid.getGridEl().id];
57264             }
57265         }
57266         Roo.util.CSS.removeStyleSheet(this.idToCssName(this.grid.id) + '-cssrules');
57267         this.bind(null, null);
57268         Roo.EventManager.removeResizeListener(this.onWindowResize, this);
57269     },
57270
57271     handleLockChange : function(){
57272         this.refresh(true);
57273     },
57274
57275     onDenyColumnLock : function(){
57276
57277     },
57278
57279     onDenyColumnHide : function(){
57280
57281     },
57282
57283     handleHdMenuClick : function(item){
57284         var index = this.hdCtxIndex;
57285         var cm = this.cm, ds = this.ds;
57286         switch(item.id){
57287             case "asc":
57288                 ds.sort(cm.getDataIndex(index), "ASC");
57289                 break;
57290             case "desc":
57291                 ds.sort(cm.getDataIndex(index), "DESC");
57292                 break;
57293             case "lock":
57294                 var lc = cm.getLockedCount();
57295                 if(cm.getColumnCount(true) <= lc+1){
57296                     this.onDenyColumnLock();
57297                     return;
57298                 }
57299                 if(lc != index){
57300                     cm.setLocked(index, true, true);
57301                     cm.moveColumn(index, lc);
57302                     this.grid.fireEvent("columnmove", index, lc);
57303                 }else{
57304                     cm.setLocked(index, true);
57305                 }
57306             break;
57307             case "unlock":
57308                 var lc = cm.getLockedCount();
57309                 if((lc-1) != index){
57310                     cm.setLocked(index, false, true);
57311                     cm.moveColumn(index, lc-1);
57312                     this.grid.fireEvent("columnmove", index, lc-1);
57313                 }else{
57314                     cm.setLocked(index, false);
57315                 }
57316             break;
57317             case 'wider': // used to expand cols on touch..
57318             case 'narrow':
57319                 var cw = cm.getColumnWidth(index);
57320                 cw += (item.id == 'wider' ? 1 : -1) * 50;
57321                 cw = Math.max(0, cw);
57322                 cw = Math.min(cw,4000);
57323                 cm.setColumnWidth(index, cw);
57324                 break;
57325                 
57326             default:
57327                 index = cm.getIndexById(item.id.substr(4));
57328                 if(index != -1){
57329                     if(item.checked && cm.getColumnCount(true) <= 1){
57330                         this.onDenyColumnHide();
57331                         return false;
57332                     }
57333                     cm.setHidden(index, item.checked);
57334                 }
57335         }
57336         return true;
57337     },
57338
57339     beforeColMenuShow : function(){
57340         var cm = this.cm,  colCount = cm.getColumnCount();
57341         this.colMenu.removeAll();
57342         for(var i = 0; i < colCount; i++){
57343             this.colMenu.add(new Roo.menu.CheckItem({
57344                 id: "col-"+cm.getColumnId(i),
57345                 text: cm.getColumnHeader(i),
57346                 checked: !cm.isHidden(i),
57347                 hideOnClick:false
57348             }));
57349         }
57350     },
57351
57352     handleHdCtx : function(g, index, e){
57353         e.stopEvent();
57354         var hd = this.getHeaderCell(index);
57355         this.hdCtxIndex = index;
57356         var ms = this.hmenu.items, cm = this.cm;
57357         ms.get("asc").setDisabled(!cm.isSortable(index));
57358         ms.get("desc").setDisabled(!cm.isSortable(index));
57359         if(this.grid.enableColLock !== false){
57360             ms.get("lock").setDisabled(cm.isLocked(index));
57361             ms.get("unlock").setDisabled(!cm.isLocked(index));
57362         }
57363         this.hmenu.show(hd, "tl-bl");
57364     },
57365
57366     handleHdOver : function(e){
57367         var hd = this.findHeaderCell(e.getTarget());
57368         if(hd && !this.headersDisabled){
57369             if(this.grid.colModel.isSortable(this.getCellIndex(hd))){
57370                this.fly(hd).addClass("x-grid-hd-over");
57371             }
57372         }
57373     },
57374
57375     handleHdOut : function(e){
57376         var hd = this.findHeaderCell(e.getTarget());
57377         if(hd){
57378             this.fly(hd).removeClass("x-grid-hd-over");
57379         }
57380     },
57381
57382     handleSplitDblClick : function(e, t){
57383         var i = this.getCellIndex(t);
57384         if(this.grid.enableColumnResize !== false && this.cm.isResizable(i) && !this.cm.isFixed(i)){
57385             this.autoSizeColumn(i, true);
57386             this.layout();
57387         }
57388     },
57389
57390     render : function(){
57391
57392         var cm = this.cm;
57393         var colCount = cm.getColumnCount();
57394
57395         if(this.grid.monitorWindowResize === true){
57396             Roo.EventManager.onWindowResize(this.onWindowResize, this, true);
57397         }
57398         var header = this.renderHeaders();
57399         var body = this.templates.body.apply({rows:""});
57400         var html = this.templates.master.apply({
57401             lockedBody: body,
57402             body: body,
57403             lockedHeader: header[0],
57404             header: header[1]
57405         });
57406
57407         //this.updateColumns();
57408
57409         this.grid.getGridEl().dom.innerHTML = html;
57410
57411         this.initElements();
57412         
57413         // a kludge to fix the random scolling effect in webkit
57414         this.el.on("scroll", function() {
57415             this.el.dom.scrollTop=0; // hopefully not recursive..
57416         },this);
57417
57418         this.scroller.on("scroll", this.handleScroll, this);
57419         this.lockedBody.on("mousewheel", this.handleWheel, this);
57420         this.mainBody.on("mousewheel", this.handleWheel, this);
57421
57422         this.mainHd.on("mouseover", this.handleHdOver, this);
57423         this.mainHd.on("mouseout", this.handleHdOut, this);
57424         this.mainHd.on("dblclick", this.handleSplitDblClick, this,
57425                 {delegate: "."+this.splitClass});
57426
57427         this.lockedHd.on("mouseover", this.handleHdOver, this);
57428         this.lockedHd.on("mouseout", this.handleHdOut, this);
57429         this.lockedHd.on("dblclick", this.handleSplitDblClick, this,
57430                 {delegate: "."+this.splitClass});
57431
57432         if(this.grid.enableColumnResize !== false && Roo.grid.SplitDragZone){
57433             new Roo.grid.SplitDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57434         }
57435
57436         this.updateSplitters();
57437
57438         if(this.grid.enableColumnMove && Roo.grid.HeaderDragZone){
57439             new Roo.grid.HeaderDragZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57440             new Roo.grid.HeaderDropZone(this.grid, this.lockedHd.dom, this.mainHd.dom);
57441         }
57442
57443         if(this.grid.enableCtxMenu !== false && Roo.menu.Menu){
57444             this.hmenu = new Roo.menu.Menu({id: this.grid.id + "-hctx"});
57445             this.hmenu.add(
57446                 {id:"asc", text: this.sortAscText, cls: "xg-hmenu-sort-asc"},
57447                 {id:"desc", text: this.sortDescText, cls: "xg-hmenu-sort-desc"}
57448             );
57449             if(this.grid.enableColLock !== false){
57450                 this.hmenu.add('-',
57451                     {id:"lock", text: this.lockText, cls: "xg-hmenu-lock"},
57452                     {id:"unlock", text: this.unlockText, cls: "xg-hmenu-unlock"}
57453                 );
57454             }
57455             if (Roo.isTouch) {
57456                  this.hmenu.add('-',
57457                     {id:"wider", text: this.columnsWiderText},
57458                     {id:"narrow", text: this.columnsNarrowText }
57459                 );
57460                 
57461                  
57462             }
57463             
57464             if(this.grid.enableColumnHide !== false){
57465
57466                 this.colMenu = new Roo.menu.Menu({id:this.grid.id + "-hcols-menu"});
57467                 this.colMenu.on("beforeshow", this.beforeColMenuShow, this);
57468                 this.colMenu.on("itemclick", this.handleHdMenuClick, this);
57469
57470                 this.hmenu.add('-',
57471                     {id:"columns", text: this.columnsText, menu: this.colMenu}
57472                 );
57473             }
57474             this.hmenu.on("itemclick", this.handleHdMenuClick, this);
57475
57476             this.grid.on("headercontextmenu", this.handleHdCtx, this);
57477         }
57478
57479         if((this.grid.enableDragDrop || this.grid.enableDrag) && Roo.grid.GridDragZone){
57480             this.dd = new Roo.grid.GridDragZone(this.grid, {
57481                 ddGroup : this.grid.ddGroup || 'GridDD'
57482             });
57483             
57484         }
57485
57486         /*
57487         for(var i = 0; i < colCount; i++){
57488             if(cm.isHidden(i)){
57489                 this.hideColumn(i);
57490             }
57491             if(cm.config[i].align){
57492                 this.css.updateRule(this.colSelector + i, "textAlign", cm.config[i].align);
57493                 this.css.updateRule(this.hdSelector + i, "textAlign", cm.config[i].align);
57494             }
57495         }*/
57496         
57497         this.updateHeaderSortState();
57498
57499         this.beforeInitialResize();
57500         this.layout(true);
57501
57502         // two part rendering gives faster view to the user
57503         this.renderPhase2.defer(1, this);
57504     },
57505
57506     renderPhase2 : function(){
57507         // render the rows now
57508         this.refresh();
57509         if(this.grid.autoSizeColumns){
57510             this.autoSizeColumns();
57511         }
57512     },
57513
57514     beforeInitialResize : function(){
57515
57516     },
57517
57518     onColumnSplitterMoved : function(i, w){
57519         this.userResized = true;
57520         var cm = this.grid.colModel;
57521         cm.setColumnWidth(i, w, true);
57522         var cid = cm.getColumnId(i);
57523         this.css.updateRule(this.colSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57524         this.css.updateRule(this.hdSelector + this.idToCssName(cid), "width", (w-this.borderWidth) + "px");
57525         this.updateSplitters();
57526         this.layout();
57527         this.grid.fireEvent("columnresize", i, w);
57528     },
57529
57530     syncRowHeights : function(startIndex, endIndex){
57531         if(this.grid.enableRowHeightSync === true && this.cm.getLockedCount() > 0){
57532             startIndex = startIndex || 0;
57533             var mrows = this.getBodyTable().rows;
57534             var lrows = this.getLockedTable().rows;
57535             var len = mrows.length-1;
57536             endIndex = Math.min(endIndex || len, len);
57537             for(var i = startIndex; i <= endIndex; i++){
57538                 var m = mrows[i], l = lrows[i];
57539                 var h = Math.max(m.offsetHeight, l.offsetHeight);
57540                 m.style.height = l.style.height = h + "px";
57541             }
57542         }
57543     },
57544
57545     layout : function(initialRender, is2ndPass){
57546         var g = this.grid;
57547         var auto = g.autoHeight;
57548         var scrollOffset = 16;
57549         var c = g.getGridEl(), cm = this.cm,
57550                 expandCol = g.autoExpandColumn,
57551                 gv = this;
57552         //c.beginMeasure();
57553
57554         if(!c.dom.offsetWidth){ // display:none?
57555             if(initialRender){
57556                 this.lockedWrap.show();
57557                 this.mainWrap.show();
57558             }
57559             return;
57560         }
57561
57562         var hasLock = this.cm.isLocked(0);
57563
57564         var tbh = this.headerPanel.getHeight();
57565         var bbh = this.footerPanel.getHeight();
57566
57567         if(auto){
57568             var ch = this.getBodyTable().offsetHeight + tbh + bbh + this.mainHd.getHeight();
57569             var newHeight = ch + c.getBorderWidth("tb");
57570             if(g.maxHeight){
57571                 newHeight = Math.min(g.maxHeight, newHeight);
57572             }
57573             c.setHeight(newHeight);
57574         }
57575
57576         if(g.autoWidth){
57577             c.setWidth(cm.getTotalWidth()+c.getBorderWidth('lr'));
57578         }
57579
57580         var s = this.scroller;
57581
57582         var csize = c.getSize(true);
57583
57584         this.el.setSize(csize.width, csize.height);
57585
57586         this.headerPanel.setWidth(csize.width);
57587         this.footerPanel.setWidth(csize.width);
57588
57589         var hdHeight = this.mainHd.getHeight();
57590         var vw = csize.width;
57591         var vh = csize.height - (tbh + bbh);
57592
57593         s.setSize(vw, vh);
57594
57595         var bt = this.getBodyTable();
57596         
57597         if(cm.getLockedCount() == cm.config.length){
57598             bt = this.getLockedTable();
57599         }
57600         
57601         var ltWidth = hasLock ?
57602                       Math.max(this.getLockedTable().offsetWidth, this.lockedHd.dom.firstChild.offsetWidth) : 0;
57603
57604         var scrollHeight = bt.offsetHeight;
57605         var scrollWidth = ltWidth + bt.offsetWidth;
57606         var vscroll = false, hscroll = false;
57607
57608         this.scrollSizer.setSize(scrollWidth, scrollHeight+hdHeight);
57609
57610         var lw = this.lockedWrap, mw = this.mainWrap;
57611         var lb = this.lockedBody, mb = this.mainBody;
57612
57613         setTimeout(function(){
57614             var t = s.dom.offsetTop;
57615             var w = s.dom.clientWidth,
57616                 h = s.dom.clientHeight;
57617
57618             lw.setTop(t);
57619             lw.setSize(ltWidth, h);
57620
57621             mw.setLeftTop(ltWidth, t);
57622             mw.setSize(w-ltWidth, h);
57623
57624             lb.setHeight(h-hdHeight);
57625             mb.setHeight(h-hdHeight);
57626
57627             if(is2ndPass !== true && !gv.userResized && expandCol){
57628                 // high speed resize without full column calculation
57629                 
57630                 var ci = cm.getIndexById(expandCol);
57631                 if (ci < 0) {
57632                     ci = cm.findColumnIndex(expandCol);
57633                 }
57634                 ci = Math.max(0, ci); // make sure it's got at least the first col.
57635                 var expandId = cm.getColumnId(ci);
57636                 var  tw = cm.getTotalWidth(false);
57637                 var currentWidth = cm.getColumnWidth(ci);
57638                 var cw = Math.min(Math.max(((w-tw)+currentWidth-2)-/*scrollbar*/(w <= s.dom.offsetWidth ? 0 : 18), g.autoExpandMin), g.autoExpandMax);
57639                 if(currentWidth != cw){
57640                     cm.setColumnWidth(ci, cw, true);
57641                     gv.css.updateRule(gv.colSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57642                     gv.css.updateRule(gv.hdSelector+gv.idToCssName(expandId), "width", (cw - gv.borderWidth) + "px");
57643                     gv.updateSplitters();
57644                     gv.layout(false, true);
57645                 }
57646             }
57647
57648             if(initialRender){
57649                 lw.show();
57650                 mw.show();
57651             }
57652             //c.endMeasure();
57653         }, 10);
57654     },
57655
57656     onWindowResize : function(){
57657         if(!this.grid.monitorWindowResize || this.grid.autoHeight){
57658             return;
57659         }
57660         this.layout();
57661     },
57662
57663     appendFooter : function(parentEl){
57664         return null;
57665     },
57666
57667     sortAscText : "Sort Ascending",
57668     sortDescText : "Sort Descending",
57669     lockText : "Lock Column",
57670     unlockText : "Unlock Column",
57671     columnsText : "Columns",
57672  
57673     columnsWiderText : "Wider",
57674     columnsNarrowText : "Thinner"
57675 });
57676
57677
57678 Roo.grid.GridView.ColumnDragZone = function(grid, hd){
57679     Roo.grid.GridView.ColumnDragZone.superclass.constructor.call(this, grid, hd, null);
57680     this.proxy.el.addClass('x-grid3-col-dd');
57681 };
57682
57683 Roo.extend(Roo.grid.GridView.ColumnDragZone, Roo.grid.HeaderDragZone, {
57684     handleMouseDown : function(e){
57685
57686     },
57687
57688     callHandleMouseDown : function(e){
57689         Roo.grid.GridView.ColumnDragZone.superclass.handleMouseDown.call(this, e);
57690     }
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.SplitDragZone = function(grid, hd, hd2){
57706     this.grid = grid;
57707     this.view = grid.getView();
57708     this.proxy = this.view.resizeProxy;
57709     Roo.grid.SplitDragZone.superclass.constructor.call(this, hd,
57710         "gridSplitters" + this.grid.getGridEl().id, {
57711         dragElId : Roo.id(this.proxy.dom), resizeFrame:false
57712     });
57713     this.setHandleElId(Roo.id(hd));
57714     this.setOuterHandleElId(Roo.id(hd2));
57715     this.scroll = false;
57716 };
57717 Roo.extend(Roo.grid.SplitDragZone, Roo.dd.DDProxy, {
57718     fly: Roo.Element.fly,
57719
57720     b4StartDrag : function(x, y){
57721         this.view.headersDisabled = true;
57722         this.proxy.setHeight(this.view.mainWrap.getHeight());
57723         var w = this.cm.getColumnWidth(this.cellIndex);
57724         var minw = Math.max(w-this.grid.minColumnWidth, 0);
57725         this.resetConstraints();
57726         this.setXConstraint(minw, 1000);
57727         this.setYConstraint(0, 0);
57728         this.minX = x - minw;
57729         this.maxX = x + 1000;
57730         this.startPos = x;
57731         Roo.dd.DDProxy.prototype.b4StartDrag.call(this, x, y);
57732     },
57733
57734
57735     handleMouseDown : function(e){
57736         ev = Roo.EventObject.setEvent(e);
57737         var t = this.fly(ev.getTarget());
57738         if(t.hasClass("x-grid-split")){
57739             this.cellIndex = this.view.getCellIndex(t.dom);
57740             this.split = t.dom;
57741             this.cm = this.grid.colModel;
57742             if(this.cm.isResizable(this.cellIndex) && !this.cm.isFixed(this.cellIndex)){
57743                 Roo.grid.SplitDragZone.superclass.handleMouseDown.apply(this, arguments);
57744             }
57745         }
57746     },
57747
57748     endDrag : function(e){
57749         this.view.headersDisabled = false;
57750         var endX = Math.max(this.minX, Roo.lib.Event.getPageX(e));
57751         var diff = endX - this.startPos;
57752         this.view.onColumnSplitterMoved(this.cellIndex, this.cm.getColumnWidth(this.cellIndex)+diff);
57753     },
57754
57755     autoOffset : function(){
57756         this.setDelta(0,0);
57757     }
57758 });/*
57759  * Based on:
57760  * Ext JS Library 1.1.1
57761  * Copyright(c) 2006-2007, Ext JS, LLC.
57762  *
57763  * Originally Released Under LGPL - original licence link has changed is not relivant.
57764  *
57765  * Fork - LGPL
57766  * <script type="text/javascript">
57767  */
57768  
57769 // private
57770 // This is a support class used internally by the Grid components
57771 Roo.grid.GridDragZone = function(grid, config){
57772     this.view = grid.getView();
57773     Roo.grid.GridDragZone.superclass.constructor.call(this, this.view.mainBody.dom, config);
57774     if(this.view.lockedBody){
57775         this.setHandleElId(Roo.id(this.view.mainBody.dom));
57776         this.setOuterHandleElId(Roo.id(this.view.lockedBody.dom));
57777     }
57778     this.scroll = false;
57779     this.grid = grid;
57780     this.ddel = document.createElement('div');
57781     this.ddel.className = 'x-grid-dd-wrap';
57782 };
57783
57784 Roo.extend(Roo.grid.GridDragZone, Roo.dd.DragZone, {
57785     ddGroup : "GridDD",
57786
57787     getDragData : function(e){
57788         var t = Roo.lib.Event.getTarget(e);
57789         var rowIndex = this.view.findRowIndex(t);
57790         var sm = this.grid.selModel;
57791             
57792         //Roo.log(rowIndex);
57793         
57794         if (sm.getSelectedCell) {
57795             // cell selection..
57796             if (!sm.getSelectedCell()) {
57797                 return false;
57798             }
57799             if (rowIndex != sm.getSelectedCell()[0]) {
57800                 return false;
57801             }
57802         
57803         }
57804         
57805         if(rowIndex !== false){
57806             
57807             // if editorgrid.. 
57808             
57809             
57810             //Roo.log([ sm.getSelectedCell() ? sm.getSelectedCell()[0] : 'NO' , rowIndex ]);
57811                
57812             //if(!sm.isSelected(rowIndex) || e.hasModifier()){
57813               //  
57814             //}
57815             if (e.hasModifier()){
57816                 sm.handleMouseDown(e, t); // non modifier buttons are handled by row select.
57817             }
57818             
57819             Roo.log("getDragData");
57820             
57821             return {
57822                 grid: this.grid,
57823                 ddel: this.ddel,
57824                 rowIndex: rowIndex,
57825                 selections:sm.getSelections ? sm.getSelections() : (
57826                     sm.getSelectedCell() ? [ this.grid.ds.getAt(sm.getSelectedCell()[0]) ] : []
57827                 )
57828             };
57829         }
57830         return false;
57831     },
57832
57833     onInitDrag : function(e){
57834         var data = this.dragData;
57835         this.ddel.innerHTML = this.grid.getDragDropText();
57836         this.proxy.update(this.ddel);
57837         // fire start drag?
57838     },
57839
57840     afterRepair : function(){
57841         this.dragging = false;
57842     },
57843
57844     getRepairXY : function(e, data){
57845         return false;
57846     },
57847
57848     onEndDrag : function(data, e){
57849         // fire end drag?
57850     },
57851
57852     onValidDrop : function(dd, e, id){
57853         // fire drag drop?
57854         this.hideProxy();
57855     },
57856
57857     beforeInvalidDrop : function(e, id){
57858
57859     }
57860 });/*
57861  * Based on:
57862  * Ext JS Library 1.1.1
57863  * Copyright(c) 2006-2007, Ext JS, LLC.
57864  *
57865  * Originally Released Under LGPL - original licence link has changed is not relivant.
57866  *
57867  * Fork - LGPL
57868  * <script type="text/javascript">
57869  */
57870  
57871
57872 /**
57873  * @class Roo.grid.ColumnModel
57874  * @extends Roo.util.Observable
57875  * This is the default implementation of a ColumnModel used by the Grid. It defines
57876  * the columns in the grid.
57877  * <br>Usage:<br>
57878  <pre><code>
57879  var colModel = new Roo.grid.ColumnModel([
57880         {header: "Ticker", width: 60, sortable: true, locked: true},
57881         {header: "Company Name", width: 150, sortable: true},
57882         {header: "Market Cap.", width: 100, sortable: true},
57883         {header: "$ Sales", width: 100, sortable: true, renderer: money},
57884         {header: "Employees", width: 100, sortable: true, resizable: false}
57885  ]);
57886  </code></pre>
57887  * <p>
57888  
57889  * The config options listed for this class are options which may appear in each
57890  * individual column definition.
57891  * <br/>RooJS Fix - column id's are not sequential but use Roo.id() - fixes bugs with layouts.
57892  * @constructor
57893  * @param {Object} config An Array of column config objects. See this class's
57894  * config objects for details.
57895 */
57896 Roo.grid.ColumnModel = function(config){
57897         /**
57898      * The config passed into the constructor
57899      */
57900     this.config = config;
57901     this.lookup = {};
57902
57903     // if no id, create one
57904     // if the column does not have a dataIndex mapping,
57905     // map it to the order it is in the config
57906     for(var i = 0, len = config.length; i < len; i++){
57907         var c = config[i];
57908         if(typeof c.dataIndex == "undefined"){
57909             c.dataIndex = i;
57910         }
57911         if(typeof c.renderer == "string"){
57912             c.renderer = Roo.util.Format[c.renderer];
57913         }
57914         if(typeof c.id == "undefined"){
57915             c.id = Roo.id();
57916         }
57917         if(c.editor && c.editor.xtype){
57918             c.editor  = Roo.factory(c.editor, Roo.grid);
57919         }
57920         if(c.editor && c.editor.isFormField){
57921             c.editor = new Roo.grid.GridEditor(c.editor);
57922         }
57923         this.lookup[c.id] = c;
57924     }
57925
57926     /**
57927      * The width of columns which have no width specified (defaults to 100)
57928      * @type Number
57929      */
57930     this.defaultWidth = 100;
57931
57932     /**
57933      * Default sortable of columns which have no sortable specified (defaults to false)
57934      * @type Boolean
57935      */
57936     this.defaultSortable = false;
57937
57938     this.addEvents({
57939         /**
57940              * @event widthchange
57941              * Fires when the width of a column changes.
57942              * @param {ColumnModel} this
57943              * @param {Number} columnIndex The column index
57944              * @param {Number} newWidth The new width
57945              */
57946             "widthchange": true,
57947         /**
57948              * @event headerchange
57949              * Fires when the text of a header changes.
57950              * @param {ColumnModel} this
57951              * @param {Number} columnIndex The column index
57952              * @param {Number} newText The new header text
57953              */
57954             "headerchange": true,
57955         /**
57956              * @event hiddenchange
57957              * Fires when a column is hidden or "unhidden".
57958              * @param {ColumnModel} this
57959              * @param {Number} columnIndex The column index
57960              * @param {Boolean} hidden true if hidden, false otherwise
57961              */
57962             "hiddenchange": true,
57963             /**
57964          * @event columnmoved
57965          * Fires when a column is moved.
57966          * @param {ColumnModel} this
57967          * @param {Number} oldIndex
57968          * @param {Number} newIndex
57969          */
57970         "columnmoved" : true,
57971         /**
57972          * @event columlockchange
57973          * Fires when a column's locked state is changed
57974          * @param {ColumnModel} this
57975          * @param {Number} colIndex
57976          * @param {Boolean} locked true if locked
57977          */
57978         "columnlockchange" : true
57979     });
57980     Roo.grid.ColumnModel.superclass.constructor.call(this);
57981 };
57982 Roo.extend(Roo.grid.ColumnModel, Roo.util.Observable, {
57983     /**
57984      * @cfg {String} header The header text to display in the Grid view.
57985      */
57986     /**
57987      * @cfg {String} dataIndex (Optional) The name of the field in the grid's {@link Roo.data.Store}'s
57988      * {@link Roo.data.Record} definition from which to draw the column's value. If not
57989      * specified, the column's index is used as an index into the Record's data Array.
57990      */
57991     /**
57992      * @cfg {Number} width (Optional) The initial width in pixels of the column. Using this
57993      * instead of {@link Roo.grid.Grid#autoSizeColumns} is more efficient.
57994      */
57995     /**
57996      * @cfg {Boolean} sortable (Optional) True if sorting is to be allowed on this column.
57997      * Defaults to the value of the {@link #defaultSortable} property.
57998      * Whether local/remote sorting is used is specified in {@link Roo.data.Store#remoteSort}.
57999      */
58000     /**
58001      * @cfg {Boolean} locked (Optional) True to lock the column in place while scrolling the Grid.  Defaults to false.
58002      */
58003     /**
58004      * @cfg {Boolean} fixed (Optional) True if the column width cannot be changed.  Defaults to false.
58005      */
58006     /**
58007      * @cfg {Boolean} resizable (Optional) False to disable column resizing. Defaults to true.
58008      */
58009     /**
58010      * @cfg {Boolean} hidden (Optional) True to hide the column. Defaults to false.
58011      */
58012     /**
58013      * @cfg {Function} renderer (Optional) A function used to generate HTML markup for a cell
58014      * given the cell's data value. See {@link #setRenderer}. If not specified, the
58015      * default renderer returns the escaped data value. If an object is returned (bootstrap only)
58016      * then it is treated as a Roo Component object instance, and it is rendered after the initial row is rendered
58017      */
58018        /**
58019      * @cfg {Roo.grid.GridEditor} editor (Optional) For grid editors - returns the grid editor 
58020      */
58021     /**
58022      * @cfg {String} align (Optional) Set the CSS text-align property of the column.  Defaults to undefined.
58023      */
58024     /**
58025      * @cfg {String} valign (Optional) Set the CSS vertical-align property of the column (eg. middle, top, bottom etc).  Defaults to undefined.
58026      */
58027     /**
58028      * @cfg {String} cursor (Optional)
58029      */
58030     /**
58031      * @cfg {String} tooltip (Optional)
58032      */
58033     /**
58034      * @cfg {Number} xs (Optional)
58035      */
58036     /**
58037      * @cfg {Number} sm (Optional)
58038      */
58039     /**
58040      * @cfg {Number} md (Optional)
58041      */
58042     /**
58043      * @cfg {Number} lg (Optional)
58044      */
58045     /**
58046      * Returns the id of the column at the specified index.
58047      * @param {Number} index The column index
58048      * @return {String} the id
58049      */
58050     getColumnId : function(index){
58051         return this.config[index].id;
58052     },
58053
58054     /**
58055      * Returns the column for a specified id.
58056      * @param {String} id The column id
58057      * @return {Object} the column
58058      */
58059     getColumnById : function(id){
58060         return this.lookup[id];
58061     },
58062
58063     
58064     /**
58065      * Returns the column for a specified dataIndex.
58066      * @param {String} dataIndex The column dataIndex
58067      * @return {Object|Boolean} the column or false if not found
58068      */
58069     getColumnByDataIndex: function(dataIndex){
58070         var index = this.findColumnIndex(dataIndex);
58071         return index > -1 ? this.config[index] : false;
58072     },
58073     
58074     /**
58075      * Returns the index for a specified column id.
58076      * @param {String} id The column id
58077      * @return {Number} the index, or -1 if not found
58078      */
58079     getIndexById : function(id){
58080         for(var i = 0, len = this.config.length; i < len; i++){
58081             if(this.config[i].id == id){
58082                 return i;
58083             }
58084         }
58085         return -1;
58086     },
58087     
58088     /**
58089      * Returns the index for a specified column dataIndex.
58090      * @param {String} dataIndex The column dataIndex
58091      * @return {Number} the index, or -1 if not found
58092      */
58093     
58094     findColumnIndex : function(dataIndex){
58095         for(var i = 0, len = this.config.length; i < len; i++){
58096             if(this.config[i].dataIndex == dataIndex){
58097                 return i;
58098             }
58099         }
58100         return -1;
58101     },
58102     
58103     
58104     moveColumn : function(oldIndex, newIndex){
58105         var c = this.config[oldIndex];
58106         this.config.splice(oldIndex, 1);
58107         this.config.splice(newIndex, 0, c);
58108         this.dataMap = null;
58109         this.fireEvent("columnmoved", this, oldIndex, newIndex);
58110     },
58111
58112     isLocked : function(colIndex){
58113         return this.config[colIndex].locked === true;
58114     },
58115
58116     setLocked : function(colIndex, value, suppressEvent){
58117         if(this.isLocked(colIndex) == value){
58118             return;
58119         }
58120         this.config[colIndex].locked = value;
58121         if(!suppressEvent){
58122             this.fireEvent("columnlockchange", this, colIndex, value);
58123         }
58124     },
58125
58126     getTotalLockedWidth : function(){
58127         var totalWidth = 0;
58128         for(var i = 0; i < this.config.length; i++){
58129             if(this.isLocked(i) && !this.isHidden(i)){
58130                 this.totalWidth += this.getColumnWidth(i);
58131             }
58132         }
58133         return totalWidth;
58134     },
58135
58136     getLockedCount : function(){
58137         for(var i = 0, len = this.config.length; i < len; i++){
58138             if(!this.isLocked(i)){
58139                 return i;
58140             }
58141         }
58142         
58143         return this.config.length;
58144     },
58145
58146     /**
58147      * Returns the number of columns.
58148      * @return {Number}
58149      */
58150     getColumnCount : function(visibleOnly){
58151         if(visibleOnly === true){
58152             var c = 0;
58153             for(var i = 0, len = this.config.length; i < len; i++){
58154                 if(!this.isHidden(i)){
58155                     c++;
58156                 }
58157             }
58158             return c;
58159         }
58160         return this.config.length;
58161     },
58162
58163     /**
58164      * Returns the column configs that return true by the passed function that is called with (columnConfig, index)
58165      * @param {Function} fn
58166      * @param {Object} scope (optional)
58167      * @return {Array} result
58168      */
58169     getColumnsBy : function(fn, scope){
58170         var r = [];
58171         for(var i = 0, len = this.config.length; i < len; i++){
58172             var c = this.config[i];
58173             if(fn.call(scope||this, c, i) === true){
58174                 r[r.length] = c;
58175             }
58176         }
58177         return r;
58178     },
58179
58180     /**
58181      * Returns true if the specified column is sortable.
58182      * @param {Number} col The column index
58183      * @return {Boolean}
58184      */
58185     isSortable : function(col){
58186         if(typeof this.config[col].sortable == "undefined"){
58187             return this.defaultSortable;
58188         }
58189         return this.config[col].sortable;
58190     },
58191
58192     /**
58193      * Returns the rendering (formatting) function defined for the column.
58194      * @param {Number} col The column index.
58195      * @return {Function} The function used to render the cell. See {@link #setRenderer}.
58196      */
58197     getRenderer : function(col){
58198         if(!this.config[col].renderer){
58199             return Roo.grid.ColumnModel.defaultRenderer;
58200         }
58201         return this.config[col].renderer;
58202     },
58203
58204     /**
58205      * Sets the rendering (formatting) function for a column.
58206      * @param {Number} col The column index
58207      * @param {Function} fn The function to use to process the cell's raw data
58208      * to return HTML markup for the grid view. The render function is called with
58209      * the following parameters:<ul>
58210      * <li>Data value.</li>
58211      * <li>Cell metadata. An object in which you may set the following attributes:<ul>
58212      * <li>css A CSS style string to apply to the table cell.</li>
58213      * <li>attr An HTML attribute definition string to apply to the data container element <i>within</i> the table cell.</li></ul>
58214      * <li>The {@link Roo.data.Record} from which the data was extracted.</li>
58215      * <li>Row index</li>
58216      * <li>Column index</li>
58217      * <li>The {@link Roo.data.Store} object from which the Record was extracted</li></ul>
58218      */
58219     setRenderer : function(col, fn){
58220         this.config[col].renderer = fn;
58221     },
58222
58223     /**
58224      * Returns the width for the specified column.
58225      * @param {Number} col The column index
58226      * @return {Number}
58227      */
58228     getColumnWidth : function(col){
58229         return this.config[col].width * 1 || this.defaultWidth;
58230     },
58231
58232     /**
58233      * Sets the width for a column.
58234      * @param {Number} col The column index
58235      * @param {Number} width The new width
58236      */
58237     setColumnWidth : function(col, width, suppressEvent){
58238         this.config[col].width = width;
58239         this.totalWidth = null;
58240         if(!suppressEvent){
58241              this.fireEvent("widthchange", this, col, width);
58242         }
58243     },
58244
58245     /**
58246      * Returns the total width of all columns.
58247      * @param {Boolean} includeHidden True to include hidden column widths
58248      * @return {Number}
58249      */
58250     getTotalWidth : function(includeHidden){
58251         if(!this.totalWidth){
58252             this.totalWidth = 0;
58253             for(var i = 0, len = this.config.length; i < len; i++){
58254                 if(includeHidden || !this.isHidden(i)){
58255                     this.totalWidth += this.getColumnWidth(i);
58256                 }
58257             }
58258         }
58259         return this.totalWidth;
58260     },
58261
58262     /**
58263      * Returns the header for the specified column.
58264      * @param {Number} col The column index
58265      * @return {String}
58266      */
58267     getColumnHeader : function(col){
58268         return this.config[col].header;
58269     },
58270
58271     /**
58272      * Sets the header for a column.
58273      * @param {Number} col The column index
58274      * @param {String} header The new header
58275      */
58276     setColumnHeader : function(col, header){
58277         this.config[col].header = header;
58278         this.fireEvent("headerchange", this, col, header);
58279     },
58280
58281     /**
58282      * Returns the tooltip for the specified column.
58283      * @param {Number} col The column index
58284      * @return {String}
58285      */
58286     getColumnTooltip : function(col){
58287             return this.config[col].tooltip;
58288     },
58289     /**
58290      * Sets the tooltip for a column.
58291      * @param {Number} col The column index
58292      * @param {String} tooltip The new tooltip
58293      */
58294     setColumnTooltip : function(col, tooltip){
58295             this.config[col].tooltip = tooltip;
58296     },
58297
58298     /**
58299      * Returns the dataIndex for the specified column.
58300      * @param {Number} col The column index
58301      * @return {Number}
58302      */
58303     getDataIndex : function(col){
58304         return this.config[col].dataIndex;
58305     },
58306
58307     /**
58308      * Sets the dataIndex for a column.
58309      * @param {Number} col The column index
58310      * @param {Number} dataIndex The new dataIndex
58311      */
58312     setDataIndex : function(col, dataIndex){
58313         this.config[col].dataIndex = dataIndex;
58314     },
58315
58316     
58317     
58318     /**
58319      * Returns true if the cell is editable.
58320      * @param {Number} colIndex The column index
58321      * @param {Number} rowIndex The row index - this is nto actually used..?
58322      * @return {Boolean}
58323      */
58324     isCellEditable : function(colIndex, rowIndex){
58325         return (this.config[colIndex].editable || (typeof this.config[colIndex].editable == "undefined" && this.config[colIndex].editor)) ? true : false;
58326     },
58327
58328     /**
58329      * Returns the editor defined for the cell/column.
58330      * return false or null to disable editing.
58331      * @param {Number} colIndex The column index
58332      * @param {Number} rowIndex The row index
58333      * @return {Object}
58334      */
58335     getCellEditor : function(colIndex, rowIndex){
58336         return this.config[colIndex].editor;
58337     },
58338
58339     /**
58340      * Sets if a column is editable.
58341      * @param {Number} col The column index
58342      * @param {Boolean} editable True if the column is editable
58343      */
58344     setEditable : function(col, editable){
58345         this.config[col].editable = editable;
58346     },
58347
58348
58349     /**
58350      * Returns true if the column is hidden.
58351      * @param {Number} colIndex The column index
58352      * @return {Boolean}
58353      */
58354     isHidden : function(colIndex){
58355         return this.config[colIndex].hidden;
58356     },
58357
58358
58359     /**
58360      * Returns true if the column width cannot be changed
58361      */
58362     isFixed : function(colIndex){
58363         return this.config[colIndex].fixed;
58364     },
58365
58366     /**
58367      * Returns true if the column can be resized
58368      * @return {Boolean}
58369      */
58370     isResizable : function(colIndex){
58371         return colIndex >= 0 && this.config[colIndex].resizable !== false && this.config[colIndex].fixed !== true;
58372     },
58373     /**
58374      * Sets if a column is hidden.
58375      * @param {Number} colIndex The column index
58376      * @param {Boolean} hidden True if the column is hidden
58377      */
58378     setHidden : function(colIndex, hidden){
58379         this.config[colIndex].hidden = hidden;
58380         this.totalWidth = null;
58381         this.fireEvent("hiddenchange", this, colIndex, hidden);
58382     },
58383
58384     /**
58385      * Sets the editor for a column.
58386      * @param {Number} col The column index
58387      * @param {Object} editor The editor object
58388      */
58389     setEditor : function(col, editor){
58390         this.config[col].editor = editor;
58391     }
58392 });
58393
58394 Roo.grid.ColumnModel.defaultRenderer = function(value)
58395 {
58396     if(typeof value == "object") {
58397         return value;
58398     }
58399         if(typeof value == "string" && value.length < 1){
58400             return "&#160;";
58401         }
58402     
58403         return String.format("{0}", value);
58404 };
58405
58406 // Alias for backwards compatibility
58407 Roo.grid.DefaultColumnModel = Roo.grid.ColumnModel;
58408 /*
58409  * Based on:
58410  * Ext JS Library 1.1.1
58411  * Copyright(c) 2006-2007, Ext JS, LLC.
58412  *
58413  * Originally Released Under LGPL - original licence link has changed is not relivant.
58414  *
58415  * Fork - LGPL
58416  * <script type="text/javascript">
58417  */
58418
58419 /**
58420  * @class Roo.grid.AbstractSelectionModel
58421  * @extends Roo.util.Observable
58422  * Abstract base class for grid SelectionModels.  It provides the interface that should be
58423  * implemented by descendant classes.  This class should not be directly instantiated.
58424  * @constructor
58425  */
58426 Roo.grid.AbstractSelectionModel = function(){
58427     this.locked = false;
58428     Roo.grid.AbstractSelectionModel.superclass.constructor.call(this);
58429 };
58430
58431 Roo.extend(Roo.grid.AbstractSelectionModel, Roo.util.Observable,  {
58432     /** @ignore Called by the grid automatically. Do not call directly. */
58433     init : function(grid){
58434         this.grid = grid;
58435         this.initEvents();
58436     },
58437
58438     /**
58439      * Locks the selections.
58440      */
58441     lock : function(){
58442         this.locked = true;
58443     },
58444
58445     /**
58446      * Unlocks the selections.
58447      */
58448     unlock : function(){
58449         this.locked = false;
58450     },
58451
58452     /**
58453      * Returns true if the selections are locked.
58454      * @return {Boolean}
58455      */
58456     isLocked : function(){
58457         return this.locked;
58458     }
58459 });/*
58460  * Based on:
58461  * Ext JS Library 1.1.1
58462  * Copyright(c) 2006-2007, Ext JS, LLC.
58463  *
58464  * Originally Released Under LGPL - original licence link has changed is not relivant.
58465  *
58466  * Fork - LGPL
58467  * <script type="text/javascript">
58468  */
58469 /**
58470  * @extends Roo.grid.AbstractSelectionModel
58471  * @class Roo.grid.RowSelectionModel
58472  * The default SelectionModel used by {@link Roo.grid.Grid}.
58473  * It supports multiple selections and keyboard selection/navigation. 
58474  * @constructor
58475  * @param {Object} config
58476  */
58477 Roo.grid.RowSelectionModel = function(config){
58478     Roo.apply(this, config);
58479     this.selections = new Roo.util.MixedCollection(false, function(o){
58480         return o.id;
58481     });
58482
58483     this.last = false;
58484     this.lastActive = false;
58485
58486     this.addEvents({
58487         /**
58488              * @event selectionchange
58489              * Fires when the selection changes
58490              * @param {SelectionModel} this
58491              */
58492             "selectionchange" : true,
58493         /**
58494              * @event afterselectionchange
58495              * Fires after the selection changes (eg. by key press or clicking)
58496              * @param {SelectionModel} this
58497              */
58498             "afterselectionchange" : true,
58499         /**
58500              * @event beforerowselect
58501              * Fires when a row is selected being selected, return false to cancel.
58502              * @param {SelectionModel} this
58503              * @param {Number} rowIndex The selected index
58504              * @param {Boolean} keepExisting False if other selections will be cleared
58505              */
58506             "beforerowselect" : true,
58507         /**
58508              * @event rowselect
58509              * Fires when a row is selected.
58510              * @param {SelectionModel} this
58511              * @param {Number} rowIndex The selected index
58512              * @param {Roo.data.Record} r The record
58513              */
58514             "rowselect" : true,
58515         /**
58516              * @event rowdeselect
58517              * Fires when a row is deselected.
58518              * @param {SelectionModel} this
58519              * @param {Number} rowIndex The selected index
58520              */
58521         "rowdeselect" : true
58522     });
58523     Roo.grid.RowSelectionModel.superclass.constructor.call(this);
58524     this.locked = false;
58525 };
58526
58527 Roo.extend(Roo.grid.RowSelectionModel, Roo.grid.AbstractSelectionModel,  {
58528     /**
58529      * @cfg {Boolean} singleSelect
58530      * True to allow selection of only one row at a time (defaults to false)
58531      */
58532     singleSelect : false,
58533
58534     // private
58535     initEvents : function(){
58536
58537         if(!this.grid.enableDragDrop && !this.grid.enableDrag){
58538             this.grid.on("mousedown", this.handleMouseDown, this);
58539         }else{ // allow click to work like normal
58540             this.grid.on("rowclick", this.handleDragableRowClick, this);
58541         }
58542
58543         this.rowNav = new Roo.KeyNav(this.grid.getGridEl(), {
58544             "up" : function(e){
58545                 if(!e.shiftKey){
58546                     this.selectPrevious(e.shiftKey);
58547                 }else if(this.last !== false && this.lastActive !== false){
58548                     var last = this.last;
58549                     this.selectRange(this.last,  this.lastActive-1);
58550                     this.grid.getView().focusRow(this.lastActive);
58551                     if(last !== false){
58552                         this.last = last;
58553                     }
58554                 }else{
58555                     this.selectFirstRow();
58556                 }
58557                 this.fireEvent("afterselectionchange", this);
58558             },
58559             "down" : function(e){
58560                 if(!e.shiftKey){
58561                     this.selectNext(e.shiftKey);
58562                 }else if(this.last !== false && this.lastActive !== false){
58563                     var last = this.last;
58564                     this.selectRange(this.last,  this.lastActive+1);
58565                     this.grid.getView().focusRow(this.lastActive);
58566                     if(last !== false){
58567                         this.last = last;
58568                     }
58569                 }else{
58570                     this.selectFirstRow();
58571                 }
58572                 this.fireEvent("afterselectionchange", this);
58573             },
58574             scope: this
58575         });
58576
58577         var view = this.grid.view;
58578         view.on("refresh", this.onRefresh, this);
58579         view.on("rowupdated", this.onRowUpdated, this);
58580         view.on("rowremoved", this.onRemove, this);
58581     },
58582
58583     // private
58584     onRefresh : function(){
58585         var ds = this.grid.dataSource, i, v = this.grid.view;
58586         var s = this.selections;
58587         s.each(function(r){
58588             if((i = ds.indexOfId(r.id)) != -1){
58589                 v.onRowSelect(i);
58590                 s.add(ds.getAt(i)); // updating the selection relate data
58591             }else{
58592                 s.remove(r);
58593             }
58594         });
58595     },
58596
58597     // private
58598     onRemove : function(v, index, r){
58599         this.selections.remove(r);
58600     },
58601
58602     // private
58603     onRowUpdated : function(v, index, r){
58604         if(this.isSelected(r)){
58605             v.onRowSelect(index);
58606         }
58607     },
58608
58609     /**
58610      * Select records.
58611      * @param {Array} records The records to select
58612      * @param {Boolean} keepExisting (optional) True to keep existing selections
58613      */
58614     selectRecords : function(records, keepExisting){
58615         if(!keepExisting){
58616             this.clearSelections();
58617         }
58618         var ds = this.grid.dataSource;
58619         for(var i = 0, len = records.length; i < len; i++){
58620             this.selectRow(ds.indexOf(records[i]), true);
58621         }
58622     },
58623
58624     /**
58625      * Gets the number of selected rows.
58626      * @return {Number}
58627      */
58628     getCount : function(){
58629         return this.selections.length;
58630     },
58631
58632     /**
58633      * Selects the first row in the grid.
58634      */
58635     selectFirstRow : function(){
58636         this.selectRow(0);
58637     },
58638
58639     /**
58640      * Select the last row.
58641      * @param {Boolean} keepExisting (optional) True to keep existing selections
58642      */
58643     selectLastRow : function(keepExisting){
58644         this.selectRow(this.grid.dataSource.getCount() - 1, keepExisting);
58645     },
58646
58647     /**
58648      * Selects the row immediately following the last selected row.
58649      * @param {Boolean} keepExisting (optional) True to keep existing selections
58650      */
58651     selectNext : function(keepExisting){
58652         if(this.last !== false && (this.last+1) < this.grid.dataSource.getCount()){
58653             this.selectRow(this.last+1, keepExisting);
58654             this.grid.getView().focusRow(this.last);
58655         }
58656     },
58657
58658     /**
58659      * Selects the row that precedes the last selected row.
58660      * @param {Boolean} keepExisting (optional) True to keep existing selections
58661      */
58662     selectPrevious : function(keepExisting){
58663         if(this.last){
58664             this.selectRow(this.last-1, keepExisting);
58665             this.grid.getView().focusRow(this.last);
58666         }
58667     },
58668
58669     /**
58670      * Returns the selected records
58671      * @return {Array} Array of selected records
58672      */
58673     getSelections : function(){
58674         return [].concat(this.selections.items);
58675     },
58676
58677     /**
58678      * Returns the first selected record.
58679      * @return {Record}
58680      */
58681     getSelected : function(){
58682         return this.selections.itemAt(0);
58683     },
58684
58685
58686     /**
58687      * Clears all selections.
58688      */
58689     clearSelections : function(fast){
58690         if(this.locked) {
58691             return;
58692         }
58693         if(fast !== true){
58694             var ds = this.grid.dataSource;
58695             var s = this.selections;
58696             s.each(function(r){
58697                 this.deselectRow(ds.indexOfId(r.id));
58698             }, this);
58699             s.clear();
58700         }else{
58701             this.selections.clear();
58702         }
58703         this.last = false;
58704     },
58705
58706
58707     /**
58708      * Selects all rows.
58709      */
58710     selectAll : function(){
58711         if(this.locked) {
58712             return;
58713         }
58714         this.selections.clear();
58715         for(var i = 0, len = this.grid.dataSource.getCount(); i < len; i++){
58716             this.selectRow(i, true);
58717         }
58718     },
58719
58720     /**
58721      * Returns True if there is a selection.
58722      * @return {Boolean}
58723      */
58724     hasSelection : function(){
58725         return this.selections.length > 0;
58726     },
58727
58728     /**
58729      * Returns True if the specified row is selected.
58730      * @param {Number/Record} record The record or index of the record to check
58731      * @return {Boolean}
58732      */
58733     isSelected : function(index){
58734         var r = typeof index == "number" ? this.grid.dataSource.getAt(index) : index;
58735         return (r && this.selections.key(r.id) ? true : false);
58736     },
58737
58738     /**
58739      * Returns True if the specified record id is selected.
58740      * @param {String} id The id of record to check
58741      * @return {Boolean}
58742      */
58743     isIdSelected : function(id){
58744         return (this.selections.key(id) ? true : false);
58745     },
58746
58747     // private
58748     handleMouseDown : function(e, t){
58749         var view = this.grid.getView(), rowIndex;
58750         if(this.isLocked() || (rowIndex = view.findRowIndex(t)) === false){
58751             return;
58752         };
58753         if(e.shiftKey && this.last !== false){
58754             var last = this.last;
58755             this.selectRange(last, rowIndex, e.ctrlKey);
58756             this.last = last; // reset the last
58757             view.focusRow(rowIndex);
58758         }else{
58759             var isSelected = this.isSelected(rowIndex);
58760             if(e.button !== 0 && isSelected){
58761                 view.focusRow(rowIndex);
58762             }else if(e.ctrlKey && isSelected){
58763                 this.deselectRow(rowIndex);
58764             }else if(!isSelected){
58765                 this.selectRow(rowIndex, e.button === 0 && (e.ctrlKey || e.shiftKey));
58766                 view.focusRow(rowIndex);
58767             }
58768         }
58769         this.fireEvent("afterselectionchange", this);
58770     },
58771     // private
58772     handleDragableRowClick :  function(grid, rowIndex, e) 
58773     {
58774         if(e.button === 0 && !e.shiftKey && !e.ctrlKey) {
58775             this.selectRow(rowIndex, false);
58776             grid.view.focusRow(rowIndex);
58777              this.fireEvent("afterselectionchange", this);
58778         }
58779     },
58780     
58781     /**
58782      * Selects multiple rows.
58783      * @param {Array} rows Array of the indexes of the row to select
58784      * @param {Boolean} keepExisting (optional) True to keep existing selections
58785      */
58786     selectRows : function(rows, keepExisting){
58787         if(!keepExisting){
58788             this.clearSelections();
58789         }
58790         for(var i = 0, len = rows.length; i < len; i++){
58791             this.selectRow(rows[i], true);
58792         }
58793     },
58794
58795     /**
58796      * Selects a range of rows. All rows in between startRow and endRow are also selected.
58797      * @param {Number} startRow The index of the first row in the range
58798      * @param {Number} endRow The index of the last row in the range
58799      * @param {Boolean} keepExisting (optional) True to retain existing selections
58800      */
58801     selectRange : function(startRow, endRow, keepExisting){
58802         if(this.locked) {
58803             return;
58804         }
58805         if(!keepExisting){
58806             this.clearSelections();
58807         }
58808         if(startRow <= endRow){
58809             for(var i = startRow; i <= endRow; i++){
58810                 this.selectRow(i, true);
58811             }
58812         }else{
58813             for(var i = startRow; i >= endRow; i--){
58814                 this.selectRow(i, true);
58815             }
58816         }
58817     },
58818
58819     /**
58820      * Deselects a range of rows. All rows in between startRow and endRow are also deselected.
58821      * @param {Number} startRow The index of the first row in the range
58822      * @param {Number} endRow The index of the last row in the range
58823      */
58824     deselectRange : function(startRow, endRow, preventViewNotify){
58825         if(this.locked) {
58826             return;
58827         }
58828         for(var i = startRow; i <= endRow; i++){
58829             this.deselectRow(i, preventViewNotify);
58830         }
58831     },
58832
58833     /**
58834      * Selects a row.
58835      * @param {Number} row The index of the row to select
58836      * @param {Boolean} keepExisting (optional) True to keep existing selections
58837      */
58838     selectRow : function(index, keepExisting, preventViewNotify){
58839         if(this.locked || (index < 0 || index >= this.grid.dataSource.getCount())) {
58840             return;
58841         }
58842         if(this.fireEvent("beforerowselect", this, index, keepExisting) !== false){
58843             if(!keepExisting || this.singleSelect){
58844                 this.clearSelections();
58845             }
58846             var r = this.grid.dataSource.getAt(index);
58847             this.selections.add(r);
58848             this.last = this.lastActive = index;
58849             if(!preventViewNotify){
58850                 this.grid.getView().onRowSelect(index);
58851             }
58852             this.fireEvent("rowselect", this, index, r);
58853             this.fireEvent("selectionchange", this);
58854         }
58855     },
58856
58857     /**
58858      * Deselects a row.
58859      * @param {Number} row The index of the row to deselect
58860      */
58861     deselectRow : function(index, preventViewNotify){
58862         if(this.locked) {
58863             return;
58864         }
58865         if(this.last == index){
58866             this.last = false;
58867         }
58868         if(this.lastActive == index){
58869             this.lastActive = false;
58870         }
58871         var r = this.grid.dataSource.getAt(index);
58872         this.selections.remove(r);
58873         if(!preventViewNotify){
58874             this.grid.getView().onRowDeselect(index);
58875         }
58876         this.fireEvent("rowdeselect", this, index);
58877         this.fireEvent("selectionchange", this);
58878     },
58879
58880     // private
58881     restoreLast : function(){
58882         if(this._last){
58883             this.last = this._last;
58884         }
58885     },
58886
58887     // private
58888     acceptsNav : function(row, col, cm){
58889         return !cm.isHidden(col) && cm.isCellEditable(col, row);
58890     },
58891
58892     // private
58893     onEditorKey : function(field, e){
58894         var k = e.getKey(), newCell, g = this.grid, ed = g.activeEditor;
58895         if(k == e.TAB){
58896             e.stopEvent();
58897             ed.completeEdit();
58898             if(e.shiftKey){
58899                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
58900             }else{
58901                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
58902             }
58903         }else if(k == e.ENTER && !e.ctrlKey){
58904             e.stopEvent();
58905             ed.completeEdit();
58906             if(e.shiftKey){
58907                 newCell = g.walkCells(ed.row-1, ed.col, -1, this.acceptsNav, this);
58908             }else{
58909                 newCell = g.walkCells(ed.row+1, ed.col, 1, this.acceptsNav, this);
58910             }
58911         }else if(k == e.ESC){
58912             ed.cancelEdit();
58913         }
58914         if(newCell){
58915             g.startEditing(newCell[0], newCell[1]);
58916         }
58917     }
58918 });/*
58919  * Based on:
58920  * Ext JS Library 1.1.1
58921  * Copyright(c) 2006-2007, Ext JS, LLC.
58922  *
58923  * Originally Released Under LGPL - original licence link has changed is not relivant.
58924  *
58925  * Fork - LGPL
58926  * <script type="text/javascript">
58927  */
58928 /**
58929  * @class Roo.grid.CellSelectionModel
58930  * @extends Roo.grid.AbstractSelectionModel
58931  * This class provides the basic implementation for cell selection in a grid.
58932  * @constructor
58933  * @param {Object} config The object containing the configuration of this model.
58934  * @cfg {Boolean} enter_is_tab Enter behaves the same as tab. (eg. goes to next cell) default: false
58935  */
58936 Roo.grid.CellSelectionModel = function(config){
58937     Roo.apply(this, config);
58938
58939     this.selection = null;
58940
58941     this.addEvents({
58942         /**
58943              * @event beforerowselect
58944              * Fires before a cell is selected.
58945              * @param {SelectionModel} this
58946              * @param {Number} rowIndex The selected row index
58947              * @param {Number} colIndex The selected cell index
58948              */
58949             "beforecellselect" : true,
58950         /**
58951              * @event cellselect
58952              * Fires when a cell is selected.
58953              * @param {SelectionModel} this
58954              * @param {Number} rowIndex The selected row index
58955              * @param {Number} colIndex The selected cell index
58956              */
58957             "cellselect" : true,
58958         /**
58959              * @event selectionchange
58960              * Fires when the active selection changes.
58961              * @param {SelectionModel} this
58962              * @param {Object} selection null for no selection or an object (o) with two properties
58963                 <ul>
58964                 <li>o.record: the record object for the row the selection is in</li>
58965                 <li>o.cell: An array of [rowIndex, columnIndex]</li>
58966                 </ul>
58967              */
58968             "selectionchange" : true,
58969         /**
58970              * @event tabend
58971              * Fires when the tab (or enter) was pressed on the last editable cell
58972              * You can use this to trigger add new row.
58973              * @param {SelectionModel} this
58974              */
58975             "tabend" : true,
58976          /**
58977              * @event beforeeditnext
58978              * Fires before the next editable sell is made active
58979              * You can use this to skip to another cell or fire the tabend
58980              *    if you set cell to false
58981              * @param {Object} eventdata object : { cell : [ row, col ] } 
58982              */
58983             "beforeeditnext" : true
58984     });
58985     Roo.grid.CellSelectionModel.superclass.constructor.call(this);
58986 };
58987
58988 Roo.extend(Roo.grid.CellSelectionModel, Roo.grid.AbstractSelectionModel,  {
58989     
58990     enter_is_tab: false,
58991
58992     /** @ignore */
58993     initEvents : function(){
58994         this.grid.on("mousedown", this.handleMouseDown, this);
58995         this.grid.getGridEl().on(Roo.isIE ? "keydown" : "keypress", this.handleKeyDown, this);
58996         var view = this.grid.view;
58997         view.on("refresh", this.onViewChange, this);
58998         view.on("rowupdated", this.onRowUpdated, this);
58999         view.on("beforerowremoved", this.clearSelections, this);
59000         view.on("beforerowsinserted", this.clearSelections, this);
59001         if(this.grid.isEditor){
59002             this.grid.on("beforeedit", this.beforeEdit,  this);
59003         }
59004     },
59005
59006         //private
59007     beforeEdit : function(e){
59008         this.select(e.row, e.column, false, true, e.record);
59009     },
59010
59011         //private
59012     onRowUpdated : function(v, index, r){
59013         if(this.selection && this.selection.record == r){
59014             v.onCellSelect(index, this.selection.cell[1]);
59015         }
59016     },
59017
59018         //private
59019     onViewChange : function(){
59020         this.clearSelections(true);
59021     },
59022
59023         /**
59024          * Returns the currently selected cell,.
59025          * @return {Array} The selected cell (row, column) or null if none selected.
59026          */
59027     getSelectedCell : function(){
59028         return this.selection ? this.selection.cell : null;
59029     },
59030
59031     /**
59032      * Clears all selections.
59033      * @param {Boolean} true to prevent the gridview from being notified about the change.
59034      */
59035     clearSelections : function(preventNotify){
59036         var s = this.selection;
59037         if(s){
59038             if(preventNotify !== true){
59039                 this.grid.view.onCellDeselect(s.cell[0], s.cell[1]);
59040             }
59041             this.selection = null;
59042             this.fireEvent("selectionchange", this, null);
59043         }
59044     },
59045
59046     /**
59047      * Returns true if there is a selection.
59048      * @return {Boolean}
59049      */
59050     hasSelection : function(){
59051         return this.selection ? true : false;
59052     },
59053
59054     /** @ignore */
59055     handleMouseDown : function(e, t){
59056         var v = this.grid.getView();
59057         if(this.isLocked()){
59058             return;
59059         };
59060         var row = v.findRowIndex(t);
59061         var cell = v.findCellIndex(t);
59062         if(row !== false && cell !== false){
59063             this.select(row, cell);
59064         }
59065     },
59066
59067     /**
59068      * Selects a cell.
59069      * @param {Number} rowIndex
59070      * @param {Number} collIndex
59071      */
59072     select : function(rowIndex, colIndex, preventViewNotify, preventFocus, /*internal*/ r){
59073         if(this.fireEvent("beforecellselect", this, rowIndex, colIndex) !== false){
59074             this.clearSelections();
59075             r = r || this.grid.dataSource.getAt(rowIndex);
59076             this.selection = {
59077                 record : r,
59078                 cell : [rowIndex, colIndex]
59079             };
59080             if(!preventViewNotify){
59081                 var v = this.grid.getView();
59082                 v.onCellSelect(rowIndex, colIndex);
59083                 if(preventFocus !== true){
59084                     v.focusCell(rowIndex, colIndex);
59085                 }
59086             }
59087             this.fireEvent("cellselect", this, rowIndex, colIndex);
59088             this.fireEvent("selectionchange", this, this.selection);
59089         }
59090     },
59091
59092         //private
59093     isSelectable : function(rowIndex, colIndex, cm){
59094         return !cm.isHidden(colIndex);
59095     },
59096
59097     /** @ignore */
59098     handleKeyDown : function(e){
59099         //Roo.log('Cell Sel Model handleKeyDown');
59100         if(!e.isNavKeyPress()){
59101             return;
59102         }
59103         var g = this.grid, s = this.selection;
59104         if(!s){
59105             e.stopEvent();
59106             var cell = g.walkCells(0, 0, 1, this.isSelectable,  this);
59107             if(cell){
59108                 this.select(cell[0], cell[1]);
59109             }
59110             return;
59111         }
59112         var sm = this;
59113         var walk = function(row, col, step){
59114             return g.walkCells(row, col, step, sm.isSelectable,  sm);
59115         };
59116         var k = e.getKey(), r = s.cell[0], c = s.cell[1];
59117         var newCell;
59118
59119       
59120
59121         switch(k){
59122             case e.TAB:
59123                 // handled by onEditorKey
59124                 if (g.isEditor && g.editing) {
59125                     return;
59126                 }
59127                 if(e.shiftKey) {
59128                     newCell = walk(r, c-1, -1);
59129                 } else {
59130                     newCell = walk(r, c+1, 1);
59131                 }
59132                 break;
59133             
59134             case e.DOWN:
59135                newCell = walk(r+1, c, 1);
59136                 break;
59137             
59138             case e.UP:
59139                 newCell = walk(r-1, c, -1);
59140                 break;
59141             
59142             case e.RIGHT:
59143                 newCell = walk(r, c+1, 1);
59144                 break;
59145             
59146             case e.LEFT:
59147                 newCell = walk(r, c-1, -1);
59148                 break;
59149             
59150             case e.ENTER:
59151                 
59152                 if(g.isEditor && !g.editing){
59153                    g.startEditing(r, c);
59154                    e.stopEvent();
59155                    return;
59156                 }
59157                 
59158                 
59159              break;
59160         };
59161         if(newCell){
59162             this.select(newCell[0], newCell[1]);
59163             e.stopEvent();
59164             
59165         }
59166     },
59167
59168     acceptsNav : function(row, col, cm){
59169         return !cm.isHidden(col) && cm.isCellEditable(col, row);
59170     },
59171     /**
59172      * Selects a cell.
59173      * @param {Number} field (not used) - as it's normally used as a listener
59174      * @param {Number} e - event - fake it by using
59175      *
59176      * var e = Roo.EventObjectImpl.prototype;
59177      * e.keyCode = e.TAB
59178      *
59179      * 
59180      */
59181     onEditorKey : function(field, e){
59182         
59183         var k = e.getKey(),
59184             newCell,
59185             g = this.grid,
59186             ed = g.activeEditor,
59187             forward = false;
59188         ///Roo.log('onEditorKey' + k);
59189         
59190         
59191         if (this.enter_is_tab && k == e.ENTER) {
59192             k = e.TAB;
59193         }
59194         
59195         if(k == e.TAB){
59196             if(e.shiftKey){
59197                 newCell = g.walkCells(ed.row, ed.col-1, -1, this.acceptsNav, this);
59198             }else{
59199                 newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59200                 forward = true;
59201             }
59202             
59203             e.stopEvent();
59204             
59205         } else if(k == e.ENTER &&  !e.ctrlKey){
59206             ed.completeEdit();
59207             e.stopEvent();
59208             newCell = g.walkCells(ed.row, ed.col+1, 1, this.acceptsNav, this);
59209         
59210                 } else if(k == e.ESC){
59211             ed.cancelEdit();
59212         }
59213                 
59214         if (newCell) {
59215             var ecall = { cell : newCell, forward : forward };
59216             this.fireEvent('beforeeditnext', ecall );
59217             newCell = ecall.cell;
59218                         forward = ecall.forward;
59219         }
59220                 
59221         if(newCell){
59222             //Roo.log('next cell after edit');
59223             g.startEditing.defer(100, g, [newCell[0], newCell[1]]);
59224         } else if (forward) {
59225             // tabbed past last
59226             this.fireEvent.defer(100, this, ['tabend',this]);
59227         }
59228     }
59229 });/*
59230  * Based on:
59231  * Ext JS Library 1.1.1
59232  * Copyright(c) 2006-2007, Ext JS, LLC.
59233  *
59234  * Originally Released Under LGPL - original licence link has changed is not relivant.
59235  *
59236  * Fork - LGPL
59237  * <script type="text/javascript">
59238  */
59239  
59240 /**
59241  * @class Roo.grid.EditorGrid
59242  * @extends Roo.grid.Grid
59243  * Class for creating and editable grid.
59244  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered - 
59245  * The container MUST have some type of size defined for the grid to fill. The container will be 
59246  * automatically set to position relative if it isn't already.
59247  * @param {Object} dataSource The data model to bind to
59248  * @param {Object} colModel The column model with info about this grid's columns
59249  */
59250 Roo.grid.EditorGrid = function(container, config){
59251     Roo.grid.EditorGrid.superclass.constructor.call(this, container, config);
59252     this.getGridEl().addClass("xedit-grid");
59253
59254     if(!this.selModel){
59255         this.selModel = new Roo.grid.CellSelectionModel();
59256     }
59257
59258     this.activeEditor = null;
59259
59260         this.addEvents({
59261             /**
59262              * @event beforeedit
59263              * Fires before cell editing is triggered. The edit event object has the following properties <br />
59264              * <ul style="padding:5px;padding-left:16px;">
59265              * <li>grid - This grid</li>
59266              * <li>record - The record being edited</li>
59267              * <li>field - The field name being edited</li>
59268              * <li>value - The value for the field being edited.</li>
59269              * <li>row - The grid row index</li>
59270              * <li>column - The grid column index</li>
59271              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59272              * </ul>
59273              * @param {Object} e An edit event (see above for description)
59274              */
59275             "beforeedit" : true,
59276             /**
59277              * @event afteredit
59278              * Fires after a cell is edited. <br />
59279              * <ul style="padding:5px;padding-left:16px;">
59280              * <li>grid - This grid</li>
59281              * <li>record - The record being edited</li>
59282              * <li>field - The field name being edited</li>
59283              * <li>value - The value being set</li>
59284              * <li>originalValue - The original value for the field, before the edit.</li>
59285              * <li>row - The grid row index</li>
59286              * <li>column - The grid column index</li>
59287              * </ul>
59288              * @param {Object} e An edit event (see above for description)
59289              */
59290             "afteredit" : true,
59291             /**
59292              * @event validateedit
59293              * Fires after a cell is edited, but before the value is set in the record. 
59294          * You can use this to modify the value being set in the field, Return false
59295              * to cancel the change. The edit event object has the following properties <br />
59296              * <ul style="padding:5px;padding-left:16px;">
59297          * <li>editor - This editor</li>
59298              * <li>grid - This grid</li>
59299              * <li>record - The record being edited</li>
59300              * <li>field - The field name being edited</li>
59301              * <li>value - The value being set</li>
59302              * <li>originalValue - The original value for the field, before the edit.</li>
59303              * <li>row - The grid row index</li>
59304              * <li>column - The grid column index</li>
59305              * <li>cancel - Set this to true to cancel the edit or return false from your handler.</li>
59306              * </ul>
59307              * @param {Object} e An edit event (see above for description)
59308              */
59309             "validateedit" : true
59310         });
59311     this.on("bodyscroll", this.stopEditing,  this);
59312     this.on(this.clicksToEdit == 1 ? "cellclick" : "celldblclick", this.onCellDblClick,  this);
59313 };
59314
59315 Roo.extend(Roo.grid.EditorGrid, Roo.grid.Grid, {
59316     /**
59317      * @cfg {Number} clicksToEdit
59318      * The number of clicks on a cell required to display the cell's editor (defaults to 2)
59319      */
59320     clicksToEdit: 2,
59321
59322     // private
59323     isEditor : true,
59324     // private
59325     trackMouseOver: false, // causes very odd FF errors
59326
59327     onCellDblClick : function(g, row, col){
59328         this.startEditing(row, col);
59329     },
59330
59331     onEditComplete : function(ed, value, startValue){
59332         this.editing = false;
59333         this.activeEditor = null;
59334         ed.un("specialkey", this.selModel.onEditorKey, this.selModel);
59335         var r = ed.record;
59336         var field = this.colModel.getDataIndex(ed.col);
59337         var e = {
59338             grid: this,
59339             record: r,
59340             field: field,
59341             originalValue: startValue,
59342             value: value,
59343             row: ed.row,
59344             column: ed.col,
59345             cancel:false,
59346             editor: ed
59347         };
59348         var cell = Roo.get(this.view.getCell(ed.row,ed.col));
59349         cell.show();
59350           
59351         if(String(value) !== String(startValue)){
59352             
59353             if(this.fireEvent("validateedit", e) !== false && !e.cancel){
59354                 r.set(field, e.value);
59355                 // if we are dealing with a combo box..
59356                 // then we also set the 'name' colum to be the displayField
59357                 if (ed.field.displayField && ed.field.name) {
59358                     r.set(ed.field.name, ed.field.el.dom.value);
59359                 }
59360                 
59361                 delete e.cancel; //?? why!!!
59362                 this.fireEvent("afteredit", e);
59363             }
59364         } else {
59365             this.fireEvent("afteredit", e); // always fire it!
59366         }
59367         this.view.focusCell(ed.row, ed.col);
59368     },
59369
59370     /**
59371      * Starts editing the specified for the specified row/column
59372      * @param {Number} rowIndex
59373      * @param {Number} colIndex
59374      */
59375     startEditing : function(row, col){
59376         this.stopEditing();
59377         if(this.colModel.isCellEditable(col, row)){
59378             this.view.ensureVisible(row, col, true);
59379           
59380             var r = this.dataSource.getAt(row);
59381             var field = this.colModel.getDataIndex(col);
59382             var cell = Roo.get(this.view.getCell(row,col));
59383             var e = {
59384                 grid: this,
59385                 record: r,
59386                 field: field,
59387                 value: r.data[field],
59388                 row: row,
59389                 column: col,
59390                 cancel:false 
59391             };
59392             if(this.fireEvent("beforeedit", e) !== false && !e.cancel){
59393                 this.editing = true;
59394                 var ed = this.colModel.getCellEditor(col, row);
59395                 
59396                 if (!ed) {
59397                     return;
59398                 }
59399                 if(!ed.rendered){
59400                     ed.render(ed.parentEl || document.body);
59401                 }
59402                 ed.field.reset();
59403                
59404                 cell.hide();
59405                 
59406                 (function(){ // complex but required for focus issues in safari, ie and opera
59407                     ed.row = row;
59408                     ed.col = col;
59409                     ed.record = r;
59410                     ed.on("complete",   this.onEditComplete,        this,       {single: true});
59411                     ed.on("specialkey", this.selModel.onEditorKey,  this.selModel);
59412                     this.activeEditor = ed;
59413                     var v = r.data[field];
59414                     ed.startEdit(this.view.getCell(row, col), v);
59415                     // combo's with 'displayField and name set
59416                     if (ed.field.displayField && ed.field.name) {
59417                         ed.field.el.dom.value = r.data[ed.field.name];
59418                     }
59419                     
59420                     
59421                 }).defer(50, this);
59422             }
59423         }
59424     },
59425         
59426     /**
59427      * Stops any active editing
59428      */
59429     stopEditing : function(){
59430         if(this.activeEditor){
59431             this.activeEditor.completeEdit();
59432         }
59433         this.activeEditor = null;
59434     },
59435         
59436          /**
59437      * Called to get grid's drag proxy text, by default returns this.ddText.
59438      * @return {String}
59439      */
59440     getDragDropText : function(){
59441         var count = this.selModel.getSelectedCell() ? 1 : 0;
59442         return String.format(this.ddText, count, count == 1 ? '' : 's');
59443     }
59444         
59445 });/*
59446  * Based on:
59447  * Ext JS Library 1.1.1
59448  * Copyright(c) 2006-2007, Ext JS, LLC.
59449  *
59450  * Originally Released Under LGPL - original licence link has changed is not relivant.
59451  *
59452  * Fork - LGPL
59453  * <script type="text/javascript">
59454  */
59455
59456 // private - not really -- you end up using it !
59457 // This is a support class used internally by the Grid components
59458
59459 /**
59460  * @class Roo.grid.GridEditor
59461  * @extends Roo.Editor
59462  * Class for creating and editable grid elements.
59463  * @param {Object} config any settings (must include field)
59464  */
59465 Roo.grid.GridEditor = function(field, config){
59466     if (!config && field.field) {
59467         config = field;
59468         field = Roo.factory(config.field, Roo.form);
59469     }
59470     Roo.grid.GridEditor.superclass.constructor.call(this, field, config);
59471     field.monitorTab = false;
59472 };
59473
59474 Roo.extend(Roo.grid.GridEditor, Roo.Editor, {
59475     
59476     /**
59477      * @cfg {Roo.form.Field} field Field to wrap (or xtyped)
59478      */
59479     
59480     alignment: "tl-tl",
59481     autoSize: "width",
59482     hideEl : false,
59483     cls: "x-small-editor x-grid-editor",
59484     shim:false,
59485     shadow:"frame"
59486 });/*
59487  * Based on:
59488  * Ext JS Library 1.1.1
59489  * Copyright(c) 2006-2007, Ext JS, LLC.
59490  *
59491  * Originally Released Under LGPL - original licence link has changed is not relivant.
59492  *
59493  * Fork - LGPL
59494  * <script type="text/javascript">
59495  */
59496   
59497
59498   
59499 Roo.grid.PropertyRecord = Roo.data.Record.create([
59500     {name:'name',type:'string'},  'value'
59501 ]);
59502
59503
59504 Roo.grid.PropertyStore = function(grid, source){
59505     this.grid = grid;
59506     this.store = new Roo.data.Store({
59507         recordType : Roo.grid.PropertyRecord
59508     });
59509     this.store.on('update', this.onUpdate,  this);
59510     if(source){
59511         this.setSource(source);
59512     }
59513     Roo.grid.PropertyStore.superclass.constructor.call(this);
59514 };
59515
59516
59517
59518 Roo.extend(Roo.grid.PropertyStore, Roo.util.Observable, {
59519     setSource : function(o){
59520         this.source = o;
59521         this.store.removeAll();
59522         var data = [];
59523         for(var k in o){
59524             if(this.isEditableValue(o[k])){
59525                 data.push(new Roo.grid.PropertyRecord({name: k, value: o[k]}, k));
59526             }
59527         }
59528         this.store.loadRecords({records: data}, {}, true);
59529     },
59530
59531     onUpdate : function(ds, record, type){
59532         if(type == Roo.data.Record.EDIT){
59533             var v = record.data['value'];
59534             var oldValue = record.modified['value'];
59535             if(this.grid.fireEvent('beforepropertychange', this.source, record.id, v, oldValue) !== false){
59536                 this.source[record.id] = v;
59537                 record.commit();
59538                 this.grid.fireEvent('propertychange', this.source, record.id, v, oldValue);
59539             }else{
59540                 record.reject();
59541             }
59542         }
59543     },
59544
59545     getProperty : function(row){
59546        return this.store.getAt(row);
59547     },
59548
59549     isEditableValue: function(val){
59550         if(val && val instanceof Date){
59551             return true;
59552         }else if(typeof val == 'object' || typeof val == 'function'){
59553             return false;
59554         }
59555         return true;
59556     },
59557
59558     setValue : function(prop, value){
59559         this.source[prop] = value;
59560         this.store.getById(prop).set('value', value);
59561     },
59562
59563     getSource : function(){
59564         return this.source;
59565     }
59566 });
59567
59568 Roo.grid.PropertyColumnModel = function(grid, store){
59569     this.grid = grid;
59570     var g = Roo.grid;
59571     g.PropertyColumnModel.superclass.constructor.call(this, [
59572         {header: this.nameText, sortable: true, dataIndex:'name', id: 'name'},
59573         {header: this.valueText, resizable:false, dataIndex: 'value', id: 'value'}
59574     ]);
59575     this.store = store;
59576     this.bselect = Roo.DomHelper.append(document.body, {
59577         tag: 'select', style:'display:none', cls: 'x-grid-editor', children: [
59578             {tag: 'option', value: 'true', html: 'true'},
59579             {tag: 'option', value: 'false', html: 'false'}
59580         ]
59581     });
59582     Roo.id(this.bselect);
59583     var f = Roo.form;
59584     this.editors = {
59585         'date' : new g.GridEditor(new f.DateField({selectOnFocus:true})),
59586         'string' : new g.GridEditor(new f.TextField({selectOnFocus:true})),
59587         'number' : new g.GridEditor(new f.NumberField({selectOnFocus:true, style:'text-align:left;'})),
59588         'int' : new g.GridEditor(new f.NumberField({selectOnFocus:true, allowDecimals:false, style:'text-align:left;'})),
59589         'boolean' : new g.GridEditor(new f.Field({el:this.bselect,selectOnFocus:true}))
59590     };
59591     this.renderCellDelegate = this.renderCell.createDelegate(this);
59592     this.renderPropDelegate = this.renderProp.createDelegate(this);
59593 };
59594
59595 Roo.extend(Roo.grid.PropertyColumnModel, Roo.grid.ColumnModel, {
59596     
59597     
59598     nameText : 'Name',
59599     valueText : 'Value',
59600     
59601     dateFormat : 'm/j/Y',
59602     
59603     
59604     renderDate : function(dateVal){
59605         return dateVal.dateFormat(this.dateFormat);
59606     },
59607
59608     renderBool : function(bVal){
59609         return bVal ? 'true' : 'false';
59610     },
59611
59612     isCellEditable : function(colIndex, rowIndex){
59613         return colIndex == 1;
59614     },
59615
59616     getRenderer : function(col){
59617         return col == 1 ?
59618             this.renderCellDelegate : this.renderPropDelegate;
59619     },
59620
59621     renderProp : function(v){
59622         return this.getPropertyName(v);
59623     },
59624
59625     renderCell : function(val){
59626         var rv = val;
59627         if(val instanceof Date){
59628             rv = this.renderDate(val);
59629         }else if(typeof val == 'boolean'){
59630             rv = this.renderBool(val);
59631         }
59632         return Roo.util.Format.htmlEncode(rv);
59633     },
59634
59635     getPropertyName : function(name){
59636         var pn = this.grid.propertyNames;
59637         return pn && pn[name] ? pn[name] : name;
59638     },
59639
59640     getCellEditor : function(colIndex, rowIndex){
59641         var p = this.store.getProperty(rowIndex);
59642         var n = p.data['name'], val = p.data['value'];
59643         
59644         if(typeof(this.grid.customEditors[n]) == 'string'){
59645             return this.editors[this.grid.customEditors[n]];
59646         }
59647         if(typeof(this.grid.customEditors[n]) != 'undefined'){
59648             return this.grid.customEditors[n];
59649         }
59650         if(val instanceof Date){
59651             return this.editors['date'];
59652         }else if(typeof val == 'number'){
59653             return this.editors['number'];
59654         }else if(typeof val == 'boolean'){
59655             return this.editors['boolean'];
59656         }else{
59657             return this.editors['string'];
59658         }
59659     }
59660 });
59661
59662 /**
59663  * @class Roo.grid.PropertyGrid
59664  * @extends Roo.grid.EditorGrid
59665  * This class represents the  interface of a component based property grid control.
59666  * <br><br>Usage:<pre><code>
59667  var grid = new Roo.grid.PropertyGrid("my-container-id", {
59668       
59669  });
59670  // set any options
59671  grid.render();
59672  * </code></pre>
59673   
59674  * @constructor
59675  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59676  * The container MUST have some type of size defined for the grid to fill. The container will be
59677  * automatically set to position relative if it isn't already.
59678  * @param {Object} config A config object that sets properties on this grid.
59679  */
59680 Roo.grid.PropertyGrid = function(container, config){
59681     config = config || {};
59682     var store = new Roo.grid.PropertyStore(this);
59683     this.store = store;
59684     var cm = new Roo.grid.PropertyColumnModel(this, store);
59685     store.store.sort('name', 'ASC');
59686     Roo.grid.PropertyGrid.superclass.constructor.call(this, container, Roo.apply({
59687         ds: store.store,
59688         cm: cm,
59689         enableColLock:false,
59690         enableColumnMove:false,
59691         stripeRows:false,
59692         trackMouseOver: false,
59693         clicksToEdit:1
59694     }, config));
59695     this.getGridEl().addClass('x-props-grid');
59696     this.lastEditRow = null;
59697     this.on('columnresize', this.onColumnResize, this);
59698     this.addEvents({
59699          /**
59700              * @event beforepropertychange
59701              * Fires before a property changes (return false to stop?)
59702              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59703              * @param {String} id Record Id
59704              * @param {String} newval New Value
59705          * @param {String} oldval Old Value
59706              */
59707         "beforepropertychange": true,
59708         /**
59709              * @event propertychange
59710              * Fires after a property changes
59711              * @param {Roo.grid.PropertyGrid} grid property grid? (check could be store)
59712              * @param {String} id Record Id
59713              * @param {String} newval New Value
59714          * @param {String} oldval Old Value
59715              */
59716         "propertychange": true
59717     });
59718     this.customEditors = this.customEditors || {};
59719 };
59720 Roo.extend(Roo.grid.PropertyGrid, Roo.grid.EditorGrid, {
59721     
59722      /**
59723      * @cfg {Object} customEditors map of colnames=> custom editors.
59724      * the custom editor can be one of the standard ones (date|string|number|int|boolean), or a
59725      * grid editor eg. Roo.grid.GridEditor(new Roo.form.TextArea({selectOnFocus:true})),
59726      * false disables editing of the field.
59727          */
59728     
59729       /**
59730      * @cfg {Object} propertyNames map of property Names to their displayed value
59731          */
59732     
59733     render : function(){
59734         Roo.grid.PropertyGrid.superclass.render.call(this);
59735         this.autoSize.defer(100, this);
59736     },
59737
59738     autoSize : function(){
59739         Roo.grid.PropertyGrid.superclass.autoSize.call(this);
59740         if(this.view){
59741             this.view.fitColumns();
59742         }
59743     },
59744
59745     onColumnResize : function(){
59746         this.colModel.setColumnWidth(1, this.container.getWidth(true)-this.colModel.getColumnWidth(0));
59747         this.autoSize();
59748     },
59749     /**
59750      * Sets the data for the Grid
59751      * accepts a Key => Value object of all the elements avaiable.
59752      * @param {Object} data  to appear in grid.
59753      */
59754     setSource : function(source){
59755         this.store.setSource(source);
59756         //this.autoSize();
59757     },
59758     /**
59759      * Gets all the data from the grid.
59760      * @return {Object} data  data stored in grid
59761      */
59762     getSource : function(){
59763         return this.store.getSource();
59764     }
59765 });/*
59766   
59767  * Licence LGPL
59768  
59769  */
59770  
59771 /**
59772  * @class Roo.grid.Calendar
59773  * @extends Roo.util.Grid
59774  * This class extends the Grid to provide a calendar widget
59775  * <br><br>Usage:<pre><code>
59776  var grid = new Roo.grid.Calendar("my-container-id", {
59777      ds: myDataStore,
59778      cm: myColModel,
59779      selModel: mySelectionModel,
59780      autoSizeColumns: true,
59781      monitorWindowResize: false,
59782      trackMouseOver: true
59783      eventstore : real data store..
59784  });
59785  // set any options
59786  grid.render();
59787   
59788   * @constructor
59789  * @param {String/HTMLElement/Roo.Element} container The element into which this grid will be rendered -
59790  * The container MUST have some type of size defined for the grid to fill. The container will be
59791  * automatically set to position relative if it isn't already.
59792  * @param {Object} config A config object that sets properties on this grid.
59793  */
59794 Roo.grid.Calendar = function(container, config){
59795         // initialize the container
59796         this.container = Roo.get(container);
59797         this.container.update("");
59798         this.container.setStyle("overflow", "hidden");
59799     this.container.addClass('x-grid-container');
59800
59801     this.id = this.container.id;
59802
59803     Roo.apply(this, config);
59804     // check and correct shorthanded configs
59805     
59806     var rows = [];
59807     var d =1;
59808     for (var r = 0;r < 6;r++) {
59809         
59810         rows[r]=[];
59811         for (var c =0;c < 7;c++) {
59812             rows[r][c]= '';
59813         }
59814     }
59815     if (this.eventStore) {
59816         this.eventStore= Roo.factory(this.eventStore, Roo.data);
59817         this.eventStore.on('load',this.onLoad, this);
59818         this.eventStore.on('beforeload',this.clearEvents, this);
59819          
59820     }
59821     
59822     this.dataSource = new Roo.data.Store({
59823             proxy: new Roo.data.MemoryProxy(rows),
59824             reader: new Roo.data.ArrayReader({}, [
59825                    'weekday0', 'weekday1', 'weekday2', 'weekday3', 'weekday4', 'weekday5', 'weekday6' ])
59826     });
59827
59828     this.dataSource.load();
59829     this.ds = this.dataSource;
59830     this.ds.xmodule = this.xmodule || false;
59831     
59832     
59833     var cellRender = function(v,x,r)
59834     {
59835         return String.format(
59836             '<div class="fc-day  fc-widget-content"><div>' +
59837                 '<div class="fc-event-container"></div>' +
59838                 '<div class="fc-day-number">{0}</div>'+
59839                 
59840                 '<div class="fc-day-content"><div style="position:relative"></div></div>' +
59841             '</div></div>', v);
59842     
59843     }
59844     
59845     
59846     this.colModel = new Roo.grid.ColumnModel( [
59847         {
59848             xtype: 'ColumnModel',
59849             xns: Roo.grid,
59850             dataIndex : 'weekday0',
59851             header : 'Sunday',
59852             renderer : cellRender
59853         },
59854         {
59855             xtype: 'ColumnModel',
59856             xns: Roo.grid,
59857             dataIndex : 'weekday1',
59858             header : 'Monday',
59859             renderer : cellRender
59860         },
59861         {
59862             xtype: 'ColumnModel',
59863             xns: Roo.grid,
59864             dataIndex : 'weekday2',
59865             header : 'Tuesday',
59866             renderer : cellRender
59867         },
59868         {
59869             xtype: 'ColumnModel',
59870             xns: Roo.grid,
59871             dataIndex : 'weekday3',
59872             header : 'Wednesday',
59873             renderer : cellRender
59874         },
59875         {
59876             xtype: 'ColumnModel',
59877             xns: Roo.grid,
59878             dataIndex : 'weekday4',
59879             header : 'Thursday',
59880             renderer : cellRender
59881         },
59882         {
59883             xtype: 'ColumnModel',
59884             xns: Roo.grid,
59885             dataIndex : 'weekday5',
59886             header : 'Friday',
59887             renderer : cellRender
59888         },
59889         {
59890             xtype: 'ColumnModel',
59891             xns: Roo.grid,
59892             dataIndex : 'weekday6',
59893             header : 'Saturday',
59894             renderer : cellRender
59895         }
59896     ]);
59897     this.cm = this.colModel;
59898     this.cm.xmodule = this.xmodule || false;
59899  
59900         
59901           
59902     //this.selModel = new Roo.grid.CellSelectionModel();
59903     //this.sm = this.selModel;
59904     //this.selModel.init(this);
59905     
59906     
59907     if(this.width){
59908         this.container.setWidth(this.width);
59909     }
59910
59911     if(this.height){
59912         this.container.setHeight(this.height);
59913     }
59914     /** @private */
59915         this.addEvents({
59916         // raw events
59917         /**
59918          * @event click
59919          * The raw click event for the entire grid.
59920          * @param {Roo.EventObject} e
59921          */
59922         "click" : true,
59923         /**
59924          * @event dblclick
59925          * The raw dblclick event for the entire grid.
59926          * @param {Roo.EventObject} e
59927          */
59928         "dblclick" : true,
59929         /**
59930          * @event contextmenu
59931          * The raw contextmenu event for the entire grid.
59932          * @param {Roo.EventObject} e
59933          */
59934         "contextmenu" : true,
59935         /**
59936          * @event mousedown
59937          * The raw mousedown event for the entire grid.
59938          * @param {Roo.EventObject} e
59939          */
59940         "mousedown" : true,
59941         /**
59942          * @event mouseup
59943          * The raw mouseup event for the entire grid.
59944          * @param {Roo.EventObject} e
59945          */
59946         "mouseup" : true,
59947         /**
59948          * @event mouseover
59949          * The raw mouseover event for the entire grid.
59950          * @param {Roo.EventObject} e
59951          */
59952         "mouseover" : true,
59953         /**
59954          * @event mouseout
59955          * The raw mouseout event for the entire grid.
59956          * @param {Roo.EventObject} e
59957          */
59958         "mouseout" : true,
59959         /**
59960          * @event keypress
59961          * The raw keypress event for the entire grid.
59962          * @param {Roo.EventObject} e
59963          */
59964         "keypress" : true,
59965         /**
59966          * @event keydown
59967          * The raw keydown event for the entire grid.
59968          * @param {Roo.EventObject} e
59969          */
59970         "keydown" : true,
59971
59972         // custom events
59973
59974         /**
59975          * @event cellclick
59976          * Fires when a cell is clicked
59977          * @param {Grid} this
59978          * @param {Number} rowIndex
59979          * @param {Number} columnIndex
59980          * @param {Roo.EventObject} e
59981          */
59982         "cellclick" : true,
59983         /**
59984          * @event celldblclick
59985          * Fires when a cell is double clicked
59986          * @param {Grid} this
59987          * @param {Number} rowIndex
59988          * @param {Number} columnIndex
59989          * @param {Roo.EventObject} e
59990          */
59991         "celldblclick" : true,
59992         /**
59993          * @event rowclick
59994          * Fires when a row is clicked
59995          * @param {Grid} this
59996          * @param {Number} rowIndex
59997          * @param {Roo.EventObject} e
59998          */
59999         "rowclick" : true,
60000         /**
60001          * @event rowdblclick
60002          * Fires when a row is double clicked
60003          * @param {Grid} this
60004          * @param {Number} rowIndex
60005          * @param {Roo.EventObject} e
60006          */
60007         "rowdblclick" : true,
60008         /**
60009          * @event headerclick
60010          * Fires when a header is clicked
60011          * @param {Grid} this
60012          * @param {Number} columnIndex
60013          * @param {Roo.EventObject} e
60014          */
60015         "headerclick" : true,
60016         /**
60017          * @event headerdblclick
60018          * Fires when a header cell is double clicked
60019          * @param {Grid} this
60020          * @param {Number} columnIndex
60021          * @param {Roo.EventObject} e
60022          */
60023         "headerdblclick" : true,
60024         /**
60025          * @event rowcontextmenu
60026          * Fires when a row is right clicked
60027          * @param {Grid} this
60028          * @param {Number} rowIndex
60029          * @param {Roo.EventObject} e
60030          */
60031         "rowcontextmenu" : true,
60032         /**
60033          * @event cellcontextmenu
60034          * Fires when a cell is right clicked
60035          * @param {Grid} this
60036          * @param {Number} rowIndex
60037          * @param {Number} cellIndex
60038          * @param {Roo.EventObject} e
60039          */
60040          "cellcontextmenu" : true,
60041         /**
60042          * @event headercontextmenu
60043          * Fires when a header is right clicked
60044          * @param {Grid} this
60045          * @param {Number} columnIndex
60046          * @param {Roo.EventObject} e
60047          */
60048         "headercontextmenu" : true,
60049         /**
60050          * @event bodyscroll
60051          * Fires when the body element is scrolled
60052          * @param {Number} scrollLeft
60053          * @param {Number} scrollTop
60054          */
60055         "bodyscroll" : true,
60056         /**
60057          * @event columnresize
60058          * Fires when the user resizes a column
60059          * @param {Number} columnIndex
60060          * @param {Number} newSize
60061          */
60062         "columnresize" : true,
60063         /**
60064          * @event columnmove
60065          * Fires when the user moves a column
60066          * @param {Number} oldIndex
60067          * @param {Number} newIndex
60068          */
60069         "columnmove" : true,
60070         /**
60071          * @event startdrag
60072          * Fires when row(s) start being dragged
60073          * @param {Grid} this
60074          * @param {Roo.GridDD} dd The drag drop object
60075          * @param {event} e The raw browser event
60076          */
60077         "startdrag" : true,
60078         /**
60079          * @event enddrag
60080          * Fires when a drag operation is complete
60081          * @param {Grid} this
60082          * @param {Roo.GridDD} dd The drag drop object
60083          * @param {event} e The raw browser event
60084          */
60085         "enddrag" : true,
60086         /**
60087          * @event dragdrop
60088          * Fires when dragged row(s) are dropped on a valid DD target
60089          * @param {Grid} this
60090          * @param {Roo.GridDD} dd The drag drop object
60091          * @param {String} targetId The target drag drop object
60092          * @param {event} e The raw browser event
60093          */
60094         "dragdrop" : true,
60095         /**
60096          * @event dragover
60097          * Fires while row(s) are being dragged. "targetId" is the id of the Yahoo.util.DD object the selected rows are being dragged over.
60098          * @param {Grid} this
60099          * @param {Roo.GridDD} dd The drag drop object
60100          * @param {String} targetId The target drag drop object
60101          * @param {event} e The raw browser event
60102          */
60103         "dragover" : true,
60104         /**
60105          * @event dragenter
60106          *  Fires when the dragged row(s) first cross another DD target while being dragged
60107          * @param {Grid} this
60108          * @param {Roo.GridDD} dd The drag drop object
60109          * @param {String} targetId The target drag drop object
60110          * @param {event} e The raw browser event
60111          */
60112         "dragenter" : true,
60113         /**
60114          * @event dragout
60115          * Fires when the dragged row(s) leave another DD target while being dragged
60116          * @param {Grid} this
60117          * @param {Roo.GridDD} dd The drag drop object
60118          * @param {String} targetId The target drag drop object
60119          * @param {event} e The raw browser event
60120          */
60121         "dragout" : true,
60122         /**
60123          * @event rowclass
60124          * Fires when a row is rendered, so you can change add a style to it.
60125          * @param {GridView} gridview   The grid view
60126          * @param {Object} rowcfg   contains record  rowIndex and rowClass - set rowClass to add a style.
60127          */
60128         'rowclass' : true,
60129
60130         /**
60131          * @event render
60132          * Fires when the grid is rendered
60133          * @param {Grid} grid
60134          */
60135         'render' : true,
60136             /**
60137              * @event select
60138              * Fires when a date is selected
60139              * @param {DatePicker} this
60140              * @param {Date} date The selected date
60141              */
60142         'select': true,
60143         /**
60144              * @event monthchange
60145              * Fires when the displayed month changes 
60146              * @param {DatePicker} this
60147              * @param {Date} date The selected month
60148              */
60149         'monthchange': true,
60150         /**
60151              * @event evententer
60152              * Fires when mouse over an event
60153              * @param {Calendar} this
60154              * @param {event} Event
60155              */
60156         'evententer': true,
60157         /**
60158              * @event eventleave
60159              * Fires when the mouse leaves an
60160              * @param {Calendar} this
60161              * @param {event}
60162              */
60163         'eventleave': true,
60164         /**
60165              * @event eventclick
60166              * Fires when the mouse click an
60167              * @param {Calendar} this
60168              * @param {event}
60169              */
60170         'eventclick': true,
60171         /**
60172              * @event eventrender
60173              * Fires before each cell is rendered, so you can modify the contents, like cls / title / qtip
60174              * @param {Calendar} this
60175              * @param {data} data to be modified
60176              */
60177         'eventrender': true
60178         
60179     });
60180
60181     Roo.grid.Grid.superclass.constructor.call(this);
60182     this.on('render', function() {
60183         this.view.el.addClass('x-grid-cal'); 
60184         
60185         (function() { this.setDate(new Date()); }).defer(100,this); //default today..
60186
60187     },this);
60188     
60189     if (!Roo.grid.Calendar.style) {
60190         Roo.grid.Calendar.style = Roo.util.CSS.createStyleSheet({
60191             
60192             
60193             '.x-grid-cal .x-grid-col' :  {
60194                 height: 'auto !important',
60195                 'vertical-align': 'top'
60196             },
60197             '.x-grid-cal  .fc-event-hori' : {
60198                 height: '14px'
60199             }
60200              
60201             
60202         }, Roo.id());
60203     }
60204
60205     
60206     
60207 };
60208 Roo.extend(Roo.grid.Calendar, Roo.grid.Grid, {
60209     /**
60210      * @cfg {Store} eventStore The store that loads events.
60211      */
60212     eventStore : 25,
60213
60214      
60215     activeDate : false,
60216     startDay : 0,
60217     autoWidth : true,
60218     monitorWindowResize : false,
60219
60220     
60221     resizeColumns : function() {
60222         var col = (this.view.el.getWidth() / 7) - 3;
60223         // loop through cols, and setWidth
60224         for(var i =0 ; i < 7 ; i++){
60225             this.cm.setColumnWidth(i, col);
60226         }
60227     },
60228      setDate :function(date) {
60229         
60230         Roo.log('setDate?');
60231         
60232         this.resizeColumns();
60233         var vd = this.activeDate;
60234         this.activeDate = date;
60235 //        if(vd && this.el){
60236 //            var t = date.getTime();
60237 //            if(vd.getMonth() == date.getMonth() && vd.getFullYear() == date.getFullYear()){
60238 //                Roo.log('using add remove');
60239 //                
60240 //                this.fireEvent('monthchange', this, date);
60241 //                
60242 //                this.cells.removeClass("fc-state-highlight");
60243 //                this.cells.each(function(c){
60244 //                   if(c.dateValue == t){
60245 //                       c.addClass("fc-state-highlight");
60246 //                       setTimeout(function(){
60247 //                            try{c.dom.firstChild.focus();}catch(e){}
60248 //                       }, 50);
60249 //                       return false;
60250 //                   }
60251 //                   return true;
60252 //                });
60253 //                return;
60254 //            }
60255 //        }
60256         
60257         var days = date.getDaysInMonth();
60258         
60259         var firstOfMonth = date.getFirstDateOfMonth();
60260         var startingPos = firstOfMonth.getDay()-this.startDay;
60261         
60262         if(startingPos < this.startDay){
60263             startingPos += 7;
60264         }
60265         
60266         var pm = date.add(Date.MONTH, -1);
60267         var prevStart = pm.getDaysInMonth()-startingPos;
60268 //        
60269         
60270         
60271         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60272         
60273         this.textNodes = this.view.el.query('.x-grid-row .x-grid-col .x-grid-cell-text');
60274         //this.cells.addClassOnOver('fc-state-hover');
60275         
60276         var cells = this.cells.elements;
60277         var textEls = this.textNodes;
60278         
60279         //Roo.each(cells, function(cell){
60280         //    cell.removeClass([ 'fc-past', 'fc-other-month', 'fc-future', 'fc-state-highlight', 'fc-state-disabled']);
60281         //});
60282         
60283         days += startingPos;
60284
60285         // convert everything to numbers so it's fast
60286         var day = 86400000;
60287         var d = (new Date(pm.getFullYear(), pm.getMonth(), prevStart)).clearTime();
60288         //Roo.log(d);
60289         //Roo.log(pm);
60290         //Roo.log(prevStart);
60291         
60292         var today = new Date().clearTime().getTime();
60293         var sel = date.clearTime().getTime();
60294         var min = this.minDate ? this.minDate.clearTime() : Number.NEGATIVE_INFINITY;
60295         var max = this.maxDate ? this.maxDate.clearTime() : Number.POSITIVE_INFINITY;
60296         var ddMatch = this.disabledDatesRE;
60297         var ddText = this.disabledDatesText;
60298         var ddays = this.disabledDays ? this.disabledDays.join("") : false;
60299         var ddaysText = this.disabledDaysText;
60300         var format = this.format;
60301         
60302         var setCellClass = function(cal, cell){
60303             
60304             //Roo.log('set Cell Class');
60305             cell.title = "";
60306             var t = d.getTime();
60307             
60308             //Roo.log(d);
60309             
60310             
60311             cell.dateValue = t;
60312             if(t == today){
60313                 cell.className += " fc-today";
60314                 cell.className += " fc-state-highlight";
60315                 cell.title = cal.todayText;
60316             }
60317             if(t == sel){
60318                 // disable highlight in other month..
60319                 cell.className += " fc-state-highlight";
60320                 
60321             }
60322             // disabling
60323             if(t < min) {
60324                 //cell.className = " fc-state-disabled";
60325                 cell.title = cal.minText;
60326                 return;
60327             }
60328             if(t > max) {
60329                 //cell.className = " fc-state-disabled";
60330                 cell.title = cal.maxText;
60331                 return;
60332             }
60333             if(ddays){
60334                 if(ddays.indexOf(d.getDay()) != -1){
60335                     // cell.title = ddaysText;
60336                    // cell.className = " fc-state-disabled";
60337                 }
60338             }
60339             if(ddMatch && format){
60340                 var fvalue = d.dateFormat(format);
60341                 if(ddMatch.test(fvalue)){
60342                     cell.title = ddText.replace("%0", fvalue);
60343                    cell.className = " fc-state-disabled";
60344                 }
60345             }
60346             
60347             if (!cell.initialClassName) {
60348                 cell.initialClassName = cell.dom.className;
60349             }
60350             
60351             cell.dom.className = cell.initialClassName  + ' ' +  cell.className;
60352         };
60353
60354         var i = 0;
60355         
60356         for(; i < startingPos; i++) {
60357             cells[i].dayName =  (++prevStart);
60358             Roo.log(textEls[i]);
60359             d.setDate(d.getDate()+1);
60360             
60361             //cells[i].className = "fc-past fc-other-month";
60362             setCellClass(this, cells[i]);
60363         }
60364         
60365         var intDay = 0;
60366         
60367         for(; i < days; i++){
60368             intDay = i - startingPos + 1;
60369             cells[i].dayName =  (intDay);
60370             d.setDate(d.getDate()+1);
60371             
60372             cells[i].className = ''; // "x-date-active";
60373             setCellClass(this, cells[i]);
60374         }
60375         var extraDays = 0;
60376         
60377         for(; i < 42; i++) {
60378             //textEls[i].innerHTML = (++extraDays);
60379             
60380             d.setDate(d.getDate()+1);
60381             cells[i].dayName = (++extraDays);
60382             cells[i].className = "fc-future fc-other-month";
60383             setCellClass(this, cells[i]);
60384         }
60385         
60386         //this.el.select('.fc-header-title h2',true).update(Date.monthNames[date.getMonth()] + " " + date.getFullYear());
60387         
60388         var totalRows = Math.ceil((date.getDaysInMonth() + date.getFirstDateOfMonth().getDay()) / 7);
60389         
60390         // this will cause all the cells to mis
60391         var rows= [];
60392         var i =0;
60393         for (var r = 0;r < 6;r++) {
60394             for (var c =0;c < 7;c++) {
60395                 this.ds.getAt(r).set('weekday' + c ,cells[i++].dayName );
60396             }    
60397         }
60398         
60399         this.cells = this.view.el.select('.x-grid-row .x-grid-col',true);
60400         for(i=0;i<cells.length;i++) {
60401             
60402             this.cells.elements[i].dayName = cells[i].dayName ;
60403             this.cells.elements[i].className = cells[i].className;
60404             this.cells.elements[i].initialClassName = cells[i].initialClassName ;
60405             this.cells.elements[i].title = cells[i].title ;
60406             this.cells.elements[i].dateValue = cells[i].dateValue ;
60407         }
60408         
60409         
60410         
60411         
60412         //this.el.select('tr.fc-week.fc-prev-last',true).removeClass('fc-last');
60413         //this.el.select('tr.fc-week.fc-next-last',true).addClass('fc-last').show();
60414         
60415         ////if(totalRows != 6){
60416             //this.el.select('tr.fc-week.fc-last',true).removeClass('fc-last').addClass('fc-next-last').hide();
60417            // this.el.select('tr.fc-week.fc-prev-last',true).addClass('fc-last');
60418        // }
60419         
60420         this.fireEvent('monthchange', this, date);
60421         
60422         
60423     },
60424  /**
60425      * Returns the grid's SelectionModel.
60426      * @return {SelectionModel}
60427      */
60428     getSelectionModel : function(){
60429         if(!this.selModel){
60430             this.selModel = new Roo.grid.CellSelectionModel();
60431         }
60432         return this.selModel;
60433     },
60434
60435     load: function() {
60436         this.eventStore.load()
60437         
60438         
60439         
60440     },
60441     
60442     findCell : function(dt) {
60443         dt = dt.clearTime().getTime();
60444         var ret = false;
60445         this.cells.each(function(c){
60446             //Roo.log("check " +c.dateValue + '?=' + dt);
60447             if(c.dateValue == dt){
60448                 ret = c;
60449                 return false;
60450             }
60451             return true;
60452         });
60453         
60454         return ret;
60455     },
60456     
60457     findCells : function(rec) {
60458         var s = rec.data.start_dt.clone().clearTime().getTime();
60459        // Roo.log(s);
60460         var e= rec.data.end_dt.clone().clearTime().getTime();
60461        // Roo.log(e);
60462         var ret = [];
60463         this.cells.each(function(c){
60464              ////Roo.log("check " +c.dateValue + '<' + e + ' > ' + s);
60465             
60466             if(c.dateValue > e){
60467                 return ;
60468             }
60469             if(c.dateValue < s){
60470                 return ;
60471             }
60472             ret.push(c);
60473         });
60474         
60475         return ret;    
60476     },
60477     
60478     findBestRow: function(cells)
60479     {
60480         var ret = 0;
60481         
60482         for (var i =0 ; i < cells.length;i++) {
60483             ret  = Math.max(cells[i].rows || 0,ret);
60484         }
60485         return ret;
60486         
60487     },
60488     
60489     
60490     addItem : function(rec)
60491     {
60492         // look for vertical location slot in
60493         var cells = this.findCells(rec);
60494         
60495         rec.row = this.findBestRow(cells);
60496         
60497         // work out the location.
60498         
60499         var crow = false;
60500         var rows = [];
60501         for(var i =0; i < cells.length; i++) {
60502             if (!crow) {
60503                 crow = {
60504                     start : cells[i],
60505                     end :  cells[i]
60506                 };
60507                 continue;
60508             }
60509             if (crow.start.getY() == cells[i].getY()) {
60510                 // on same row.
60511                 crow.end = cells[i];
60512                 continue;
60513             }
60514             // different row.
60515             rows.push(crow);
60516             crow = {
60517                 start: cells[i],
60518                 end : cells[i]
60519             };
60520             
60521         }
60522         
60523         rows.push(crow);
60524         rec.els = [];
60525         rec.rows = rows;
60526         rec.cells = cells;
60527         for (var i = 0; i < cells.length;i++) {
60528             cells[i].rows = Math.max(cells[i].rows || 0 , rec.row + 1 );
60529             
60530         }
60531         
60532         
60533     },
60534     
60535     clearEvents: function() {
60536         
60537         if (!this.eventStore.getCount()) {
60538             return;
60539         }
60540         // reset number of rows in cells.
60541         Roo.each(this.cells.elements, function(c){
60542             c.rows = 0;
60543         });
60544         
60545         this.eventStore.each(function(e) {
60546             this.clearEvent(e);
60547         },this);
60548         
60549     },
60550     
60551     clearEvent : function(ev)
60552     {
60553         if (ev.els) {
60554             Roo.each(ev.els, function(el) {
60555                 el.un('mouseenter' ,this.onEventEnter, this);
60556                 el.un('mouseleave' ,this.onEventLeave, this);
60557                 el.remove();
60558             },this);
60559             ev.els = [];
60560         }
60561     },
60562     
60563     
60564     renderEvent : function(ev,ctr) {
60565         if (!ctr) {
60566              ctr = this.view.el.select('.fc-event-container',true).first();
60567         }
60568         
60569          
60570         this.clearEvent(ev);
60571             //code
60572        
60573         
60574         
60575         ev.els = [];
60576         var cells = ev.cells;
60577         var rows = ev.rows;
60578         this.fireEvent('eventrender', this, ev);
60579         
60580         for(var i =0; i < rows.length; i++) {
60581             
60582             cls = '';
60583             if (i == 0) {
60584                 cls += ' fc-event-start';
60585             }
60586             if ((i+1) == rows.length) {
60587                 cls += ' fc-event-end';
60588             }
60589             
60590             //Roo.log(ev.data);
60591             // how many rows should it span..
60592             var cg = this.eventTmpl.append(ctr,Roo.apply({
60593                 fccls : cls
60594                 
60595             }, ev.data) , true);
60596             
60597             
60598             cg.on('mouseenter' ,this.onEventEnter, this, ev);
60599             cg.on('mouseleave' ,this.onEventLeave, this, ev);
60600             cg.on('click', this.onEventClick, this, ev);
60601             
60602             ev.els.push(cg);
60603             
60604             var sbox = rows[i].start.select('.fc-day-content',true).first().getBox();
60605             var ebox = rows[i].end.select('.fc-day-content',true).first().getBox();
60606             //Roo.log(cg);
60607              
60608             cg.setXY([sbox.x +2, sbox.y +(ev.row * 20)]);    
60609             cg.setWidth(ebox.right - sbox.x -2);
60610         }
60611     },
60612     
60613     renderEvents: function()
60614     {   
60615         // first make sure there is enough space..
60616         
60617         if (!this.eventTmpl) {
60618             this.eventTmpl = new Roo.Template(
60619                 '<div class="roo-dynamic fc-event fc-event-hori fc-event-draggable ui-draggable {fccls} {cls}"  style="position: absolute" unselectable="on">' +
60620                     '<div class="fc-event-inner">' +
60621                         '<span class="fc-event-time">{time}</span>' +
60622                         '<span class="fc-event-title" qtip="{qtip}">{title}</span>' +
60623                     '</div>' +
60624                     '<div class="ui-resizable-heandle ui-resizable-e">&nbsp;&nbsp;&nbsp;</div>' +
60625                 '</div>'
60626             );
60627                 
60628         }
60629                
60630         
60631         
60632         this.cells.each(function(c) {
60633             //Roo.log(c.select('.fc-day-content div',true).first());
60634             c.select('.fc-day-content div',true).first().setHeight(Math.max(34, (c.rows || 1) * 20));
60635         });
60636         
60637         var ctr = this.view.el.select('.fc-event-container',true).first();
60638         
60639         var cls;
60640         this.eventStore.each(function(ev){
60641             
60642             this.renderEvent(ev);
60643              
60644              
60645         }, this);
60646         this.view.layout();
60647         
60648     },
60649     
60650     onEventEnter: function (e, el,event,d) {
60651         this.fireEvent('evententer', this, el, event);
60652     },
60653     
60654     onEventLeave: function (e, el,event,d) {
60655         this.fireEvent('eventleave', this, el, event);
60656     },
60657     
60658     onEventClick: function (e, el,event,d) {
60659         this.fireEvent('eventclick', this, el, event);
60660     },
60661     
60662     onMonthChange: function () {
60663         this.store.load();
60664     },
60665     
60666     onLoad: function () {
60667         
60668         //Roo.log('calendar onload');
60669 //         
60670         if(this.eventStore.getCount() > 0){
60671             
60672            
60673             
60674             this.eventStore.each(function(d){
60675                 
60676                 
60677                 // FIXME..
60678                 var add =   d.data;
60679                 if (typeof(add.end_dt) == 'undefined')  {
60680                     Roo.log("Missing End time in calendar data: ");
60681                     Roo.log(d);
60682                     return;
60683                 }
60684                 if (typeof(add.start_dt) == 'undefined')  {
60685                     Roo.log("Missing Start time in calendar data: ");
60686                     Roo.log(d);
60687                     return;
60688                 }
60689                 add.start_dt = typeof(add.start_dt) == 'string' ? Date.parseDate(add.start_dt,'Y-m-d H:i:s') : add.start_dt,
60690                 add.end_dt = typeof(add.end_dt) == 'string' ? Date.parseDate(add.end_dt,'Y-m-d H:i:s') : add.end_dt,
60691                 add.id = add.id || d.id;
60692                 add.title = add.title || '??';
60693                 
60694                 this.addItem(d);
60695                 
60696              
60697             },this);
60698         }
60699         
60700         this.renderEvents();
60701     }
60702     
60703
60704 });
60705 /*
60706  grid : {
60707                 xtype: 'Grid',
60708                 xns: Roo.grid,
60709                 listeners : {
60710                     render : function ()
60711                     {
60712                         _this.grid = this;
60713                         
60714                         if (!this.view.el.hasClass('course-timesheet')) {
60715                             this.view.el.addClass('course-timesheet');
60716                         }
60717                         if (this.tsStyle) {
60718                             this.ds.load({});
60719                             return; 
60720                         }
60721                         Roo.log('width');
60722                         Roo.log(_this.grid.view.el.getWidth());
60723                         
60724                         
60725                         this.tsStyle =  Roo.util.CSS.createStyleSheet({
60726                             '.course-timesheet .x-grid-row' : {
60727                                 height: '80px'
60728                             },
60729                             '.x-grid-row td' : {
60730                                 'vertical-align' : 0
60731                             },
60732                             '.course-edit-link' : {
60733                                 'color' : 'blue',
60734                                 'text-overflow' : 'ellipsis',
60735                                 'overflow' : 'hidden',
60736                                 'white-space' : 'nowrap',
60737                                 'cursor' : 'pointer'
60738                             },
60739                             '.sub-link' : {
60740                                 'color' : 'green'
60741                             },
60742                             '.de-act-sup-link' : {
60743                                 'color' : 'purple',
60744                                 'text-decoration' : 'line-through'
60745                             },
60746                             '.de-act-link' : {
60747                                 'color' : 'red',
60748                                 'text-decoration' : 'line-through'
60749                             },
60750                             '.course-timesheet .course-highlight' : {
60751                                 'border-top-style': 'dashed !important',
60752                                 'border-bottom-bottom': 'dashed !important'
60753                             },
60754                             '.course-timesheet .course-item' : {
60755                                 'font-family'   : 'tahoma, arial, helvetica',
60756                                 'font-size'     : '11px',
60757                                 'overflow'      : 'hidden',
60758                                 'padding-left'  : '10px',
60759                                 'padding-right' : '10px',
60760                                 'padding-top' : '10px' 
60761                             }
60762                             
60763                         }, Roo.id());
60764                                 this.ds.load({});
60765                     }
60766                 },
60767                 autoWidth : true,
60768                 monitorWindowResize : false,
60769                 cellrenderer : function(v,x,r)
60770                 {
60771                     return v;
60772                 },
60773                 sm : {
60774                     xtype: 'CellSelectionModel',
60775                     xns: Roo.grid
60776                 },
60777                 dataSource : {
60778                     xtype: 'Store',
60779                     xns: Roo.data,
60780                     listeners : {
60781                         beforeload : function (_self, options)
60782                         {
60783                             options.params = options.params || {};
60784                             options.params._month = _this.monthField.getValue();
60785                             options.params.limit = 9999;
60786                             options.params['sort'] = 'when_dt';    
60787                             options.params['dir'] = 'ASC';    
60788                             this.proxy.loadResponse = this.loadResponse;
60789                             Roo.log("load?");
60790                             //this.addColumns();
60791                         },
60792                         load : function (_self, records, options)
60793                         {
60794                             _this.grid.view.el.select('.course-edit-link', true).on('click', function() {
60795                                 // if you click on the translation.. you can edit it...
60796                                 var el = Roo.get(this);
60797                                 var id = el.dom.getAttribute('data-id');
60798                                 var d = el.dom.getAttribute('data-date');
60799                                 var t = el.dom.getAttribute('data-time');
60800                                 //var id = this.child('span').dom.textContent;
60801                                 
60802                                 //Roo.log(this);
60803                                 Pman.Dialog.CourseCalendar.show({
60804                                     id : id,
60805                                     when_d : d,
60806                                     when_t : t,
60807                                     productitem_active : id ? 1 : 0
60808                                 }, function() {
60809                                     _this.grid.ds.load({});
60810                                 });
60811                            
60812                            });
60813                            
60814                            _this.panel.fireEvent('resize', [ '', '' ]);
60815                         }
60816                     },
60817                     loadResponse : function(o, success, response){
60818                             // this is overridden on before load..
60819                             
60820                             Roo.log("our code?");       
60821                             //Roo.log(success);
60822                             //Roo.log(response)
60823                             delete this.activeRequest;
60824                             if(!success){
60825                                 this.fireEvent("loadexception", this, o, response);
60826                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60827                                 return;
60828                             }
60829                             var result;
60830                             try {
60831                                 result = o.reader.read(response);
60832                             }catch(e){
60833                                 Roo.log("load exception?");
60834                                 this.fireEvent("loadexception", this, o, response, e);
60835                                 o.request.callback.call(o.request.scope, null, o.request.arg, false);
60836                                 return;
60837                             }
60838                             Roo.log("ready...");        
60839                             // loop through result.records;
60840                             // and set this.tdate[date] = [] << array of records..
60841                             _this.tdata  = {};
60842                             Roo.each(result.records, function(r){
60843                                 //Roo.log(r.data);
60844                                 if(typeof(_this.tdata[r.data.when_dt.format('j')]) == 'undefined'){
60845                                     _this.tdata[r.data.when_dt.format('j')] = [];
60846                                 }
60847                                 _this.tdata[r.data.when_dt.format('j')].push(r.data);
60848                             });
60849                             
60850                             //Roo.log(_this.tdata);
60851                             
60852                             result.records = [];
60853                             result.totalRecords = 6;
60854                     
60855                             // let's generate some duumy records for the rows.
60856                             //var st = _this.dateField.getValue();
60857                             
60858                             // work out monday..
60859                             //st = st.add(Date.DAY, -1 * st.format('w'));
60860                             
60861                             var date = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60862                             
60863                             var firstOfMonth = date.getFirstDayOfMonth();
60864                             var days = date.getDaysInMonth();
60865                             var d = 1;
60866                             var firstAdded = false;
60867                             for (var i = 0; i < result.totalRecords ; i++) {
60868                                 //var d= st.add(Date.DAY, i);
60869                                 var row = {};
60870                                 var added = 0;
60871                                 for(var w = 0 ; w < 7 ; w++){
60872                                     if(!firstAdded && firstOfMonth != w){
60873                                         continue;
60874                                     }
60875                                     if(d > days){
60876                                         continue;
60877                                     }
60878                                     firstAdded = true;
60879                                     var dd = (d > 0 && d < 10) ? "0"+d : d;
60880                                     row['weekday'+w] = String.format(
60881                                                     '<span style="font-size: 16px;"><b>{0}</b></span>'+
60882                                                     '<span class="course-edit-link" style="color:blue;" data-id="0" data-date="{1}"> Add New</span>',
60883                                                     d,
60884                                                     date.format('Y-m-')+dd
60885                                                 );
60886                                     added++;
60887                                     if(typeof(_this.tdata[d]) != 'undefined'){
60888                                         Roo.each(_this.tdata[d], function(r){
60889                                             var is_sub = '';
60890                                             var deactive = '';
60891                                             var id = r.id;
60892                                             var desc = (r.productitem_id_descrip) ? r.productitem_id_descrip : '';
60893                                             if(r.parent_id*1>0){
60894                                                 is_sub = (r.productitem_id_visible*1 < 1) ? 'de-act-sup-link' :'sub-link';
60895                                                 id = r.parent_id;
60896                                             }
60897                                             if(r.productitem_id_visible*1 < 1 && r.parent_id*1 < 1){
60898                                                 deactive = 'de-act-link';
60899                                             }
60900                                             
60901                                             row['weekday'+w] += String.format(
60902                                                     '<br /><span class="course-edit-link {3} {4}" qtip="{5}" data-id="{0}">{2} - {1}</span>',
60903                                                     id, //0
60904                                                     r.product_id_name, //1
60905                                                     r.when_dt.format('h:ia'), //2
60906                                                     is_sub, //3
60907                                                     deactive, //4
60908                                                     desc // 5
60909                                             );
60910                                         });
60911                                     }
60912                                     d++;
60913                                 }
60914                                 
60915                                 // only do this if something added..
60916                                 if(added > 0){ 
60917                                     result.records.push(_this.grid.dataSource.reader.newRow(row));
60918                                 }
60919                                 
60920                                 
60921                                 // push it twice. (second one with an hour..
60922                                 
60923                             }
60924                             //Roo.log(result);
60925                             this.fireEvent("load", this, o, o.request.arg);
60926                             o.request.callback.call(o.request.scope, result, o.request.arg, true);
60927                         },
60928                     sortInfo : {field: 'when_dt', direction : 'ASC' },
60929                     proxy : {
60930                         xtype: 'HttpProxy',
60931                         xns: Roo.data,
60932                         method : 'GET',
60933                         url : baseURL + '/Roo/Shop_course.php'
60934                     },
60935                     reader : {
60936                         xtype: 'JsonReader',
60937                         xns: Roo.data,
60938                         id : 'id',
60939                         fields : [
60940                             {
60941                                 'name': 'id',
60942                                 'type': 'int'
60943                             },
60944                             {
60945                                 'name': 'when_dt',
60946                                 'type': 'string'
60947                             },
60948                             {
60949                                 'name': 'end_dt',
60950                                 'type': 'string'
60951                             },
60952                             {
60953                                 'name': 'parent_id',
60954                                 'type': 'int'
60955                             },
60956                             {
60957                                 'name': 'product_id',
60958                                 'type': 'int'
60959                             },
60960                             {
60961                                 'name': 'productitem_id',
60962                                 'type': 'int'
60963                             },
60964                             {
60965                                 'name': 'guid',
60966                                 'type': 'int'
60967                             }
60968                         ]
60969                     }
60970                 },
60971                 toolbar : {
60972                     xtype: 'Toolbar',
60973                     xns: Roo,
60974                     items : [
60975                         {
60976                             xtype: 'Button',
60977                             xns: Roo.Toolbar,
60978                             listeners : {
60979                                 click : function (_self, e)
60980                                 {
60981                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
60982                                     sd.setMonth(sd.getMonth()-1);
60983                                     _this.monthField.setValue(sd.format('Y-m-d'));
60984                                     _this.grid.ds.load({});
60985                                 }
60986                             },
60987                             text : "Back"
60988                         },
60989                         {
60990                             xtype: 'Separator',
60991                             xns: Roo.Toolbar
60992                         },
60993                         {
60994                             xtype: 'MonthField',
60995                             xns: Roo.form,
60996                             listeners : {
60997                                 render : function (_self)
60998                                 {
60999                                     _this.monthField = _self;
61000                                    // _this.monthField.set  today
61001                                 },
61002                                 select : function (combo, date)
61003                                 {
61004                                     _this.grid.ds.load({});
61005                                 }
61006                             },
61007                             value : (function() { return new Date(); })()
61008                         },
61009                         {
61010                             xtype: 'Separator',
61011                             xns: Roo.Toolbar
61012                         },
61013                         {
61014                             xtype: 'TextItem',
61015                             xns: Roo.Toolbar,
61016                             text : "Blue: in-active, green: in-active sup-event, red: de-active, purple: de-active sup-event"
61017                         },
61018                         {
61019                             xtype: 'Fill',
61020                             xns: Roo.Toolbar
61021                         },
61022                         {
61023                             xtype: 'Button',
61024                             xns: Roo.Toolbar,
61025                             listeners : {
61026                                 click : function (_self, e)
61027                                 {
61028                                     var sd = Date.parseDate(_this.monthField.getValue(), "Y-m-d");
61029                                     sd.setMonth(sd.getMonth()+1);
61030                                     _this.monthField.setValue(sd.format('Y-m-d'));
61031                                     _this.grid.ds.load({});
61032                                 }
61033                             },
61034                             text : "Next"
61035                         }
61036                     ]
61037                 },
61038                  
61039             }
61040         };
61041         
61042         *//*
61043  * Based on:
61044  * Ext JS Library 1.1.1
61045  * Copyright(c) 2006-2007, Ext JS, LLC.
61046  *
61047  * Originally Released Under LGPL - original licence link has changed is not relivant.
61048  *
61049  * Fork - LGPL
61050  * <script type="text/javascript">
61051  */
61052  
61053 /**
61054  * @class Roo.LoadMask
61055  * A simple utility class for generically masking elements while loading data.  If the element being masked has
61056  * an underlying {@link Roo.data.Store}, the masking will be automatically synchronized with the store's loading
61057  * process and the mask element will be cached for reuse.  For all other elements, this mask will replace the
61058  * element's UpdateManager load indicator and will be destroyed after the initial load.
61059  * @constructor
61060  * Create a new LoadMask
61061  * @param {String/HTMLElement/Roo.Element} el The element or DOM node, or its id
61062  * @param {Object} config The config object
61063  */
61064 Roo.LoadMask = function(el, config){
61065     this.el = Roo.get(el);
61066     Roo.apply(this, config);
61067     if(this.store){
61068         this.store.on('beforeload', this.onBeforeLoad, this);
61069         this.store.on('load', this.onLoad, this);
61070         this.store.on('loadexception', this.onLoadException, this);
61071         this.removeMask = false;
61072     }else{
61073         var um = this.el.getUpdateManager();
61074         um.showLoadIndicator = false; // disable the default indicator
61075         um.on('beforeupdate', this.onBeforeLoad, this);
61076         um.on('update', this.onLoad, this);
61077         um.on('failure', this.onLoad, this);
61078         this.removeMask = true;
61079     }
61080 };
61081
61082 Roo.LoadMask.prototype = {
61083     /**
61084      * @cfg {Boolean} removeMask
61085      * True to create a single-use mask that is automatically destroyed after loading (useful for page loads),
61086      * False to persist the mask element reference for multiple uses (e.g., for paged data widgets).  Defaults to false.
61087      */
61088     /**
61089      * @cfg {String} msg
61090      * The text to display in a centered loading message box (defaults to 'Loading...')
61091      */
61092     msg : 'Loading...',
61093     /**
61094      * @cfg {String} msgCls
61095      * The CSS class to apply to the loading message element (defaults to "x-mask-loading")
61096      */
61097     msgCls : 'x-mask-loading',
61098
61099     /**
61100      * Read-only. True if the mask is currently disabled so that it will not be displayed (defaults to false)
61101      * @type Boolean
61102      */
61103     disabled: false,
61104
61105     /**
61106      * Disables the mask to prevent it from being displayed
61107      */
61108     disable : function(){
61109        this.disabled = true;
61110     },
61111
61112     /**
61113      * Enables the mask so that it can be displayed
61114      */
61115     enable : function(){
61116         this.disabled = false;
61117     },
61118     
61119     onLoadException : function()
61120     {
61121         Roo.log(arguments);
61122         
61123         if (typeof(arguments[3]) != 'undefined') {
61124             Roo.MessageBox.alert("Error loading",arguments[3]);
61125         } 
61126         /*
61127         try {
61128             if (this.store && typeof(this.store.reader.jsonData.errorMsg) != 'undefined') {
61129                 Roo.MessageBox.alert("Error loading",this.store.reader.jsonData.errorMsg);
61130             }   
61131         } catch(e) {
61132             
61133         }
61134         */
61135     
61136         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61137     },
61138     // private
61139     onLoad : function()
61140     {
61141         (function() { this.el.unmask(this.removeMask); }).defer(50, this);
61142     },
61143
61144     // private
61145     onBeforeLoad : function(){
61146         if(!this.disabled){
61147             (function() { this.el.mask(this.msg, this.msgCls); }).defer(50, this);
61148         }
61149     },
61150
61151     // private
61152     destroy : function(){
61153         if(this.store){
61154             this.store.un('beforeload', this.onBeforeLoad, this);
61155             this.store.un('load', this.onLoad, this);
61156             this.store.un('loadexception', this.onLoadException, this);
61157         }else{
61158             var um = this.el.getUpdateManager();
61159             um.un('beforeupdate', this.onBeforeLoad, this);
61160             um.un('update', this.onLoad, this);
61161             um.un('failure', this.onLoad, this);
61162         }
61163     }
61164 };/*
61165  * Based on:
61166  * Ext JS Library 1.1.1
61167  * Copyright(c) 2006-2007, Ext JS, LLC.
61168  *
61169  * Originally Released Under LGPL - original licence link has changed is not relivant.
61170  *
61171  * Fork - LGPL
61172  * <script type="text/javascript">
61173  */
61174
61175
61176 /**
61177  * @class Roo.XTemplate
61178  * @extends Roo.Template
61179  * Provides a template that can have nested templates for loops or conditionals. The syntax is:
61180 <pre><code>
61181 var t = new Roo.XTemplate(
61182         '&lt;select name="{name}"&gt;',
61183                 '&lt;tpl for="options"&gt;&lt;option value="{value:trim}"&gt;{text:ellipsis(10)}&lt;/option&gt;&lt;/tpl&gt;',
61184         '&lt;/select&gt;'
61185 );
61186  
61187 // then append, applying the master template values
61188  </code></pre>
61189  *
61190  * Supported features:
61191  *
61192  *  Tags:
61193
61194 <pre><code>
61195       {a_variable} - output encoded.
61196       {a_variable.format:("Y-m-d")} - call a method on the variable
61197       {a_variable:raw} - unencoded output
61198       {a_variable:toFixed(1,2)} - Roo.util.Format."toFixed"
61199       {a_variable:this.method_on_template(...)} - call a method on the template object.
61200  
61201 </code></pre>
61202  *  The tpl tag:
61203 <pre><code>
61204         &lt;tpl for="a_variable or condition.."&gt;&lt;/tpl&gt;
61205         &lt;tpl if="a_variable or condition"&gt;&lt;/tpl&gt;
61206         &lt;tpl exec="some javascript"&gt;&lt;/tpl&gt;
61207         &lt;tpl name="named_template"&gt;&lt;/tpl&gt; (experimental)
61208   
61209         &lt;tpl for="."&gt;&lt;/tpl&gt; - just iterate the property..
61210         &lt;tpl for=".."&gt;&lt;/tpl&gt; - iterates with the parent (probably the template) 
61211 </code></pre>
61212  *      
61213  */
61214 Roo.XTemplate = function()
61215 {
61216     Roo.XTemplate.superclass.constructor.apply(this, arguments);
61217     if (this.html) {
61218         this.compile();
61219     }
61220 };
61221
61222
61223 Roo.extend(Roo.XTemplate, Roo.Template, {
61224
61225     /**
61226      * The various sub templates
61227      */
61228     tpls : false,
61229     /**
61230      *
61231      * basic tag replacing syntax
61232      * WORD:WORD()
61233      *
61234      * // you can fake an object call by doing this
61235      *  x.t:(test,tesT) 
61236      * 
61237      */
61238     re : /\{([\w-\.]+)(?:\:([\w\.]*)(?:\((.*?)?\))?)?\}/g,
61239
61240     /**
61241      * compile the template
61242      *
61243      * This is not recursive, so I'm not sure how nested templates are really going to be handled..
61244      *
61245      */
61246     compile: function()
61247     {
61248         var s = this.html;
61249      
61250         s = ['<tpl>', s, '</tpl>'].join('');
61251     
61252         var re     = /<tpl\b[^>]*>((?:(?=([^<]+))\2|<(?!tpl\b[^>]*>))*?)<\/tpl>/,
61253             nameRe = /^<tpl\b[^>]*?for="(.*?)"/,
61254             ifRe   = /^<tpl\b[^>]*?if="(.*?)"/,
61255             execRe = /^<tpl\b[^>]*?exec="(.*?)"/,
61256             namedRe = /^<tpl\b[^>]*?name="(\w+)"/,  // named templates..
61257             m,
61258             id     = 0,
61259             tpls   = [];
61260     
61261         while(true == !!(m = s.match(re))){
61262             var forMatch   = m[0].match(nameRe),
61263                 ifMatch   = m[0].match(ifRe),
61264                 execMatch   = m[0].match(execRe),
61265                 namedMatch   = m[0].match(namedRe),
61266                 
61267                 exp  = null, 
61268                 fn   = null,
61269                 exec = null,
61270                 name = forMatch && forMatch[1] ? forMatch[1] : '';
61271                 
61272             if (ifMatch) {
61273                 // if - puts fn into test..
61274                 exp = ifMatch && ifMatch[1] ? ifMatch[1] : null;
61275                 if(exp){
61276                    fn = new Function('values', 'parent', 'with(values){ return '+(Roo.util.Format.htmlDecode(exp))+'; }');
61277                 }
61278             }
61279             
61280             if (execMatch) {
61281                 // exec - calls a function... returns empty if true is  returned.
61282                 exp = execMatch && execMatch[1] ? execMatch[1] : null;
61283                 if(exp){
61284                    exec = new Function('values', 'parent', 'with(values){ '+(Roo.util.Format.htmlDecode(exp))+'; }');
61285                 }
61286             }
61287             
61288             
61289             if (name) {
61290                 // for = 
61291                 switch(name){
61292                     case '.':  name = new Function('values', 'parent', 'with(values){ return values; }'); break;
61293                     case '..': name = new Function('values', 'parent', 'with(values){ return parent; }'); break;
61294                     default:   name = new Function('values', 'parent', 'with(values){ return '+name+'; }');
61295                 }
61296             }
61297             var uid = namedMatch ? namedMatch[1] : id;
61298             
61299             
61300             tpls.push({
61301                 id:     namedMatch ? namedMatch[1] : id,
61302                 target: name,
61303                 exec:   exec,
61304                 test:   fn,
61305                 body:   m[1] || ''
61306             });
61307             if (namedMatch) {
61308                 s = s.replace(m[0], '');
61309             } else { 
61310                 s = s.replace(m[0], '{xtpl'+ id + '}');
61311             }
61312             ++id;
61313         }
61314         this.tpls = [];
61315         for(var i = tpls.length-1; i >= 0; --i){
61316             this.compileTpl(tpls[i]);
61317             this.tpls[tpls[i].id] = tpls[i];
61318         }
61319         this.master = tpls[tpls.length-1];
61320         return this;
61321     },
61322     /**
61323      * same as applyTemplate, except it's done to one of the subTemplates
61324      * when using named templates, you can do:
61325      *
61326      * var str = pl.applySubTemplate('your-name', values);
61327      *
61328      * 
61329      * @param {Number} id of the template
61330      * @param {Object} values to apply to template
61331      * @param {Object} parent (normaly the instance of this object)
61332      */
61333     applySubTemplate : function(id, values, parent)
61334     {
61335         
61336         
61337         var t = this.tpls[id];
61338         
61339         
61340         try { 
61341             if(t.test && !t.test.call(this, values, parent)){
61342                 return '';
61343             }
61344         } catch(e) {
61345             Roo.log("Xtemplate.applySubTemplate 'test': Exception thrown");
61346             Roo.log(e.toString());
61347             Roo.log(t.test);
61348             return ''
61349         }
61350         try { 
61351             
61352             if(t.exec && t.exec.call(this, values, parent)){
61353                 return '';
61354             }
61355         } catch(e) {
61356             Roo.log("Xtemplate.applySubTemplate 'exec': Exception thrown");
61357             Roo.log(e.toString());
61358             Roo.log(t.exec);
61359             return ''
61360         }
61361         try {
61362             var vs = t.target ? t.target.call(this, values, parent) : values;
61363             parent = t.target ? values : parent;
61364             if(t.target && vs instanceof Array){
61365                 var buf = [];
61366                 for(var i = 0, len = vs.length; i < len; i++){
61367                     buf[buf.length] = t.compiled.call(this, vs[i], parent);
61368                 }
61369                 return buf.join('');
61370             }
61371             return t.compiled.call(this, vs, parent);
61372         } catch (e) {
61373             Roo.log("Xtemplate.applySubTemplate : Exception thrown");
61374             Roo.log(e.toString());
61375             Roo.log(t.compiled);
61376             return '';
61377         }
61378     },
61379
61380     compileTpl : function(tpl)
61381     {
61382         var fm = Roo.util.Format;
61383         var useF = this.disableFormats !== true;
61384         var sep = Roo.isGecko ? "+" : ",";
61385         var undef = function(str) {
61386             Roo.log("Property not found :"  + str);
61387             return '';
61388         };
61389         
61390         var fn = function(m, name, format, args)
61391         {
61392             //Roo.log(arguments);
61393             args = args ? args.replace(/\\'/g,"'") : args;
61394             //["{TEST:(a,b,c)}", "TEST", "", "a,b,c", 0, "{TEST:(a,b,c)}"]
61395             if (typeof(format) == 'undefined') {
61396                 format= 'htmlEncode';
61397             }
61398             if (format == 'raw' ) {
61399                 format = false;
61400             }
61401             
61402             if(name.substr(0, 4) == 'xtpl'){
61403                 return "'"+ sep +'this.applySubTemplate('+name.substr(4)+', values, parent)'+sep+"'";
61404             }
61405             
61406             // build an array of options to determine if value is undefined..
61407             
61408             // basically get 'xxxx.yyyy' then do
61409             // (typeof(xxxx) == 'undefined' || typeof(xxx.yyyy) == 'undefined') ?
61410             //    (function () { Roo.log("Property not found"); return ''; })() :
61411             //    ......
61412             
61413             var udef_ar = [];
61414             var lookfor = '';
61415             Roo.each(name.split('.'), function(st) {
61416                 lookfor += (lookfor.length ? '.': '') + st;
61417                 udef_ar.push(  "(typeof(" + lookfor + ") == 'undefined')"  );
61418             });
61419             
61420             var udef_st = '((' + udef_ar.join(" || ") +") ? undef('" + name + "') : "; // .. needs )
61421             
61422             
61423             if(format && useF){
61424                 
61425                 args = args ? ',' + args : "";
61426                  
61427                 if(format.substr(0, 5) != "this."){
61428                     format = "fm." + format + '(';
61429                 }else{
61430                     format = 'this.call("'+ format.substr(5) + '", ';
61431                     args = ", values";
61432                 }
61433                 
61434                 return "'"+ sep +   udef_st   +    format + name + args + "))"+sep+"'";
61435             }
61436              
61437             if (args.length) {
61438                 // called with xxyx.yuu:(test,test)
61439                 // change to ()
61440                 return "'"+ sep + udef_st  + name + '(' +  args + "))"+sep+"'";
61441             }
61442             // raw.. - :raw modifier..
61443             return "'"+ sep + udef_st  + name + ")"+sep+"'";
61444             
61445         };
61446         var body;
61447         // branched to use + in gecko and [].join() in others
61448         if(Roo.isGecko){
61449             body = "tpl.compiled = function(values, parent){  with(values) { return '" +
61450                    tpl.body.replace(/(\r\n|\n)/g, '\\n').replace(/'/g, "\\'").replace(this.re, fn) +
61451                     "';};};";
61452         }else{
61453             body = ["tpl.compiled = function(values, parent){  with (values) { return ['"];
61454             body.push(tpl.body.replace(/(\r\n|\n)/g,
61455                             '\\n').replace(/'/g, "\\'").replace(this.re, fn));
61456             body.push("'].join('');};};");
61457             body = body.join('');
61458         }
61459         
61460         Roo.debug && Roo.log(body.replace(/\\n/,'\n'));
61461        
61462         /** eval:var:tpl eval:var:fm eval:var:useF eval:var:undef  */
61463         eval(body);
61464         
61465         return this;
61466     },
61467
61468     applyTemplate : function(values){
61469         return this.master.compiled.call(this, values, {});
61470         //var s = this.subs;
61471     },
61472
61473     apply : function(){
61474         return this.applyTemplate.apply(this, arguments);
61475     }
61476
61477  });
61478
61479 Roo.XTemplate.from = function(el){
61480     el = Roo.getDom(el);
61481     return new Roo.XTemplate(el.value || el.innerHTML);
61482 };